CanvasRefined/js/popup.js
Guy Sandler 03c61eb51d background changer 1
it works too well and too many things are transparent
2026-02-09 14:52:45 -08:00

1965 lines
89 KiB
JavaScript

const syncedSwitches = ['remind', 'tab_icons', 'hide_feedback', 'dark_mode', 'remlogo', 'full_width', 'auto_dark', 'assignments_due', 'gpa_calc', 'gradient_cards', 'disable_color_overlay', 'dashboard_grades', 'dashboard_notes', 'better_todo', 'condensed_cards'];
const syncedSubOptions = [
"todo_colors",
"device_dark",
"relative_dues",
"card_overdues",
"todo_overdues",
"gpa_calc_prepend",
"auto_dark",
"auto_dark_start",
"auto_dark_end",
"num_assignments",
"assignment_date_format",
"todo_hr24",
"grade_hover",
"hide_completed",
"num_todo_items",
"hover_preview",
"scheduledReminder",
"scheduledReminderTime",
"customCardStyles",
"imageSize",
"cardRoundness",
"cardSpacing",
"cardWidth",
"cardHeight",
];
const localSwitches = [];
//const apiurl = "http://localhost:3000";
const apiurl = "https://bettercanvas.diditupe.dev";
const defaultOptions = {
"local": {
"previous_colors": null,
"previous_theme": null,
"errors": [],
},
"sync": {
"dark_preset": {
"background-0": "#161616",
"background-1": "#1e1e1e",
"background-2": "#262626",
"borders": "#3c3c3c",
"text-0": "#f5f5f5",
"text-1": "#e2e2e2",
"text-2": "#ababab",
"links": "#56Caf0",
"sidebar": "#1e1e1e",
"sidebar-text": "#f5f5f5"
},
"new_install": true,
"assignments_due": true,
"gpa_calc": false,
"dark_mode": true,
"gradent_cards": false,
"disable_color_overlay": false,
"auto_dark": false,
"auto_dark_start": { "hour": "20", "minute": "00" },
"auto_dark_end": { "hour": "08", "minute": "00" },
"num_assignments": 4,
"custom_domain": [""],
"assignments_done": [],
"dashboard_grades": false,
"assignment_date_format": false,
"dashboard_notes": false,
"dashboard_notes_text": "",
"better_todo": false,
"todo_hr24": false,
"condensed_cards": false,
"custom_cards": {},
"custom_cards_2": {},
"custom_cards_3": {},
"custom_assignments": [],
"custom_assignments_overflow": ["custom_assignments"],
"grade_hover": false,
"hide_completed": false,
"num_todo_items": 4,
"custom_font": { "link": "", "family": "" },
"hover_preview": true,
"full_width": null,
"remlogo": null,
"gpa_calc_bounds": {
"A+": { "cutoff": 97, "gpa": 4.3 },
"A": { "cutoff": 93, "gpa": 4 },
"A-": { "cutoff": 90, "gpa": 3.7 },
"B+": { "cutoff": 87, "gpa": 3.3 },
"B": { "cutoff": 83, "gpa": 3 },
"B-": { "cutoff": 80, "gpa": 2.7 },
"C+": { "cutoff": 77, "gpa": 2.3 },
"C": { "cutoff": 73, "gpa": 2 },
"C-": { "cutoff": 70, "gpa": 1.7 },
"D+": { "cutoff": 67, "gpa": 1.3 },
"D": { "cutoff": 63, "gpa": 1 },
"D-": { "cutoff": 60, "gpa": .7 },
"F": { "cutoff": 0, "gpa": 0 }
},
"todo_overdues": false,
"card_overdues": false,
"relative_dues": false,
"hide_feedback": false,
"dark_mode_fix": [],
"assignment_states": {},
"tab_icons": false,
"todo_colors": false,
"device_dark": false,
"cumulative_gpa": { "name": "Cumulative GPA", "hidden": false, "weight": "dnc", "credits": 999, "gr": 3.21 },
"show_updates": false,
"card_method_date": false,
"card_method_dashboard": false,
"card_limit": 25,
"scheduledReminder": false,
"scheduledReminderTime": { "hour": "09", "minute": "00" },
"imageSize": 100,
"cardRoundness": 5,
"cardSpacing": 0,
"cardWidth": 262,
"cardHeight": 250,
"customCardStyles": false,
"customBackgroundLink": "",
"customBackgroundUpload": "",
}
};
sendFromPopup("getCards");
// refresh the cards if new ones were just recieved
chrome.storage.onChanged.addListener((changes) => {
if (changes["custom_cards"]) {
if (Object.keys(changes["custom_cards"].oldValue).length !== Object.keys(changes["custom_cards"].newValue).length) {
displayAdvancedCards();
}
}
});
function displayErrors() {
chrome.storage.local.get("errors", storage => {
storage["errors"].forEach(e => {
document.querySelector("#error_log_output").value += (e + "\n\n");
})
});
}
function displayDarkModeFixUrls() {
let output = document.getElementById("dark-mode-fix-urls");
output.textContent = "";
chrome.storage.sync.get("dark_mode_fix", sync => {
sync["dark_mode_fix"].forEach(url => {
//let div = makeElement("div", "customization-button", output, url);
let div = makeElement("div", output, { "className": "customization-button", "textContent": url });
div.classList.add("fixed-url");
let btn = makeElement("button", div, { "className": "dd", "textContent": "x" });
btn.addEventListener("click", () => {
chrome.storage.sync.get("dark_mode_fix", sync => {
for (let i = 0; i < sync["dark_mode_fix"].length; i++) {
if (sync["dark_mode_fix"][i] === url) {
sync["dark_mode_fix"].splice(i);
chrome.storage.sync.set({ "dark_mode_fix": sync["dark_mode_fix"] }).then(() => div.remove());
}
}
});
})
})
})
}
document.addEventListener("DOMContentLoaded", setup);
function setupAssignmentsSlider(initial) {
let el = document.querySelector('#numAssignmentsSlider');
el.value = initial;
document.querySelector('#numAssignments').textContent = initial;
el.addEventListener('input', function () {
document.querySelector('#numAssignments').textContent = this.value;
chrome.storage.sync.set({ "num_assignments": this.value });
});
}
function setupTodoSlider(initial) {
let el = document.querySelector('#numTodoItemsSlider');
el.value = initial;
document.querySelector('#numTodoItems').textContent = initial;
document.querySelector('#numTodoItemsSlider').addEventListener('input', function () {
document.querySelector('#numTodoItems').textContent = this.value;
chrome.storage.sync.set({ "num_todo_items": this.value });
});
}
function setupAutoDarkInput(initial, time) {
let el = document.querySelector('#' + time);
el.value = initial.hour + ":" + initial.minute;
el.addEventListener('change', function () {
let timeinput = { "hour": this.value.split(':')[0], "minute": this.value.split(':')[1] };
time === "auto_dark_start" ? chrome.storage.sync.set({ auto_dark_start: timeinput }) : chrome.storage.sync.set({ auto_dark_end: timeinput });
});
}
function setupScheduledReminderInput(initial) {
let el = document.querySelector('#scheduledReminderTime');
el.value = initial.hour + ":" + initial.minute;
el.addEventListener('change', function () {
let timeinput = { "hour": this.value.split(':')[0], "minute": this.value.split(':')[1] };
chrome.storage.sync.set({ scheduledReminderTime: timeinput });
});
}
function setupCardLimitSlider(initial) {
let el = document.querySelector("#card_limit");
el.value = initial;
document.querySelector("#card_limit_num").textContent = initial;
el.addEventListener("change", (e) => {
chrome.storage.sync.set({ "custom_cards": {}, "custom_cards_2": {}, "custom_cards_3": {}, "card_limit": parseInt(e.target.value)});
});
el.addEventListener("input", (e) => {
document.querySelector("#card_limit_num").textContent = e.target.value;
})
}
function setupDashboardMethod(initial) {
const el = document.getElementById("card_method_dashboard");
el.checked = initial === true ? true : false;
el.addEventListener("change", (e) => {
chrome.storage.sync.set({ "custom_cards": {}, "custom_cards_2": {}, "custom_cards_3": {}, "card_method_dashboard": e.target.checked });
});
}
function setupImageSizeInput(initial) {
let el = document.querySelector("#imageSize");
el.value = initial;
el.addEventListener("input", (e) => {
chrome.storage.sync.set({ "imageSize": e.target.value });
});
}
function setupCardRoundnessInput(initial) {
let el = document.querySelector("#cardRoundness");
el.value = initial;
el.addEventListener("input", (e) => {
chrome.storage.sync.set({ "cardRoundness": e.target.value });
});
}
function setupCardSpacingInput(initial) {
let el = document.querySelector("#cardSpacing");
el.value = initial;
el.addEventListener("input", (e) => {
chrome.storage.sync.set({ "cardSpacing": e.target.value });
});
}
function setupCardWidthInput(initial) {
let el = document.querySelector("#cardWidth");
el.value = initial;
el.addEventListener("input", (e) => {
chrome.storage.sync.set({ "cardWidth": e.target.value });
});
}
function setupCardHeightInput(initial) {
let el = document.querySelector("#cardHeight");
el.value = initial;
el.addEventListener("input", (e) => {
chrome.storage.sync.set({ "cardHeight": e.target.value });
});
}
function setupCustomBackgroundLink(initial) {
let el = document.querySelector("#customBackgroundLink");
el.value = initial || "";
el.addEventListener("input", (e) => {
chrome.storage.sync.set({ "customBackgroundLink": e.target.value });
})
}
function setupCustomBackgroundUpload(initial) {
let el = document.querySelector("#customBackgroundUpload");
el.addEventListener("change", handleBackgroundUpload);
}
function handleBackgroundUpload(e) {
const file = e.target.files[0];
if (file) {
const reader = new FileReader();
reader.onload = function(event) {
const base64 = event.target.result;
chrome.storage.sync.set({
"customBackgroundUpload": base64,
"customBackgroundLink": ""
});
document.querySelector("#customBackgroundLink").value = "Using uploaded file";
}
reader.readAsDataURL(file);
}
}
function setup() {
const menu = {
"switches": syncedSwitches,
"checkboxes": ['browser_show_likes', 'gpa_calc_weighted', 'gpa_calc_cumulative', /*'card_method_date',*/ 'show_updates', 'todo_colors', 'device_dark', 'relative_dues', 'card_overdues', 'todo_overdues', 'gpa_calc_prepend', 'auto_dark', 'assignment_date_format', 'todo_hr24', 'grade_hover', 'hide_completed', 'hover_preview', 'scheduledReminder', 'customCardStyles'],
"tabs": {
"advanced-settings": { "setup": displayAdvancedCards, "tab": ".advanced" },
"gpa-bounds-btn": { "setup": displayGPABounds, "tab": ".gpa-bounds-container" },
"custom-font-btn": { "setup": displayCustomFont, "tab": ".custom-font-container" },
"card-colors-btn": { "setup": null, "tab": ".card-colors-container" },
"customize-dark-btn": { "setup": displayDarkModeFixUrls, "tab": ".customize-dark" },
"import-export-btn": { "setup": displayThemeList, "tab": ".import-export" },
"report-issue-btn": { "setup": displayErrors, "tab": ".report-issue-container" },
"updates-btn": { "setup": null, "tab": ".updates-container" }
},
"special": [
{ "identifier": "auto_dark_start", "setup": (initial) => setupAutoDarkInput(initial, "auto_dark_start") },
{ "identifier": "auto_dark_end", "setup": (initial) => setupAutoDarkInput(initial, "auto_dark_end") },
{ "identifier": "num_assignments", "setup": (initial) => setupAssignmentsSlider(initial) },
{ "identifier": "num_todo_items", "setup": (initial) => setupTodoSlider(initial) },
{ "identifier": "card_limit", "setup": (initial) => setupCardLimitSlider(initial) },
{ "identifier": "card_method_dashboard", "setup": (initial) => setupDashboardMethod(initial) },
{ "identifier": "custom_styles", "setup": (initial) => setupCustomStyle(initial) },
{ "identifier": "scheduledReminderTime", "setup": (initial) => setupScheduledReminderInput(initial) },
{ "identifier": "imageSize", "setup": (initial) => setupImageSizeInput(initial) },
{ "identifier": "cardRoundness", "setup": (initial) => setupCardRoundnessInput(initial) },
{ "identifier": "cardSpacing", "setup": (initial) => setupCardSpacingInput(initial) },
{ "identifier": "cardWidth", "setup": (initial) => setupCardWidthInput(initial) },
{ "identifier": "cardHeight", "setup": (initial) => setupCardHeightInput(initial) },
{ "identifier": "customBackgroundLink", "setup": (initial) => setupCustomBackgroundLink(initial)},
{ "identifier": "customBackgroundUpload", "setup": (initial) => setupCustomBackgroundUpload(initial)},
],
}
chrome.storage.sync.get(menu.switches, sync => {
menu.switches.forEach(option => {
let optionSwitch = document.getElementById(option);
let status = sync[option] === true ? "#on" : "#off";
optionSwitch.querySelector(status).checked = true;
optionSwitch.querySelector(status).classList.add('checked');
optionSwitch.querySelector(".slider").addEventListener("mouseup", () => {
let status = !optionSwitch.querySelector("#on").checked;
optionSwitch.querySelector("#on").checked = status;
optionSwitch.querySelector("#on").classList.toggle("checked");
optionSwitch.querySelector("#off").classList.toggle("checked");
chrome.storage.sync.set({ [option]: status });
if (option === "auto_dark") {
toggleDarkModeDisable(status);
}
});
});
});
chrome.storage.sync.get(menu.checkboxes, sync => {
menu.checkboxes.forEach(option => {
document.querySelector("#" + option).addEventListener("change", function (e) {
let status = this.checked;
chrome.storage.sync.set(JSON.parse(`{"${option}": ${status}}`));
});
const value = sync[option] !== undefined ? sync[option] : defaultOptions.sync[option];
document.querySelector("#" + option).checked = value;
});
/*
document.querySelector('#autodark_start').value = result.auto_dark_start["hour"] + ":" + result.auto_dark_start["minute"];
document.querySelector('#autodark_end').value = result.auto_dark_end["hour"] + ":" + result.auto_dark_end["minute"];
document.querySelector("#assignment_date_format").checked = result.assignment_date_format == true;
document.querySelector("#todo_hr24").checked = result.todo_hr24 == true;
*/
toggleDarkModeDisable(sync.auto_dark);
});
const specialOptions = menu.special.map(obj => obj.identifier);
chrome.storage.sync.get(specialOptions, sync => {
console.log(sync);
menu.special.forEach(option => {
if (option.setup !== null) {
const value = sync[option.identifier] !== undefined ? sync[option.identifier] : defaultOptions.sync[option.identifier];
option.setup(value);
}
});
})
/*
// checkboxes
menu.checkboxes.forEach(checkbox => {
document.querySelector("#" + checkbox).addEventListener('change', function () {
let status = this.checked;
chrome.storage.sync.set(JSON.parse(`{"${checkbox}": ${status}}`));
});
});
*/
// activate tab buttons
document.querySelectorAll(".tab-btn").forEach(btn => {
btn.addEventListener("click", () => {
if (menu.tabs[btn.id].setup !== null) menu.tabs[btn.id].setup();
document.querySelector(".main").style.display = "none";
document.querySelector(menu.tabs[btn.id].tab).style.display = "block";
window.scrollTo(0, 0);
});
});
// activate the back buttons on each tab
document.querySelectorAll(".back-btn").forEach(btn => {
btn.addEventListener("click", function () {
document.querySelectorAll(".tab").forEach(tab => {
tab.style.display = "none";
});
document.querySelector(".main").style.display = "block";
});
});
// give everything the appropirate i18n text
document.querySelectorAll('[data-i18n]').forEach(text => {
text.innerText = chrome.i18n.getMessage(text.dataset.i18n);
});
// activate dark mode inspector button
document.querySelector("#inspector-btn").addEventListener("click", async function () {
document.querySelector("#inspector-output").textContent = (await sendFromPopup("inspect"))["selectors"];
});
// activate dark mode fixer button
document.querySelector("#fix-dm-btn").addEventListener("click", async function () {
let output = await sendFromPopup("fixdm");
if (output.path === "bettercanvas-none" || output.path === "bettercanvas-darkmode_off") return;
let rating = "bad";
if (output.time < 100) {
rating = "good";
} else if (output.time < 250) {
rating = "ok";
}
document.getElementById("fix-dm-output").textContent = "Fix took " + Math.round(output.time) + "ms (rating: " + rating + ")";
chrome.storage.sync.get("dark_mode_fix", sync => {
if (sync["dark_mode_fix"].includes(output.path)) return;
sync["dark_mode_fix"].push(output.path);
chrome.storage.sync.set({ "dark_mode_fix": sync["dark_mode_fix"] }).then(() => displayDarkModeFixUrls());
})
});
// activate storage dump button
document.querySelector("#rk_btn").addEventListener("click", () => {
chrome.storage.local.get(null, local => {
chrome.storage.sync.get(null, sync => {
document.querySelector("#rk_output").value = JSON.stringify(local) + JSON.stringify(sync);
})
})
});
// activate storage reset button
document.querySelector("#storage-reset-btn").addEventListener("click", () => {
chrome.storage.sync.set(defaultOptions["sync"]);
});
// activate custom url input
document.querySelector('#customDomain').addEventListener('input', function () {
let domains = this.value.split(",");
domains.forEach((domain, index) => {
let val = domain.replace(" ", "");
if (val === "") return;
//if (!val.includes("https://") && !val.includes("http://")) val = "https://" + val;
try {
let url = new URL(val);
domains[index] = url.hostname;
clearAlert();
} catch (e) {
domains[index] = val;
displayAlert(true, "The URL you entered appears to be invalid, so it might not work.");
}
});
chrome.storage.sync.set({ custom_domain: domains });
});
// setup custom url
chrome.storage.sync.get(["custom_domain"], storage => {
document.querySelector("#customDomain").value = storage.custom_domain ? storage.custom_domain : "";
});
// activate import input box
document.querySelector("#import-input").addEventListener("input", (e) => {
const obj = JSON.parse(e.target.value);
importTheme(obj);
});
// activate export checkbox
document.querySelectorAll(".export-details input").forEach(input => {
input.addEventListener("change", () => {
chrome.storage.sync.get(syncedSwitches.concat(syncedSubOptions).concat(["dark_preset", "custom_cards", "custom_font", "gpa_calc_bounds"]), async storage => {
//chrome.storage.local.get(["dark_preset"], async local => {
let final = {};
for await (item of document.querySelectorAll(".export-details input")) {
if (item.checked) {
switch (item.id) {
case "export-toggles":
final = { ...final, ...(await getExport(storage, syncedSwitches.concat(syncedSubOptions))) };
break;
case "export-dark":
final = { ...final, ...(await getExport(storage, ["dark_preset"])) };
break;
case "export-cards":
final = { ...final, ...(await getExport(storage, ["custom_cards"])) };
break;
case "export-font":
final = { ...final, ...(await getExport(storage, ["custom_font"])) };
break;
case "export-colors":
final = { ...final, ...(await getExport(storage, ["card_colors"])) }
break;
case "export-gpa":
final = { ...final, ...(await getExport(storage, ["gpa_calc_bounds"])) }
break;
case "export-customStyles":
final = { ...final, ...(await getExport(storage, ["custom_styles"])) };
break;
}
}
}
document.querySelector("#export-output").value = JSON.stringify(final);
//});
});
});
});
// activate revert to original button
document.querySelector("#theme-revert").addEventListener("click", () => {
chrome.storage.local.get("previous_theme", local => {
if (local["previous_theme"] !== null) {
importTheme(local["previous_theme"]);
}
});
});
document.querySelector("#alert").addEventListener("click", clearAlert);
document.querySelectorAll(".preset-button.customization-button").forEach(btn => btn.addEventListener("click", changeToPresetCSS));
// activate card color inputs
document.querySelector("#singleColorInput").addEventListener("change", e => document.querySelector("#singleColorText").value = e.target.value);
document.querySelector("#singleColorText").addEventListener("change", e => document.querySelector("#singleColorInput").value = e.target.value);
document.querySelector("#gradientColorFrom").addEventListener("change", e => document.querySelector("#gradientColorFromText").value = e.target.value);
document.querySelector("#gradientColorFromText").addEventListener("change", e => document.querySelector("#gradientColorFrom").value = e.target.value);
document.querySelector("#gradientColorTo").addEventListener("change", e => document.querySelector("#gradientColorToText").value = e.target.value);
document.querySelector("#gradientColorToText").addEventListener("change", e => document.querySelector("#gradientColorTo").value = e.target.value);
document.querySelector("#setSingleColor").addEventListener("click", () => {
let colors = [document.querySelector("#singleColorInput").value];;
sendFromPopup("setcolors", colors);
});
document.querySelector("#setGradientColor").addEventListener("click", () => {
chrome.storage.sync.get("custom_cards", sync => {
length = 0;
Object.keys(sync["custom_cards"]).forEach(key => {
if (sync["custom_cards"][key].hidden !== true) length++;
});
let colors = [];
let from = document.querySelector("#gradientColorFrom").value;
let to = document.querySelector("#gradientColorTo").value;
for (let i = 1; i <= length; i++) {
colors.push(getColorInGradient(i / length, from, to));
}
sendFromPopup("setcolors", colors);
});
});
// activate revert to original card colors button
document.querySelector("#revert-colors").addEventListener("click", () => {
chrome.storage.local.get("previous_colors", local => {
if (local["previous_colors"] !== null) {
sendFromPopup("setcolors", local["previous_colors"].colors);
}
})
})
// activate every card color palette button
document.querySelectorAll(".preset-button.colors-button").forEach(btn => {
const colors = getPalette(btn.querySelector("p").textContent);
let preview = btn.querySelector(".colors-preview");
colors.forEach(color => {
let div = makeElement("div", preview, { "className": "color-preview"});
div.style.background = color;
});
btn.addEventListener("click", () => {
sendFromPopup("setcolors", colors);
})
});
/*
['autodark_start', 'autodark_end'].forEach(function (timeset) {
document.querySelector('#' + timeset).addEventListener('change', function () {
let timeinput = { "hour": this.value.split(':')[0], "minute": this.value.split(':')[1] };
timeset === "autodark_start" ? chrome.storage.sync.set({ auto_dark_start: timeinput }) : chrome.storage.sync.set({ auto_dark_end: timeinput });
});
});
*/
// activate sidebar tool radio
["#radio-sidebar-image", "#radio-sidebar-gradient", "#radio-sidebar-solid"].forEach(radio => {
document.querySelector(radio).addEventListener("click", () => {
chrome.storage.sync.get(["dark_preset"], storage => {
let mode = radio === "#radio-sidebar-image" ? "image" : radio === "#radio-sidebar-gradient" ? "gradient" : "solid";
displaySidebarMode(mode, storage["dark_preset"]["sidebar"]);
});
})
});
// theme browser controls
document.getElementById("premade-themes-left").addEventListener("click", () => changePage(-1));
document.getElementById("premade-themes-right").addEventListener("click", () => changePage(1));
document.getElementById("theme-sorts").addEventListener("click", () => {
const el = document.getElementById("theme-sort-selector");
if (el.classList.contains("open")) {
clickout();
} else {
el.classList.add("open");
setTimeout(() => {
document.addEventListener("click", clickout);
}, 10);
}
});
document.getElementById("theme-search").addEventListener("change", async (e) => {
searchFor = e.target.value;
console.log(searchFor);
// current_page_num = 1;
// displayThemeList(0);
// linear search
let themesToShow = [];
for (let i = 0; i < themes.length; i++) {
if (themes[i].title.toLowerCase().includes(searchFor.toLowerCase())) {
themesToShow.push(themes[i]);
}
}
console.log(themesToShow);
displayThemeSearchList(themesToShow);
});
// activate theme save button
document.getElementById("save-theme").addEventListener("click", saveCurrentTheme);
// activate submit theme button
// document.getElementById("submit-theme-btn").addEventListener("click", submitTheme);
document.getElementById("submit-theme-btn-1").addEventListener("click", () => {
document.getElementById("submit-popup").classList.add("open");
});
document.getElementById("cancel-theme-btn").addEventListener("click", () => {
document.getElementById("submit-popup").classList.remove("open");
})
// update theme button preview on input
document.getElementById("submit-title").addEventListener("input", (e) => {
document.getElementById("theme-button-title-preview").textContent = e.target.value.replaceAll(" ", "");
});
// update theme button preview on input
document.getElementById("submit-credits").addEventListener("input", (e) => {
document.getElementById("theme-button-creator-preview").textContent = e.target.value;
});
// activate the show button to open the theme submission drawer
document.getElementById("show-submit-form").addEventListener("click", (e) => {
const drawer = document.getElementById("submit-drawer");
if (drawer.style.display === "none") {
drawer.style.display = "block";
e.target.textContent = "Hide";
} else {
drawer.style.display = "none";
e.target.textContent = "Show";
}
});
// activate theme browser opt out
// document.getElementById("new_browser_out").addEventListener("click", () => {
chrome.storage.sync.set({ "new_browser": false });
current_page_num = 1;
displayThemeList(0);
// displayAlert(false, "Success! You are now viewing the old theme browser. This one will no longer recieve updates, but there is still plenty to choose from.");
// });
// activate theme browser opt in
// document.getElementById("new_browser_in").addEventListener("click", registerUser);
document.querySelectorAll(".theme-sort-btn").forEach(btn => {
btn.addEventListener("click", (e) => {
themeSort(e.target.textContent);
});
});
// browser settings buttons
document.getElementById("browser-settings-btn").addEventListener("click", () => {
document.getElementById("browser-settings-popup").classList.add("open");
});
document.getElementById("close-settings-btn").addEventListener("click", () => {
displayThemeList(0);
document.getElementById("browser-settings-popup").classList.remove("open");
});
// document.getElementById("reset-optin").addEventListener("click", () => {
// chrome.storage.sync.set({ "new_browser": null });
// document.getElementById("opt-in").style.display = "block";
// });
// document.getElementById("view-submissions-btn").addEventListener("click", displayMySubmissions);
document.getElementById("submit-form-btn").addEventListener("click", displayThemeSubmissionForm);
document.getElementById("gpa-plus-minus").addEventListener("click", () => {
applyGPAPreset({
"A+": { "cutoff": 97, "gpa": 4.0 },
"A": { "cutoff": 93, "gpa": 4.0 },
"A-": { "cutoff": 90, "gpa": 3.7 },
"B+": { "cutoff": 87, "gpa": 3.3 },
"B": { "cutoff": 83, "gpa": 3.0 },
"B-": { "cutoff": 80, "gpa": 2.7 },
"C+": { "cutoff": 77, "gpa": 2.3 },
"C": { "cutoff": 73, "gpa": 2.0 },
"C-": { "cutoff": 70, "gpa": 1.7 },
"D+": { "cutoff": 67, "gpa": 1.3 },
"D": { "cutoff": 63, "gpa": 1.0 },
"D-": { "cutoff": 60, "gpa": 0.7 },
"F": { "cutoff": 0, "gpa": 0 }
});
});
document.getElementById("gpa-by-letter").addEventListener("click", () => {
applyGPAPreset({
"A+": { "cutoff": 97, "gpa": 4.0 },
"A": { "cutoff": 93, "gpa": 4.0 },
"A-": { "cutoff": 90, "gpa": 4.0 },
"B+": { "cutoff": 87, "gpa": 3.0 },
"B": { "cutoff": 83, "gpa": 3.0 },
"B-": { "cutoff": 89, "gpa": 3.0 },
"C+": { "cutoff": 77, "gpa": 2.0 },
"C": { "cutoff": 73, "gpa": 2.0 },
"C-": { "cutoff": 70, "gpa": 2.0 },
"D+": { "cutoff": 67, "gpa": 1.0 },
"D": { "cutoff": 63, "gpa": 1.0 },
"D-": { "cutoff": 60, "gpa": 1.0 },
"F": { "cutoff": 0, "gpa": 0 }
});
});
document.getElementById("imageSize").addEventListener("input", (e) => {
const value = e.target.value;
chrome.storage.sync.set({ "imageSize": value });
document.querySelector("#imageSizeValue").textContent = value + "%";
})
document.getElementById("cardRoundness").addEventListener("input", (e) => {
const value = e.target.value;
chrome.storage.sync.set({ "cardRoundness": value });
document.querySelector("#cardRoundnessValue").textContent = value + "px";
})
document.getElementById("cardSpacing").addEventListener("input", (e) => {
const value = e.target.value;
chrome.storage.sync.set({ "cardSpacing": value });
document.querySelector("#cardSpacingValue").textContent = value + "px";
})
document.getElementById("cardWidth").addEventListener("input", (e) => {
const value = e.target.value;
chrome.storage.sync.set({ "cardWidth": value });
document.querySelector("#cardWidthValue").textContent = value + "%";
});
document.getElementById("cardHeight").addEventListener("input", (e) => {
const value = e.target.value;
chrome.storage.sync.set({ "cardHeight": value });
document.querySelector("#cardHeightValue").textContent = value + "%";
});
document.getElementById("clearCustomBackground").addEventListener("click", () => {
chrome.storage.sync.set({
"customBackgroundLink": "",
"customBackgroundUpload": ""
});
document.querySelector("#customBackgroundLink").value = "";
document.querySelector("#customBackgroundUpload").value = "";
});
}
function applyGPAPreset(bounds) {
chrome.storage.sync.set({ "gpa_calc_bounds": bounds }, () => {
displayGPABounds();
});
}
function setupCustomStyle(initial) {
const el = document.getElementById("custom-styles");
el.value = initial;
el.addEventListener("change", (e) => {
chrome.storage.sync.set({ "custom_styles": e.target.value });
});
}
function displayThemeSubmissionForm() {
document.getElementById("submit-form").style.display = "block";
document.getElementById("view-submissions").style.display = "none";
document.getElementById("submit-form-btn").classList.add("active");
document.getElementById("view-submissions-btn").classList.remove("active");
}
async function displayMySubmissions() { //TODO: remake
const sync = await chrome.storage.sync.get("id");
const res = await fetch(`${apiurl}/api/themes/submissions?id=${sync["id"]}`);
const data = await res.json();
//if (data?.errors !== false) return;
document.getElementById("submit-form").style.display = "none";
document.getElementById("view-submissions").style.display = "block";
document.getElementById("submit-form-btn").classList.remove("active");
document.getElementById("view-submissions-btn").classList.add("active");
const el = document.getElementById("latest-submissions");
el.textContent = "";
if (data.message.length === 0) {
el.textContent = "You haven't submitted any themes yet.";
}
data.message.forEach(theme => {
const container = makeElement("div", el, {"className": "submitted-theme" });
const btn = makeElement("button", container, { "className": "theme-button clickable customization-button", "style": `min-width:105px;max-width:105px;background-image:linear-gradient(rgba(0, 0, 0, 0.44), rgba(0, 0, 0, 0.44)), url(${theme.preview})` });
const title = makeElement("p", btn, { "className": "theme-button-title clickable", "textContent": theme.title });
const credits = makeElement("p", btn, { "className": "theme-button-creator clickable", "textContent": theme.credits });
const details = makeElement("div", container, { "className": "submitted-theme-details" });
const top = makeElement("div", details, { "style": "display:flex;justify-content:space-between;align-items:center" });
const tag = makeElement("span", top, { "className": "submitted-theme-tag", "textContent": theme.approved === 1 ? "Approved" : theme.approved === 0 ? "Pending" : "Rejected", "style": `background: ${theme.approved === 1 ? "#ad3a74" : theme.approved === 0 ? "#514e4e": "#000"}` });
const msg = makeElement("p", details, { "textContent": theme.approved === 1 ? "Looks great! Thanks for submitting" : theme.approved === 0 ? "Your theme is still awaiting approval." : `Your theme was rejected${theme.reason ? (": " + theme.reason) : " because it did not meet the theme guidelines."}`});
const ago = makeElement("span", top, { "className": "submitted-theme-time", "textContent": `${getRelativeDate(new Date(parseInt(theme.time))).time} ago` });
});
}
async function getExport(storage, options) {
let final = {};
for (const option of options) {
switch (option) {
case "custom_cards":
let arr = [];
Object.keys(storage["custom_cards"]).forEach(key => {
if (storage["custom_cards"][key].img !== "") arr.push(storage["custom_cards"][key].img);
});
if (arr.length === 0) {
arr = ["none"];
}
final["custom_cards"] = arr;
break;
case "card_colors":
final["card_colors"] = [];
try {
final["card_colors"] = await sendFromPopup("getcolors");
} catch (e) {
console.log(e);
}
break;
case "custom_styles":
final["customCardStyles"] = storage["customCardStyles"];
final["imageSize"] = storage["imageSize"];
final["cardRoundness"] = storage["cardRoundness"];
final["cardSpacing"] = storage["cardSpacing"];
final["cardWidth"] = storage["cardWidth"];
final["cardHeight"] = storage["cardHeight"];
break;
default:
final[option] = storage[option];
}
}
return final;
}
let pageTimeout = false;
function changePage(direction) {
if (pageTimeout) return;
pageTimeout = true;
displayThemeList(direction);
setTimeout(() => {
pageTimeout = false;
}, 500);
}
const colorValues = {
"red": 1,
"pink": 2,
"orange": 3,
"yellow": 4,
"lightgreen": 5,
"green": 6,
"lightblue": 7,
"blue": 8,
"lightpurple": 9,
"purple": 10,
"lightpurple": 11,
"beige": 12,
"brown": 13,
"gray": 14,
}
function themeSortFn(method) {
let themes = getTheme("all");
switch (method) {
case "New":
return themes.reverse();
case "Old":
return themes;
case "Color":
return themes.sort((a, b) => {
//return (colorValues[a.color] || 88) - (colorValues[b.color] || 88)
return (colorValues[a.color] || (a.color !== "whiteblack" && a.color.includes("white") ? 15 : 16)) - (colorValues[b.color] || (b.color !== "whiteblack" && b.color.includes("white") ? 15 : 16))
})
return themes.sort((a, b) => {
return a.color < b.color ? 1 : -1;
})
case "ABC":
return themes.sort((a, b) => {
return a.title.toLowerCase() > b.title.toLowerCase() ? 1 : -1;
})
default:
return shuffle(themes).sort((a, b) => {
a = a.score + "";
b = b.score + "";
a = parseInt(a.charAt(0)) + parseInt(a.charAt(1)) + parseInt(a.charAt(2)) + parseInt(a.charAt(3));
b = parseInt(b.charAt(0)) + parseInt(b.charAt(1)) + parseInt(b.charAt(2)) + parseInt(b.charAt(3));
return b - a;
});
}
}
let cache = {};
// new theme sort button
function themeSort(sort) {
current_sort = sort;
current_page_num = 1;
allThemes = themeSortFn(current_sort);
displayThemeList(0);
}
function clickout() {
setTimeout(() => {
document.getElementById("theme-sort-selector").classList.remove("open");
document.removeEventListener("click", clickout);
}, 10);
}
/*
function sortThemes(method) {
const sortMethods = ["Popular", "ABC", "New", "Old"];
const index = sortMethods.indexOf(method);
const next = index + 1 === sortMethods.length ? 0 : index + 1;
current_sort = sortMethods[next];
allThemes = themeSortFn(current_sort);
document.querySelectorAll(".theme-sort-btn").forEach(btn => {
if (btn.id.includes(method)) {
btn.style.color = "#fff";
btn.style.background = "var(--inputbg)"
} else {
btn.style.color = "#adadad";
btn.style.background = "none"
}
});
current_page_num = 1;
displayThemeList(0);
//cache = {};
}
*/
// shuffle function for the score sorting so theres no order bias
function shuffle (arr) {
var j, x, index;
for (index = arr.length - 1; index > 0; index--) {
j = Math.floor(Math.random() * (index + 1));
x = arr[index];
arr[index] = arr[j];
arr[j] = x;
}
return arr;
}
let current_page_num = 1;
let maxPage = 0;
let searchFor = "";
let current_sort = "Popular";
let allThemes = themeSortFn(current_sort);
//sortThemes(current_sort);
function shortScore(score) {
if (score >= 1400) {
return (Math.floor(score / 1000) + "." + Math.round((score % 1000) / 100)) + "k";
}
return score;
}
let fallback = false;
async function submitTheme() { //TODO: remake
const sync = await chrome.storage.sync.get(null);
// if (sync["new_browser"] !== true) {
// displayAlert(true, "You'll need to opt in to the new browser if you want to submit your theme. If you've opted out and want to opt in, you can scroll down to the bottom of this page and opt back in.");
// return;
// }
const theme = await getExport(sync, ["custom_cards", "card_colors", "dark_preset", "custom_font", "gradient_cards", "disable_color_overlay"]);
const title = document.getElementById("submit-title");
const credits = document.getElementById("submit-credits");
if (title.value === "") {
displayAlert(true, "The title of your theme can't be empty");
return;
}
if (credits.value === "") {
displayAlert(true, "The credits for your theme can't be empty");
return;
}
const body = JSON.stringify({
"identity": sync["id"],
"title": title.value,
"credits": credits.value,
"theme": JSON.stringify(theme)
});
fetch(`${apiurl}/api/themes/submit`, { //
"method": "POST",
"body": body,
"headers": {
"Content-Type": "application/json",
},
}).then(res => res.json())
.then(data => {
console.log(data);
if (data.errors === false) {
displayAlert(false, "Thanks for submitting your theme! I will try to approve it soon, but not every theme may be accepted.");
document.getElementById("submit-popup").classList.remove("open");
} else {
displayAlert(true, `Submission error: ${data.message} Please contact sandlerguy5@gmail.com if you believe this is incorrect.`);
}
});
}
async function registerUser() { // TODO: remake
try {
let id;
const sync = await chrome.storage.sync.get("id");
if (sync["id"] && sync["id"] !== "") {
id = sync["id"]
} else {
const res = await fetch(`${apiurl}/api/register`);
const data = await res.json();
id = data.id;
}
chrome.storage.sync.set({ "id": id }).then(async () => {
// test to see if the id was set correctly
// don't know why this is happening ??
const test = await chrome.storage.sync.get("id");
if (test["id"] === undefined || test["id"] === "") throw new Error();
// show the new browser
chrome.storage.sync.set({ "new_browser": true }).then(() => {
document.getElementById("opt-in").style.display = "none";
current_page_num = 1;
displayThemeList(0);
displayAlert(false, "Success! You should be able to see the new themes browser now. Enjoy!");
});
}).catch(e => {
displayAlert(true, "There was an error connecting an ID to your account. Please try again, and if this error persists, contact sandlerguy5@gmail.com!");
});
} catch (e) {
console.log(e);
displayAlert(true, "There was an error opting in. Please contact sandlerguy5@gmail.com if this error persists!");
}
}
function saveCurrentTheme() {
const allOptions = syncedSwitches.concat(syncedSubOptions).concat(["dark_preset", "custom_cards", "custom_font", "gpa_calc_bounds", "card_colors"]);
chrome.storage.local.get("saved_themes", local => {
chrome.storage.sync.get(allOptions, async sync => {
let current = await getExport(sync, allOptions);
let trimmed = {
"disable_color_overlay": current["disable_color_overlay"],
"gradient_cards": current["gradient_cards"],
"dark_mode": current["dark_mode"],
"dark_preset": current["dark_preset"],
"custom_cards": current["custom_cards"],
"card_colors": current["card_colors"] === null ? [current["dark_preset"]["links"]] : current["card_colors"],
"custom_font": current["custom_font"]
}
const now = new Date();
local["saved_themes"][now.getTime()] = trimmed;
chrome.storage.local.set({ "saved_themes": local["saved_themes"] }).then(() => {
displaySavedThemes();
});
});
});
}
async function displayThemeList(direction = 0) {
// const sync = await chrome.storage.sync.get("new_browser");
// if (sync["new_browser"] === true && fallback === false) {
// displayThemeListNew(direction);
// } else {
displayThemeListOld(direction);
// }
// remove the opt-in notice
// if (sync["new_browser"] !== null && document.getElementById("opt-in")) document.getElementById("opt-in").style.display = "none";
}
function createThemeButton(location, theme) {
let themeBtn = makeElement("button", location, { "className": "theme-button clickable" });
themeBtn.classList.add("customization-button");
if (!themeBtn.style.background) themeBtn.style.backgroundImage = "linear-gradient(#00000070, #00000070), url(" + theme.preview + ")";
if (theme.title) makeElement("p", themeBtn, { "className": "theme-button-title clickable", "textContent": theme.title.replaceAll(" ", "") });
if (theme.credits) makeElement("p", themeBtn, { "className": "theme-button-creator clickable", "textContent": theme.credits });
return themeBtn;
}
function createThemeLikeBtn(location, initial, score, show) {
const likeBtn = makeElement("div", location, {"className": "theme-button-like"});
if (initial === true) {
likeBtn.classList.add("theme-liked");
score += 1;
}
const amount = makeElement("span", likeBtn, { "className": "theme-button-like-amount", "textContent": shortScore(score) });
if (show === true) amount.classList.add("showalways");
likeBtn.innerHTML += `<svg xmlns="http://www.w3.org/2000/svg" width="12" height="12" viewBox="0 0 24 24"><path stroke="none" d="M0 0h24v24H0z" fill="none"/><path d="M6.979 3.074a6 6 0 0 1 4.988 1.425l.037 .033l.034 -.03a6 6 0 0 1 4.733 -1.44l.246 .036a6 6 0 0 1 3.364 10.008l-.18 .185l-.048 .041l-7.45 7.379a1 1 0 0 1 -1.313 .082l-.094 -.082l-7.493 -7.422a6 6 0 0 1 3.176 -10.215z" /></svg>`;
return likeBtn;
}
let likeThemeTimeout = false;
function setLikeTimeout() {
if (likeThemeTimeout === true) return;
likeThemeTimeout = true;
setTimeout(() => {
likeThemeTimeout = false;
}, 1000);
}
async function likeTheme(location, code, score) { // TODO: remake
if (likeThemeTimeout === true) return;
const sync = await chrome.storage.sync.get("id");
const local = await chrome.storage.local.get("liked_themes");
const setLikeStatus = (direction) => {
let output = local;
if (direction === -1) {
location.classList.remove("theme-liked");
location.querySelector(".theme-button-like-amount").textContent = shortScore(score);
output = local["liked_themes"].filter(x => x !== code);
} else if (direction === 1) {
location.classList += (" theme-liked animate-like");
location.querySelector(".theme-button-like-amount").textContent = shortScore(score + 1);
output = [...local["liked_themes"], code];
}
return output;
}
// show the updated like status immediately
setLikeStatus(location.classList.contains("theme-liked") ? -1 : 1);
const res = await fetch(`${apiurl}/api/themes/theme/${code}/like`, {
"method": "POST",
"body": JSON.stringify({ "id": sync["id"] }),
"headers": {
"Content-Type": "application/json"
},
});
const data = await res.json();
if (data.errors === false) {
const direction = parseInt(data.message);
// update the like status if there is some disagreement with the server
const update = setLikeStatus(direction);
chrome.storage.local.set({ "liked_themes": update }).then(setLikeTimeout);
} else {
setLikeTimeout();
}
}
async function getAndLoadTheme(code) { // todo: remake
const key = `themes/${code}`;
let output = {};
if (cache[key]) {
output = cache[key];
console.log("got this theme from the cache.");
} else {
const res = await fetch(`${apiurl}/api/themes/theme/${code}`);
const data = await res.json();
output = JSON.parse(data.message.exports);
cache[key] = output;
}
importTheme(output);
}
async function displayThemeListNew(direction) { // TODO: remake
document.getElementById("theme-current-sort").textContent = current_sort;
if (direction === -1 && current_page_num > 1) current_page_num--;
if (direction === 1 && current_page_num < maxPage) current_page_num++;
let themes = [];
let apiLink = `${current_sort.toLowerCase()}?page=${current_page_num}` + (searchFor === "" ? "" : `&searchFor=${searchFor}`);
if (current_sort === "Liked") {
const sync = await chrome.storage.sync.get("id");
const local = await chrome.storage.local.get("liked_themes");
if (sync["id"] && sync["id"] !== "") {
apiLink += `&id=${sync["id"]}`;
maxPage = Math.ceil(local["liked_themes"].length / 28);
} else { // fallback if there is no id
current_page_num = 1;
apiLink = `popular?page=${current_page_num}` + (searchFor === "" ? "" : `&searchFor=${searchFor}`);
}
}
// fetch api, fallback if necessary
if (cache[apiLink]) {
themes = cache[apiLink]["themes"];
maxPage = cache[apiLink]["pages"] || maxPage;
} else {
try {
const res = await fetch(`${apiurl}/api/themes/${apiLink}`, {
method: "get",
headers: {
"Content-Type": "application/json"
},
});
const data = await res.json();
if (data.errors === true) throw new Error(data.message);
themes = data.message.themes;
cache[apiLink] = data.message;
if (data?.message?.pages) {
maxPage = data.message.pages;
}
} catch (e) {
console.log(e);
current_page_num = 1;
fallback = true;
displayAlert(true, "There was a problem getting themes from the Better Canvas server, so the old themes browser is being displayed for now.");
displayThemeListOld(0);
return;
}
}
let container = document.getElementById("premade-themes");
container.textContent = "";
const local = await chrome.storage.local.get("liked_themes");
const sync = await chrome.storage.sync.get("browser_show_likes");
themes.forEach(theme => {
const themeBtn = createThemeButton(container, theme);
themeBtn.addEventListener("click", (e) => {
if (!e.target.classList.contains("clickable")) return;
// getAndLoadTheme(theme.code)
});
const liked = local["liked_themes"].includes(theme.code);
// TODO: remake
const likeBtn = createThemeLikeBtn(themeBtn, liked, theme.score, sync["browser_show_likes"]);
// likeBtn.addEventListener("click" , (e) => likeTheme(likeBtn, theme.code, theme.score));
});
if (themes.length === 0) {
container.innerHTML = `<div id="themes-empty">Nothing here</div>`;
}
document.getElementById("premade-themes-pagenum").textContent = current_page_num + " of " + maxPage;
// set the submit theme button to the first custom card image
try {
const sync = await chrome.storage.sync.get("custom_cards");
const exports = await getExport(sync, ["custom_cards"]);
document.getElementById("theme-button-img").style.background = `linear-gradient(#00000070, #00000070), url(${exports["custom_cards"][0]}) no-repeat center center / cover`;
} catch (e) {
console.log(e);
}
displaySavedThemes();
}
function displayThemeListOld(pageDir = 0) {
//const keys = Object.keys(themes);
document.getElementById("theme-current-sort").textContent = current_sort;
const perPage = 24;
const maxPage = Math.ceil(allThemes.length / perPage);
if (pageDir === -1 && current_page_num > 1) current_page_num--;
if (pageDir === 1 && current_page_num < maxPage) current_page_num++;
let container = document.getElementById("premade-themes");
container.textContent = "";
let start = (current_page_num - 1) * perPage, end = start + perPage;
allThemes.forEach((theme, index) => {
if (index < start || index >= end) return;
let themeBtn = makeElement("button", container, { "className": "theme-button" });
themeBtn.classList.add("customization-button");
if (!themeBtn.style.background) themeBtn.style.backgroundImage = "linear-gradient(#00000070, #00000070), url(" + theme.preview + ")";
let split = theme.title.split(" by ");
makeElement("p", themeBtn, {"className": "theme-button-title", "textContent": split[0] });
makeElement("p", themeBtn, {"className": "theme-button-creator", "textContent": split[1] });
themeBtn.addEventListener("click", () => {
const allOptions = syncedSwitches.concat(syncedSubOptions).concat(["dark_preset", "custom_cards", "custom_font", "gpa_calc_bounds", "card_colors"]);
chrome.storage.sync.get(allOptions, sync => {
chrome.storage.local.get(["previous_theme"], async local => {
if (local["previous_theme"] === null) {
let previous = await getExport(sync, allOptions);
chrome.storage.local.set({ "previous_theme": previous });
}
importTheme(theme.exports);
});
});
});
});
document.getElementById("premade-themes-pagenum").textContent = current_page_num + " of " + maxPage;
displaySavedThemes();
}
function displayThemeSearchList(themesToShow, pageDir = 0) {
document.getElementById("theme-current-sort").textContent = current_sort;
const perPage = 24;
const maxPage = Math.ceil(themesToShow.length / perPage);
if (pageDir == -1 && current_page_num > 1) current_page_num--;
if (pageDir == 1 && curren_page_num < maxPage) current_page_num++;
let container = document.getElementById("premade-themes");
container.textContent = "";
let start = (current_page_num - 1) * perPage, end = start + perPage;
themesToShow.forEach((theme, index) => {
if (index < start || index >= end) return;
let themeBtn = makeElement("button", container, { "className": "theme-button" });
themeBtn.classList.add("customization-button");
if (!themeBtn.style.background) themeBtn.style.backgroundImage = "linear-gradient(#00000070, #00000070), url(" + theme.preview + ")";
let split = theme.title.split(" by ");
makeElement("p", themeBtn, {"className": "theme-button-title", "textContent": split[0] });
makeElement("p", themeBtn, {"className": "theme-button-creator", "textContent": split[1] });
themeBtn.addEventListener("click", () => {
const allOptions = syncedSwitches.concat(syncedSubOptions).concat(["dark_preset", "custom_cards", "custom_font", "gpa_calc_bounds", "card_colors"]);
chrome.storage.sync.get(allOptions, sync => {
chrome.storage.local.get(["previous_theme"], async local => {
if (local["previous_theme"] === null) {
let previous = await getExport(sync, allOptions);
chrome.storage.local.set({ "previous_theme": previous });
}
importTheme(theme.exports);
});
});
});
}
)
}
function getRelativeDate(date, short = false) {
let now = new Date();
let timeSince = (now.getTime() - date.getTime()) / 60000;
let time = "min";
timeSince = Math.abs(timeSince);
if (timeSince >= 60) {
timeSince /= 60;
time = short ? "h" : "hour";
if (timeSince >= 24) {
timeSince /= 24;
time = short ? "d" : "day";
if (timeSince >= 7) {
timeSince /= 7;
time = short ? "w" : "week";
}
}
}
timeSince = Math.round(timeSince);
let relative = timeSince + (short ? "" : " ") + time + (timeSince > 1 && !short ? "s" : "");
return { time: relative, ms: now.getTime() - date.getTime() };
}
function displaySavedThemes() {
chrome.storage.local.get("saved_themes", local => {
const target = document.getElementById("saved-themes");
target.textContent = "";
Object.keys(local["saved_themes"]).forEach((key, index) => {
const created = new Date(parseInt(key));
let btn = makeElement("div", target, { "className": "saved-theme" });
let title = makeElement("p", btn, { "className": "theme-button-title", "textContent": `Theme ${index + 1}`});
let date = makeElement("p", btn, { "className": "theme-button-creator", "textContent": `${getRelativeDate(created).time} ago` });
let remove = makeElement("div", btn, { "className": "theme-button-remove", "textContent": "x" });
btn.style.backgroundImage = `linear-gradient(rgba(0, 0, 0, 0.44), rgba(0, 0, 0, 0.44)), url(${local["saved_themes"][key]["custom_cards"][0]})`;
btn.addEventListener("click", () => {
importTheme(local["saved_themes"][key]);
});
remove.addEventListener("click", () => {
chrome.storage.local.get("saved_themes", local => {
delete local["saved_themes"][key];
chrome.storage.local.set({ "saved_themes": local["saved_themes"] }).then(() => {
btn.remove();
})
})
});
});
});
}
function getTheme(name) {
// get localThemes from themes.js
console.log(themes);
if (name === "all") return themes;
for (const theme in themes) if (theme.title === name) return theme
return {};
}
function importTheme(theme) {
try {
let keys = Object.keys(theme);
let final = {};
chrome.storage.sync.get("custom_cards", sync => {
keys.forEach(key => {
switch (key) {
case "dark_preset":
changeToPresetCSS(null, theme["dark_preset"]);
break;
case "card_colors":
sendFromPopup("setcolors", theme["card_colors"]);
break;
case "custom_cards":
if (theme["custom_cards"].length > 0) {
let pos = 0;
Object.keys(sync["custom_cards"]).forEach(key => {
sync["custom_cards"][key].img = theme["custom_cards"][pos];
pos = (pos === theme["custom_cards"].length - 1) ? 0 : pos + 1;
});
}
final["custom_cards"] = sync["custom_cards"];
break;
default:
final[key] = theme[key];
break;
}
});
chrome.storage.sync.set(final);
});
} catch (e) {
console.log(e);
}
}
function updateCards(key, value) {
chrome.storage.sync.get(["custom_cards"], result => {
chrome.storage.sync.set({ "custom_cards": { ...result["custom_cards"], [key]: { ...result["custom_cards"][key], ...value } } }, () => {
if (chrome.runtime.lastError) {
displayAlert(true, "The data you're entering is exceeding the storage limit, so it won't save. Try using shorter links, and make sure to press \"copy image address\" and not \"copy image\" for links.");
}
})
});
}
function displayCustomFont() {
chrome.storage.sync.get(["custom_font"], storage => {
let el = document.querySelector(".custom-font");
let linkContainer = document.querySelector(".custom-font-flex") || makeElement("div", el, {"className": "custom-font-flex" });
linkContainer.innerHTML = '<span>https://fonts.googleapis.com/css2?family=</span><input class="card-input" id="custom-font-link"></input>';
let link = linkContainer.querySelector("#custom-font-link");
link.value = storage.custom_font.link;
link.addEventListener("change", function (e) {
let linkVal = e.target.value.split(":")[0];
let familyVal = linkVal.replace("+", " ");
linkVal += linkVal === "" ? "" : ":wght@400;700";
familyVal = linkVal === "" ? "" : "'" + familyVal + "'";
chrome.storage.sync.set({ "custom_font": { "link": linkVal, "family": familyVal } });
link.value = linkVal;
});
const popularFonts = ["Arimo", "Barriecito", "Barlow", "Caveat", "Cinzel", "Comfortaa", "Corben", "DM Sans", "Expletus Sans", "Gluten", "Happy Monkey", "Inconsolata", "Inria Sans", "Jost", "Kanit", "Karla", "Kode Mono", "Lobster", "Lora", "Madimi One", "Mali", "Montserrat", "Nanum Myeongjo", "Open Sans", "Oswald", "Permanent Marker", "Playfair Display", "Poetsen One", "Poppins", "Quicksand", "Rakkas", "Redacted Script", "Roboto Mono", "Rubik", "Silkscreen", "Sixtyfour", "Syne Mono", "Tektur", "Texturina", "Ysabeau Infant", "Yuji Syuku"];
let quickFonts = document.querySelector("#quick-fonts");
quickFonts.textContent = "";
let noFont = makeElement("button", quickFonts, { "className": "customization-button", "textContent": "None" });
noFont.addEventListener("click", () => {
chrome.storage.sync.set({ "custom_font": { "link": "", "family": "" } });
link.value = "";
})
popularFonts.forEach(font => {
let btn = makeElement("button", quickFonts, { "className":"customization-button", "textContent": font });
btn.addEventListener("click", () => {
let linkVal = font.replace(" ", "+") + ":wght@400;700";
chrome.storage.sync.set({ "custom_font": { "link": linkVal, "family": "'" + font + "'" } });
link.value = linkVal;
});
});
});
}
function displayGPABounds() {
chrome.storage.sync.get(["gpa_calc_bounds"], storage => {
const order = ["A+", "A", "A-", "B+", "B", "B-", "C+", "C", "C-", "D+", "D", "D-", "F"];
const el = document.querySelector(".gpa-bounds");
el.textContent = "";
order.forEach(key => {
let inputs = makeElement("div", el, { "className": "gpa-bounds-item" });
inputs.innerHTML += '<div><span class="gpa-bounds-grade"></span><input class="gpa-bounds-input gpa-bounds-cutoff" type="text"></input><span style="margin-left:6px;margin-right:6px;">%</span><input class="gpa-bounds-input gpa-bounds-gpa" type="text" value=></input><span style="margin-left:6px">GPA</span></div>';
inputs.querySelector(".gpa-bounds-grade").textContent = key;
inputs.querySelector(".gpa-bounds-cutoff").value = storage["gpa_calc_bounds"][key].cutoff;
inputs.querySelector(".gpa-bounds-gpa").value = storage["gpa_calc_bounds"][key].gpa;
inputs.querySelector(".gpa-bounds-cutoff").addEventListener("change", function (e) {
chrome.storage.sync.get(["gpa_calc_bounds"], existing => {
chrome.storage.sync.set({ "gpa_calc_bounds": { ...existing["gpa_calc_bounds"], [key]: { ...existing["gpa_calc_bounds"][key], "cutoff": parseFloat(e.target.value) } } });
});
});
inputs.querySelector(".gpa-bounds-gpa").addEventListener("change", function (e) {
chrome.storage.sync.get(["gpa_calc_bounds"], existing => {
chrome.storage.sync.set({ "gpa_calc_bounds": { ...existing["gpa_calc_bounds"], [key]: { ...existing["gpa_calc_bounds"][key], "gpa": parseFloat(e.target.value) } } });
});
});
});
});
}
let removeAlert = null;
function clearAlert() {
clearTimeout(removeAlert);
document.querySelector("#alert").style.bottom = "-400px";
}
function displayAlert(bad, msg) {
clearTimeout(removeAlert);
document.querySelector("#alert").style.bottom = "0";
document.querySelector("#alert").textContent = msg;
document.querySelector("#alert").style.background = bad ? "#e7495ed9" : "#468b46d9";
removeAlert = setTimeout(() => {
clearAlert();
}, 15000);
}
function setCustomImage(key, val) {
if (val !== "" && val !== "none") {
let test = new Image();
test.onerror = () => {
displayAlert(true, "It seems that the image link you provided isn't working. Make sure to right click on any images you want to use and select \"copy image address\" to get the correct link.");
// ensures storage limit error will override previous error
updateCards(key, { "img": val });
}
test.onload = clearAlert;
test.src = val;
}
updateCards(key, { "img": val });
}
function displayAdvancedCards() {
sendFromPopup("getCards");
chrome.storage.sync.get(["custom_cards", "custom_cards_2"], storage => {
document.querySelector(".advanced-cards").innerHTML = '<div id="advanced-current"></div><div id="advanced-past"><h2>Past Courses</h2></div>';
const keys = storage["custom_cards"] ? Object.keys(storage["custom_cards"]) : [];
if (keys.length > 0) {
let currentEnrollment = keys.reduce((max, key) => storage["custom_cards"][key]?.eid > max ? storage["custom_cards"][key].eid : max, -1);
keys.forEach(key => {
let term = document.querySelector("#advanced-past");
if (storage["custom_cards"][key].eid === currentEnrollment) {
term = document.querySelector("#advanced-current");
}
let card = storage["custom_cards"][key];
let card_2 = storage["custom_cards_2"][key] || {};
if (!card || !card_2 || !card_2["links"] || card_2["links"]["custom"]) {
console.log(key + " error...");
console.log("card = ", card, "card_2", card_2, "links", card_2["links"]);
} else {
let container = makeElement("div", term, { "className": "custom-card" });
container.classList.add("option-container");
container.innerHTML = '<div class="custom-card-header"><p class="custom-card-title"></p><div class="custom-card-hide"><p class="custom-key">Hide</p></div></div><div class="custom-card-inputs"><div class="custom-card-left"><div class="custom-card-image"><span class="custom-key">Image</span></div><div class="custom-card-name"><span class="custom-key">Name</span></div><div class="custom-card-code"><span class="custom-key">Code</span></div></div><div class="custom-links-container"><p class="custom-key">Links</p><div class="custom-links"></div></div></div>';
let imgInput = makeElement("input", container.querySelector(".custom-card-image"), { "className": "card-input" });
let nameInput = makeElement("input", container.querySelector(".custom-card-name"), { "className": "card-input" });
let codeInput = makeElement("input", container.querySelector(".custom-card-code"), { "className": "card-input" });
let hideInput = makeElement("input", container.querySelector(".custom-card-hide"), { "className": "card-input-checkbox" });
imgInput.placeholder = "Image url";
nameInput.placeholder = "Custom name";
codeInput.placeholder = "Custom code";
hideInput.type = "checkbox";
imgInput.value = card.img;
nameInput.value = card.name;
codeInput.value = card.code;
hideInput.checked = card.hidden;
if (card.img && card.img !== "") container.style.background = "linear-gradient(155deg, #1e1e1eeb 20%, #1e1e1ecc), url(\"" + card.img + "\") center / cover no-repeat";
imgInput.addEventListener("change", e => {
setCustomImage(key, e.target.value);
container.style.background = e.target.value === "" ? "var(--containerbg)" : "linear-gradient(155deg, #1e1e1eeb 20%, #1e1e1ecc), url(\"" + e.target.value + "\") center / cover no-repeat";
});
nameInput.addEventListener("change", function (e) { updateCards(key, { "name": e.target.value }) });
codeInput.addEventListener("change", function (e) { updateCards(key, { "code": e.target.value }) });
hideInput.addEventListener("change", function (e) { updateCards(key, { "hidden": e.target.checked }) });
container.querySelector(".custom-card-title").textContent = card.default;
for (let i = 0; i < 4; i++) {
let customLink = makeElement("input", container.querySelector(".custom-links"), { "className": "card-input" });
customLink.value = card_2.links[i].is_default ? "default" : card_2.links[i].path;
customLink.addEventListener("change", function (e) {
chrome.storage.sync.get("custom_cards_2", storage => {
let newLinks = storage.custom_cards_2[key].links;
if (e.target.value === "" || e.target.value === "default") {
console.log("this value is empty....")
//newLinks[i] = { "type": storage.custom_cards_2[key].links.default[i].type, "default": true };
newLinks[i] = { "default": newLinks[i].default, "is_default": true, "path": newLinks[i].default };
customLink.value = "default";
} else {
//newLinks[i] = { "type": getLinkType(e.target.value), "path": e.target.value, "default": false };
let val = e.target.value;
if (!e.target.value.includes("https://") && e.target.value !== "none") val = "https://" + val;
newLinks[i] = { "default": newLinks[i].default, "is_default": false, "path": val };
customLink.value = val;
}
chrome.storage.sync.set({ "custom_cards_2": { ...storage.custom_cards_2, [key]: { ...storage.custom_cards_2[key], "links": newLinks } } })
});
});
}
};
});
} else {
document.querySelector(".advanced-cards").innerHTML = `<div class="option-container"><h3>Couldn't find your cards!<br/>You may need to refresh your Canvas page and/or this menu page.<br/><br/>If you're having issues please contact me - sandlerguy5@gmail.com</h3></div>`;
}
});
}
/*
chrome.runtime.onMessage.addListener(message => {
if (message === "getCardsComplete") {
displayAdvancedCards();
}
});
*/
/*
syncedSwitches.forEach(function (option) {
let optionSwitch = document.querySelector('#' + option);
chrome.storage.sync.get(option, function (result) {
let status = result[option] === true ? "#on" : "#off";
optionSwitch.querySelector(status).checked = true;
optionSwitch.querySelector(status).classList.add('checked');
});
optionSwitch.querySelector(".slider").addEventListener('mouseup', function () {
optionSwitch.querySelector("#on").checked = !optionSwitch.querySelector("#on").checked;
optionSwitch.querySelector("#on").classList.toggle('checked');
optionSwitch.querySelector("#off").classList.toggle('checked');
let status = optionSwitch.querySelector("#on").checked;
chrome.storage.sync.set({ [option]: status });
if (option === "auto_dark") {
toggleDarkModeDisable(status);
}
});
});
*/
/*
localSwitches.forEach(option => {
let optionSwitch = document.querySelector('#' + option);
chrome.storage.local.get(option, function (result) {
let status = result[option] === true ? "#on" : "#off";
optionSwitch.querySelector(status).checked = true;
optionSwitch.querySelector(status).classList.add('checked');
});
optionSwitch.querySelector(".slider").addEventListener('mouseup', function () {
optionSwitch.querySelector("#on").checked = !optionSwitch.querySelector("#on").checked;
optionSwitch.querySelector("#on").classList.toggle('checked');
optionSwitch.querySelector("#off").classList.toggle('checked');
let status = optionSwitch.querySelector("#on").checked;
chrome.storage.local.set({ [option]: status });
/*
switch (option) {
case 'dark_mode': chrome.storage.local.set({ dark_mode: status }); sendFromPopup("darkmode"); break;
}
/
});
});
*/
function toggleDarkModeDisable(disabled) {
let darkSwitch = document.querySelector('#dark_mode');
if (disabled === true) {
darkSwitch.classList.add('switch_disabled');
darkSwitch.style.pointerEvents = "none";
} else {
darkSwitch.classList.remove('switch_disabled');
darkSwitch.style.pointerEvents = "auto";
}
}
// customization tab
function getPalette(name) {
const colors = {
"Blues": ["#ade8f4", "#90e0ef", "#48cae4", "#00b4d8", "#0096c7"],
"Reds": ["#e01e37", "#c71f37", "#b21e35", "#a11d33", "#6e1423"],
"Rainbow": ["#ff0000", "#ff5200", "#efea5a", "#3cf525", "#147df5", "#be0aff"],
"Candy": ["#cdb4db", "#ffc8dd", "#ffafcc", "#bde0fe", "#a2d2ff"],
"Purples": ["#e0aaff", "#c77dff", "#9d4edd", "#7b2cbf", "#5a189a"],
"Pastels": ["#fff1e6", "#fde2e4", "#fad2e1", "#bee1e6", "#cddafd"],
"Ocean": ["#22577a", "#38a3a5", "#57cc99", "#80ed99", "#c7f9cc"],
"Sunset": ["#eaac8b", "#e56b6f", "#b56576", "#6d597a", "#355070"],
"Army": ["#6b705c", "#a5a58d", "#b7b7a4", "#ffe8d6", "#ddbea9", "#cb997e"],
"Pinks": ["#ff0a54", "#ff5c8a", "#ff85a1", "#ff99ac", "#fbb1bd"],
"Watermelon": ["#386641", "#6a994e", "#a7c957", "#f2e8cf", "#bc4749"],
"Popsicle": ["#70d6ff", "#ff70a6", "#ff9770", "#ffd670", "#e9ff70"],
"Chess": ["#ffffff", "#000000"],
"Greens": ["#d8f3dc", "#b7e4c7", "#95d5b2", "#74c69d", "#52b788"],
"Fade": ["#ff69eb", "#ff86c8", "#ffa3a5", "#ffbf81", "#ffdc5e"],
"Oranges": ["#ffc971", "#ffb627", "#ff9505", "#e2711d", "#cc5803"],
"Mesa": ["#f6bd60", "#f28482", "#f5cac3", "#84a59d", "#f7ede2"],
"Berries": ["#4cc9f0", "#4361ee", "#713aed", "#9348c3", "#f72585"],
"Fade2": ["#f2f230", "#C2F261", "#91f291", "#61F2C2", "#30f2f2"],
"Muted": ["#E7E6F7", "#E3D0D8", "#AEA3B0", "#827081", "#C6D2ED"],
"Base": ["#e3b505", "#95190C", "#610345", "#107E7D", "#044B7F"],
"Fruit": ["#7DDF64", "#C0DF85", "#DEB986", "#DB6C79", "#ED4D6E"],
"Night": ["#25171A", "#4B244A", "#533A7B", "#6969B3", "#7F86C6"]
}
return colors[name] || [];
}
function componentToHex(c) {
var hex = c.toString(16);
return hex.length == 1 ? "0" + hex : hex;
}
function getColorInGradient(d, from, to) {
let pat = /^#?([a-f\d]{2})([a-f\d]{2})([a-f\d]{2})$/i;
var exec1 = pat.exec(from);
var exec2 = pat.exec(to);
let a1 = [parseInt(exec1[1], 16), parseInt(exec1[2], 16), parseInt(exec1[3], 16)];
let a2 = [parseInt(exec2[1], 16), parseInt(exec2[2], 16), parseInt(exec2[3], 16)];
let rgb = a1.map((x, i) => Math.floor(a1[i] + d * (a2[i] - a1[i])));
return "#" + componentToHex(rgb[0]) + componentToHex(rgb[1]) + componentToHex(rgb[2]);
}
/*
function getColors(preset) {
console.log(preset)
Object.keys(preset).forEach(key => {
try {
let c = document.querySelector("#dp_" + key);
let color = c.querySelector('input[type="color"]');
let text = c.querySelector('input[type="text"]');
[color, text].forEach(changer => {
changer.value = preset[key];
changer.addEventListener("change", function (e) {
changeCSS(key, e.target.value);
});
});
} catch (e) {
console.log("couldn't get " + key)
console.log(e);
}
});
}
*/
/*
function getColors2(data) {
const colors = data.split(":root")[1].split("--bcstop")[0];
const backgroundcolors = document.querySelector("#option-background");
const textcolors = document.querySelector("#option-text");
colors.split(";").forEach(function (color) {
const type = color.split(":")[0].replace("{", "").replace("}", "");
const currentColor = color.split(":")[1];
if (type) {
let container = makeElement("div", "changer-container", type.includes("background") ? backgroundcolors : textcolors);
let colorChange = makeElement("input", "card-input", container);
let colorChangeText = makeElement("input", "card-input", container);
colorChangeText.type = "text";
colorChangeText.value = currentColor;
colorChange.type = "color";
colorChange.value = currentColor;
[colorChange, colorChangeText].forEach(changer => {
changer.addEventListener("change", function (e) {
changeCSS(type, e.target.value);
});
});
}
})
}
*/
function displaySidebarMode(mode, style) {
style = style.replace(" ", "");
let match = style.match(/linear-gradient\((?<color1>\#\w*),(?<color2>\#\w*)\)/);
let c1 = c2 = "#000000";
if (mode === "image") {
document.querySelector("#radio-sidebar-image").checked = true;
document.querySelector("#sidebar-color2").style.display = "flex";
document.querySelector("#sidebar-image").style.display = "flex";
if (style.includes("url") && match) {
if (match.groups.color1) c1 = match.groups.color1.replace("c7", "");
if (match.groups.color2) c2 = match.groups.color2.replace("c7", "");
}
let url = style.match(/url\(\"(?<url>.*)\"\)/);
document.querySelector('#sidebar-image input[type="text"]').value = url && url.groups.url ? url.groups.url : "";
} else if (mode === "gradient") {
document.querySelector("#radio-sidebar-gradient").checked = true;
document.querySelector("#sidebar-color2").style.display = "flex";
document.querySelector("#sidebar-image").style.display = "none";
if (!style.includes("url") && match) {
if (match.groups.color1) c1 = match.groups.color1;
if (match.groups.color2) c2 = match.groups.color2;
}
} else {
document.querySelector("#radio-sidebar-solid").checked = true;
document.querySelector("#sidebar-color2").style.display = "none";
document.querySelector("#sidebar-image").style.display = "none";
c1 = match ? "#000000" : style;
}
document.querySelector('#sidebar-color1 input[type="text"]').value = c1;
document.querySelector('#sidebar-color1 input[type="color"]').value = c1;
document.querySelector('#sidebar-color2 input[type="text"]').value = c2;
document.querySelector('#sidebar-color2 input[type="color"]').value = c2;
}
let presetChangeTimeout = null;
chrome.storage.sync.get(["dark_preset"], storage => {
let tab = document.querySelector(".customize-dark");
Object.keys(storage["dark_preset"]).forEach(key => {
if (key !== "sidebar") {
let c = tab.querySelector("#dp_" + key);
let color = c.querySelector('input[type="color"]');
let text = c.querySelector('input[type="text"]');
[color, text].forEach(changer => {
changer.value = storage["dark_preset"][key];
changer.addEventListener("input", function (e) {
clearTimeout(presetChangeTimeout);
presetChangeTimeout = setTimeout(() => changeCSS(key, e.target.value), 200);
});
});
} else {
let mode = storage["dark_preset"][key].includes("url") ? "image" : storage["dark_preset"][key].includes("gradient") ? "gradient" : "solid";
displaySidebarMode(mode, storage["dark_preset"][key]);
let changeSidebar = () => {
let c1 = tab.querySelector('#sidebar-color1 input[type="text"]').value.replace("c7", "");
let c2 = tab.querySelector('#sidebar-color2 input[type="text"]').value.replace("c7", "");
let url = tab.querySelector('#sidebar-image input[type="text"]').value;
if (tab.querySelector("#radio-sidebar-image").checked) {
changeCSS(key, `linear-gradient(${c1}c7, ${c2}c7), center url("${url}")`);
} else if (tab.querySelector("#radio-sidebar-gradient").checked) {
changeCSS(key, `linear-gradient(${c1}, ${c2})`);
} else {
changeCSS(key, c1);
}
}
["#sidebar-color1", "#sidebar-color2"].forEach(group => {
['input[type="text"]', 'input[type="color"]'].forEach(input => {
document.querySelector(group + " " + input).addEventListener("input", e => {
['input[type="text"]', 'input[type="color"]'].forEach(i => {
document.querySelector(group + " " + i).value = e.target.value;
});
clearTimeout(presetChangeTimeout);
presetChangeTimeout = setTimeout(() => changeSidebar(), 200);
});
});
});
document.querySelector('#sidebar-image input[type="text"').addEventListener("change", () => changeSidebar());
}
});
});
function refreshColors() {
chrome.storage.sync.get(["dark_preset"], storage => {
Object.keys(storage["dark_preset"]).forEach(key => {
let c = document.querySelector("#dp_" + key);
let color = c.querySelector('input[type="color"]');
let text = c.querySelector('input[type="text"]');
color.value = storage["dark_preset"][key];
text.value = storage["dark_preset"][key];
});
let mode = storage["dark_preset"]["sidebar"].includes("url") ? "image" : storage["dark_preset"]["sidebar"].includes("gradient") ? "gradient" : "solid";
displaySidebarMode(mode, storage["dark_preset"]["sidebar"]);
});
}
function changeCSS(name, color) {
chrome.storage.sync.get("dark_preset", storage => {
storage["dark_preset"][name] = color;
chrome.storage.sync.set({ "dark_preset": storage["dark_preset"] }).then(() => refreshColors());
});
}
function changeToPresetCSS(e, preset = null) {
const presets = {
"dark-lighter": { "background-0": "#272727", "background-1": "#353535", "background-2": "#404040", "borders": "#454545", "sidebar": "#353535", "text-0": "#f5f5f5", "text-1": "#e2e2e2", "text-2": "#ababab", "links": "#56Caf0", "sidebar-text": "#f5f5f5" },
"dark-light": { "background-0": "#202020", "background-1": "#2e2e2e", "background-2": "#4e4e4e", "borders": "#404040", "sidebar": "#2e2e2e", "text-0": "#f5f5f5", "text-1": "#e2e2e2", "text-2": "#ababab", "links": "#56Caf0", "sidebar-text": "#f5f5f5" },
"dark-default": { "background-0": "#161616", "background-1": "#1e1e1e", "background-2": "#262626", "borders": "#3c3c3c", "text-0": "#f5f5f5", "text-1": "#e2e2e2", "text-2": "#ababab", "links": "#56Caf0", "sidebar": "#1e1e1e", "sidebar-text": "#f5f5f5" },
"dark-dark": { "background-0": "#101010", "background-1": "#121212", "background-2": "#1a1a1a", "borders": "#272727", "sidebar": "#121212", "text-0": "#f5f5f5", "text-1": "#e2e2e2", "text-2": "#ababab", "links": "#56Caf0", "sidebar-text": "#f5f5f5" },
"dark-darker": { "background-0": "#000000", "background-1": "#000000", "background-2": "#000000", "borders": "#000000", "sidebar": "#000000", "text-0": "#c5c5c5", "text-1": "#c5c5c5", "text-2": "#c5c5c5", "links": "#c5c5c5", "sidebar-text": "#c5c5c5" },
"dark-blue": { "background-0": "#14181d", "background-1": "#1a2026", "background-2": "#212930", "borders": "#2e3943", "sidebar": "#1a2026", "text-0": "#f5f5f5", "text-1": "#e2e2e2", "text-2": "#ababab", "links": "#56Caf0", "sidebar-text": "#f5f5f5" },
"dark-mint": { "background-0": "#0f0f0f", "background-1": "#0c0c0c", "background-2": "#141414", "borders": "#1e1e1e", "sidebar": "#0c0c0c", "text-0": "#f5f5f5", "text-1": "#e2e2e2", "text-2": "#ababab", "links": "#7CF3CB", "sidebar-text": "#f5f5f5" },
"dark-burn": { "background-0": "#ffffff", "background-1": "#ffffff", "background-2": "#ffffff", "borders": "#cccccc", "sidebar": "#ffffff", "text-0": "#cccccc", "text-1": "#cccccc", "text-2": "#cccccc", "links": "#cccccc", "sidebar-text": "#cccccc" },
"dark-unicorn": { "background-0": "#ff6090", "background-1": "#00C1FF", "background-2": "#FFFF00", "borders": "#FFFF00", "sidebar": "#00C1FF", "text-0": "#ffffff", "text-1": "#ffffff", "text-2": "#ffffff", "links": "#000000", "sidebar-text": "#ffffff" },
"dark-lightmode": { "background-0": "#ffffff", "background-1": "#f5f5f5", "background-2": "#d4d4d4", "borders": "#c7cdd1", "links": "#04ff00", "sidebar": "#04ff00", "sidebar-text": "#ffffff", "text-0": "#2d3b45", "text-1": "#919191", "text-2": "#a5a5a5" },
"dark-catppuccin": { "background-0": "#11111b", "background-1": "#181825", "background-2": "#1e1e2e", "borders": "#4f5463", "text-0": "#cdd6f4", "text-1": "#7f849c", "text-2": "#a6e3a1", "links": "#f5c2e7", "sidebar": "#181825", "sidebar-text": "#7f849c" },
"dark-sage": { "background-0": "#2f3e46", "background-1": "#354f52", "background-2": "#52796f", "borders": "#84a98c", "links": "#d8f5c7", "sidebar": "#354f52", "sidebar-text": "#e2e8de", "text-0": "#e2e8de", "text-1": "#cad2c5", "text-2": "#adb1aa" },
"dark-pink": { "background-0": "#ffffff", "background-1": "#ffe0ed", "background-2": "#ff0066", "borders": "#ff007b", "links": "#ff0088", "sidebar": "#f490b3", "sidebar-text": "#ffffff", "text-0": "#ff0095", "text-1": "#ff8f8f", "text-2": "#ff5c5c" },
"dark-coral": {"background-0":"#131c26","background-1":"#0e1721","background-2":"#151c24","borders":"#0e1721","links":"#f88379","sidebar":"#131c26","sidebar-text":"#f88379","text-0":"#f88379","text-1":"#f88379","text-2":"#f88379"},
}
if (preset === null) preset = presets[e.target.id] || presets["default"];
applyPreset(preset);
}
function applyPreset(preset) {
chrome.storage.sync.set({ "dark_preset": preset }).then(() => refreshColors());
}
/*
function setToDefaults() {
fetch(chrome.runtime.getURL('js/darkcss.json'))
.then((resp) => resp.json())
.then(function (result) {
chrome.storage.local.set({ "dark_css": result["dark_css"], "dark_preset": { "background-0": "#161616", "background-1": "#1e1e1e", "background-2": "#262626", "borders": "#3c3c3c", "text-0": "#f5f5f5", "text-1": "#e2e2e2", "text-2": "#ababab", "links": "#56Caf0", "sidebar": "#1e1e1e", "sidebar-text": "#f5f5f5" } }).then(() => refreshColors());
});
}
*/
/*
function makeElement(element, elclass, location, text) {
let creation = document.createElement(element);
creation.className = elclass;
creation.textContent = text;
location.appendChild(creation);
return creation
}*/
function makeElement(element, location, options) {
let creation = document.createElement(element);
Object.keys(options).forEach(key => {
creation[key] = options[key];
});
location.appendChild(creation);
return creation
}
async function sendFromPopup(message, options = {}) {
let response = new Promise((resolve, reject) => {
chrome.tabs.query({ currentWindow: true }).then(async tabs => {
for (let i = 0; i < tabs.length; i++) {
try {
let res = await chrome.tabs.sendMessage(tabs[i].id, { "message": message, "options": options });
if (res) resolve(res);
} catch (e) {
}
}
resolve(null);
});
})
return await response;
}