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']; 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" }, } }; 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 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'], "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) } ], } 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; } } } 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 } }); }); } 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; 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 += ``; 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 = `
Nothing here
`; } 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 = 'https://fonts.googleapis.com/css2?family='; 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 += '
%GPA
'; 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 = '

Past Courses

'; 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 = '

Hide

Image
Name
Code
'; 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 = `

Couldn't find your cards!
You may need to refresh your Canvas page and/or this menu page.

If you're having issues please contact me - sandlerguy5@gmail.com

`; } }); } /* 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\((?\#\w*),(?\#\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\(\"(?.*)\"\)/); 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; }