class SettingsGroup { constructor( configName, toggleFunction, setCallback = null, updateCallback = null ) { this.configName = configName; this.configValue = config[configName]; if (this.configValue === true || this.configValue === false) { this.onOff = true; } else { this.onOff = false; } this.toggleFunction = toggleFunction; this.setCallback = setCallback; this.updateCallback = updateCallback; this.updateButton(); $(document).on( "click", `.pageSettings .section.${this.configName} .button`, (e) => { if (this.onOff) { if ($(e.currentTarget).hasClass("on")) { this.toggleFunction(true); } else { this.toggleFunction(false); } this.updateButton(); if (this.setCallback !== null) this.setCallback(); } else { let value = $(e.currentTarget).attr(configName); let params = $(e.currentTarget).attr("params"); if (params === undefined) { this.toggleFunction(value); } else { this.toggleFunction(value, ...params); } this.updateButton(); if (this.setCallback !== null) this.setCallback(); } } ); } updateButton() { this.configValue = config[this.configName]; $(`.pageSettings .section.${this.configName} .button`).removeClass( "active" ); if (this.onOff) { let onoffstring; if (this.configValue) { onoffstring = "on"; } else { onoffstring = "off"; } $( `.pageSettings .section.${this.configName} .buttons .button.${onoffstring}` ).addClass("active"); } else { $( `.pageSettings .section.${this.configName} .button[${this.configName}='${this.configValue}']` ).addClass("active"); } if (this.updateCallback !== null) this.updateCallback(); } } let settingsGroups = {}; settingsGroups.smoothCaret = new SettingsGroup("smoothCaret", setSmoothCaret); settingsGroups.difficulty = new SettingsGroup("difficulty", setDifficulty); settingsGroups.quickTab = new SettingsGroup("quickTab", setQuickTabMode); settingsGroups.showLiveWpm = new SettingsGroup( "showLiveWpm", setShowLiveWpm, () => { settingsGroups.keymapMode.updateButton(); } ); settingsGroups.showTimerProgress = new SettingsGroup( "showTimerProgress", setShowTimerProgress ); settingsGroups.keymapMode = new SettingsGroup( "keymapMode", changeKeymapMode, () => { settingsGroups.showLiveWpm.updateButton(); }, () => { if (config.keymapMode === "off") { $(".pageSettings .section.keymapStyle").addClass("hidden"); $(".pageSettings .section.keymapLayout").addClass("hidden"); } else { $(".pageSettings .section.keymapStyle").removeClass("hidden"); $(".pageSettings .section.keymapLayout").removeClass("hidden"); } } ); settingsGroups.keymapMatrix = new SettingsGroup( "keymapStyle", changeKeymapStyle ); settingsGroups.keymapLayout = new SettingsGroup( "keymapLayout", changeKeymapLayout ); settingsGroups.showKeyTips = new SettingsGroup( "showKeyTips", setKeyTips, null, () => { if (config.showKeyTips) { $(".pageSettings .tip").removeClass("hidden"); } else { $(".pageSettings .tip").addClass("hidden"); } } ); settingsGroups.freedomMode = new SettingsGroup( "freedomMode", setFreedomMode, () => { settingsGroups.confidenceMode.updateButton(); } ); settingsGroups.confidenceMode = new SettingsGroup( "confidenceMode", setConfidenceMode, () => { settingsGroups.freedomMode.updateButton(); settingsGroups.stopOnError.updateButton(); } ); settingsGroups.indicateTypos = new SettingsGroup("indicateTypos", setIndicateTypos); settingsGroups.blindMode = new SettingsGroup("blindMode", setBlindMode); settingsGroups.quickEnd = new SettingsGroup("quickEnd", setQuickEnd); // settingsGroups.readAheadMode = new SettingsGroup( // "readAheadMode", // setReadAheadMode // ); settingsGroups.alwaysShowWordsHistory = new SettingsGroup( "alwaysShowWordsHistory", setAlwaysShowWordsHistory ); settingsGroups.flipTestColors = new SettingsGroup( "flipTestColors", setFlipTestColors ); settingsGroups.swapEscAndTab = new SettingsGroup( "swapEscAndTab", setSwapEscAndTab ); settingsGroups.showOutOfFocusWarning = new SettingsGroup( "showOutOfFocusWarning", setShowOutOfFocusWarning ); settingsGroups.colorfulMode = new SettingsGroup( "colorfulMode", setColorfulMode ); settingsGroups.startGraphsAtZero = new SettingsGroup( "startGraphsAtZero", setStartGraphsAtZero ); settingsGroups.randomTheme = new SettingsGroup("randomTheme", setRandomTheme); settingsGroups.stopOnError = new SettingsGroup( "stopOnError", setStopOnError, () => { settingsGroups.confidenceMode.updateButton(); } ); settingsGroups.playSoundOnError = new SettingsGroup( "playSoundOnError", setPlaySoundOnError ); settingsGroups.playSoundOnClick = new SettingsGroup( "playSoundOnClick", setPlaySoundOnClick, () => { if (config.playSoundOnClick !== "off") // clickSounds[config.playSoundOnClick][0].sounds[0].play(); playClickSound(); } ); settingsGroups.showAllLines = new SettingsGroup( "showAllLines", setShowAllLines ); settingsGroups.paceCaret = new SettingsGroup("paceCaret", setPaceCaret, () => { if (config.paceCaret === "custom") { $( ".pageSettings .section.paceCaret input.customPaceCaretSpeed" ).removeClass("hidden"); } else { $(".pageSettings .section.paceCaret input.customPaceCaretSpeed").addClass( "hidden" ); } }); settingsGroups.smoothLineScroll = new SettingsGroup( "smoothLineScroll", setSmoothLineScroll ); settingsGroups.capsLockBackspace = new SettingsGroup( "capsLockBackspace", setCapsLockBackspace ); settingsGroups.layout = new SettingsGroup("layout", changeLayout); settingsGroups.language = new SettingsGroup("language", changeLanguage); settingsGroups.fontSize = new SettingsGroup("fontSize", changeFontSize); settingsGroups.caretStyle = new SettingsGroup("caretStyle", setCaretStyle); settingsGroups.paceCaretStyle = new SettingsGroup("paceCaretStyle", setPaceCaretStyle); settingsGroups.timerStyle = new SettingsGroup("timerStyle", setTimerStyle); settingsGroups.timerOpacity = new SettingsGroup( "timerOpacity", setTimerOpacity ); settingsGroups.timerColor = new SettingsGroup("timerColor", setTimerColor); settingsGroups.fontFamily = new SettingsGroup("fontFamily", setFontFamily); settingsGroups.alwaysShowDecimalPlaces = new SettingsGroup( "alwaysShowDecimalPlaces", setAlwaysShowDecimalPlaces ); fillSettingsPage(); async function fillSettingsPage() { refreshThemeButtons(); let langEl = $(".pageSettings .section.language .buttons").empty(); Object.keys(words).forEach((language) => { if (language === "english_10k") return; langEl.append( `
` ); if (language === "english_expanded") { langEl.append( ` ` ); } }); let layoutEl = $(".pageSettings .section.layout .buttons").empty(); Object.keys(layouts).forEach((layout) => { layoutEl.append( ` ` ); }); let keymapEl = $(".pageSettings .section.keymapLayout .buttons").empty(); Object.keys(layouts).forEach((layout) => { if (layout.toString() != "default") { keymapEl.append( ` ` ); } }); let funboxEl = $(".pageSettings .section.funbox .buttons").empty(); funboxEl.append(` `); getFunboxList().then((funboxModes) => { funboxModes.forEach((funbox) => { if (funbox.name === "mirror") { funboxEl.append( ` ` ); } else { funboxEl.append( ` ` ); } }); }); let fontsEl = $(".pageSettings .section.fontFamily .buttons").empty(); getFontsList().then((fonts) => { fonts.forEach((font) => { fontsEl.append( ` ` ); }); }); } function refreshThemeButtons() { let favThemesEl = $( ".pageSettings .section.themes .favThemes.buttons" ).empty(); let themesEl = $(".pageSettings .section.themes .allThemes.buttons").empty(); let activeThemeName = config.theme; if (config.randomTheme !== "off" && randomTheme !== null) { activeThemeName = randomTheme; } getSortedThemesList().then((themes) => { //first show favourites if (config.favThemes.length > 0) { favThemesEl.css({ paddingBottom: "1rem" }); themes.forEach((theme) => { if (config.favThemes.includes(theme.name)) { let activeTheme = activeThemeName === theme.name ? "active" : ""; favThemesEl.append( ` ` ); } }); } else { favThemesEl.css({ paddingBottom: "0" }); } //then the rest themes.forEach((theme) => { if (!config.favThemes.includes(theme.name)) { let activeTheme = activeThemeName === theme.name ? "active" : ""; themesEl.append( ` ` ); } }); }); } function updateSettingsPage() { Object.keys(settingsGroups).forEach((group) => { settingsGroups[group].updateButton(); }); refreshTagsSettingsSection(); setActiveFunboxButton(); setActiveThemeButton(); setActiveThemeTab(); setCustomThemeInputs(); updateDiscordSettingsSection(); refreshThemeButtons(); if (config.paceCaret === "custom") { $( ".pageSettings .section.paceCaret input.customPaceCaretSpeed" ).removeClass("hidden"); $( ".pageSettings .section.paceCaret input.customPaceCaretSpeed" ).val(config.paceCaretCustomSpeed); } else { $(".pageSettings .section.paceCaret input.customPaceCaretSpeed").addClass( "hidden" ); } } function showCustomThemeShare() { if ($("#customThemeShareWrapper").hasClass("hidden")) { let save = []; $.each( $(".pageSettings .section.customTheme [type='color']"), (index, element) => { save.push($(element).attr("value")); } ); $("#customThemeShareWrapper input").val(JSON.stringify(save)); $("#customThemeShareWrapper") .stop(true, true) .css("opacity", 0) .removeClass("hidden") .animate({ opacity: 1 }, 100, (e) => { $("#customThemeShare input").focus(); $("#customThemeShare input").select(); $("#customThemeShare input").focus(); }); } } function hideCustomThemeShare() { if (!$("#customThemeShareWrapper").hasClass("hidden")) { try { config.customThemeColors = JSON.parse( $("#customThemeShareWrapper input").val() ); } catch (e) { showNotification( "Something went wrong. Reverting to default custom colors.", 3000 ); config.customThemeColors = defaultConfig.customThemeColors; } setCustomThemeInputs(); applyCustomThemeColors(); $("#customThemeShareWrapper input").val(""); $("#customThemeShareWrapper") .stop(true, true) .css("opacity", 1) .animate( { opacity: 0, }, 100, (e) => { $("#customThemeShareWrapper").addClass("hidden"); } ); } } $("#customThemeShareWrapper").click((e) => { if ($(e.target).attr("id") === "customThemeShareWrapper") { hideCustomThemeShare(); } }); $("#customThemeShare .button").click((e) => { hideCustomThemeShare(); }); $("#shareCustomThemeButton").click((e) => { // showCustomThemeShare(); if (e.shiftKey) { showCustomThemeShare(); } else { let share = []; $.each( $(".pageSettings .section.customTheme [type='color']"), (index, element) => { share.push($(element).attr("value")); } ); let url = "https://monkey-type.com?" + objectToQueryString({ customTheme: share }); navigator.clipboard.writeText(url).then(function () { showNotification("URL Copied to clipboard", 2000); }, function (err) { showNotification("Something went wrong when copying the URL: " + err, 5000); }); } }); function toggleFavouriteTheme(themename) { if (config.favThemes.includes(themename)) { //already favourite, remove config.favThemes = config.favThemes.filter((t) => { if (t !== themename) { return t; } }); } else { //add to favourites config.favThemes.push(themename); } saveConfigToCookie(); refreshThemeButtons(); } function showAccountSettingsSection() { $(`.sectionGroupTitle[group='account']`).removeClass("hidden"); $(`.settingsGroup.account`).removeClass("hidden"); refreshTagsSettingsSection(); updateDiscordSettingsSection(); } function hideAccountSettingsSection() { $(`.sectionGroupTitle[group='account']`).addClass("hidden"); $(`.settingsGroup.account`).addClass("hidden"); } function refreshTagsSettingsSection() { if (firebase.auth().currentUser !== null && dbSnapshot !== null) { let tagsEl = $(".pageSettings .section.tags .tagsList").empty(); dbSnapshot.tags.forEach((tag) => { if (tag.active === true) { tagsEl.append(`