// ==UserScript== // @name [Pokeclicker] Additional Visual Settings // @namespace Pokeclicker Scripts // @author Ephenia (Credit: Optimatum) // @description Adds additional settings for hiding some visual things to help out with performance. Also, includes various features that help with ease of accessibility. // @copyright https://github.com/Ephenia // @license GPL-3.0 License // @version 3.0 // @homepageURL https://github.com/Ephenia/Pokeclicker-Scripts/ // @supportURL https://github.com/Ephenia/Pokeclicker-Scripts/issues // @match https://www.pokeclicker.com/ // @icon https://www.google.com/s2/favicons?domain=pokeclicker.com // @grant unsafeWindow // @run-at document-idle // @downloadURL https://update.greasyfork.icu/scripts/434130/%5BPokeclicker%5D%20Additional%20Visual%20Settings.user.js // @updateURL https://update.greasyfork.icu/scripts/434130/%5BPokeclicker%5D%20Additional%20Visual%20Settings.meta.js // ==/UserScript== // TODO disable party attack number + tooltip class AdditionalVisualSettings { static graphicsDisabledSettings = { route: { header: ko.observable(false), pokemon: ko.observable(false), catchIcon: ko.observable(false), healthbar: ko.observable(false), attack: ko.observable(false), }, gym: { header: ko.observable(false), timer: ko.observable(false), pokemon: ko.observable(false), healthbar: ko.observable(false), attack: ko.observable(false), }, dungeon: { header: ko.observable(false), timer: ko.observable(false), images: ko.observable(false), attack: ko.observable(false), }, battleFrontier: { header: ko.observable(false), timer: ko.observable(false), pokemon: ko.observable(false), healthbar: ko.observable(false), }, }; static autoClickerIntegration = ko.observable(JSON.parse(localStorage.getItem('AVSautoClickerIntegration') || 'false')); // Disable graphics unless autoclicker integration is on and autoclicker is not running static graphicsSettingsActive = ko.computed({ read: () => !(typeof EnhancedAutoClicker === 'function' && this.autoClickerIntegration() && !EnhancedAutoClicker.autoClickState()), deferEvaluation: true }); static loadGraphicsSettings() { try { const savedSettings = JSON.parse(localStorage.getItem('AVSgraphicsDisabledSettings') || '{}'); Object.keys(this.graphicsDisabledSettings).forEach(state => { Object.keys(this.graphicsDisabledSettings[state]).forEach(setting => { if (savedSettings[state]?.[setting] != undefined) { const val = !!savedSettings[state][setting]; this.graphicsDisabledSettings[state][setting](val); } }); }); } catch { this.saveGraphicsSettings(); } } static saveGraphicsSettings() { const settingsToSave = {}; Object.keys(this.graphicsDisabledSettings).forEach(state => { settingsToSave[state] = {}; Object.keys(this.graphicsDisabledSettings[state]).forEach(setting => { settingsToSave[state][setting] = this.graphicsDisabledSettings[state][setting](); }); }); localStorage.setItem('AVSgraphicsDisabledSettings', JSON.stringify(settingsToSave)); } static initOnLoad() { this.addGraphicsBindings(); this.addOptimizeVitamins(); } static initVisualSettings() { this.loadGraphicsSettings(); // Add shortcut menu icons const getMenu = document.getElementById('startMenu'); const shortcutsToAdd = [ ['quick-settings', '#settingsModal', ''], ['quick-inventory', '#showItemsModal', ''], ['quick-pokedex', '#pokedexModal', ''], ]; shortcutsToAdd.forEach(([id, modal, source]) => { const quickElem = document.createElement('img'); quickElem.id = id; quickElem.src = source; quickElem.setAttribute('href', modal); quickElem.setAttribute('data-toggle', 'modal'); getMenu.prepend(quickElem); }); // Add AVS settings options to scripts tab const settingsBody = createScriptSettingsContainer('Additional Visual Settings'); let elem = document.createElement('tr'); elem.innerHTML = ``; settingsBody.appendChild(elem); // Graphics-disabling settings Object.keys(this.graphicsDisabledSettings).forEach(state => { elem = document.createElement('tr'); elem.innerHTML = `${GameConstants.camelCaseToString(state)}`; let innerElem = elem.querySelector('td'); Object.keys(this.graphicsDisabledSettings[state]).forEach(setting => { const container = document.createElement('div'); container.className = 'px-3' container.innerHTML = `${GameConstants.camelCaseToString(setting)} `; const checkbox = container.querySelector('input'); checkbox.checked = this.graphicsDisabledSettings[state][setting](); checkbox.addEventListener('change', event => { this.graphicsDisabledSettings[state][setting](event.target.checked); this.saveGraphicsSettings(); }); innerElem.appendChild(container); }); settingsBody.appendChild(elem); }); // EnhancedAutoClicker integration setting, if the script is present if (typeof EnhancedAutoClicker === 'function') { elem = document.createElement('tr'); elem.innerHTML = `` + ``; settingsBody.appendChild(elem); const checkbox = elem.querySelector('input'); checkbox.checked = this.autoClickerIntegration(); checkbox.addEventListener('change', event => { this.autoClickerIntegration(event.target.checked); localStorage.setItem('AVSautoClickerIntegration', this.autoClickerIntegration()); }); } // Create travel shortcut buttons on town map const travelShortcutsToAdd = [ ['dock-button', 'Dock', {left: 32, top: 0}, MapHelper.openShipModal], ['gyms-button', 'Gyms', {left: 75, top: -8}, () => { AdditionalVisualSettings.generateRegionGymsList(); $('#gymsShortcutModal').modal('show'); }], ['dungeons-button', 'Dungeons', {left: 121, top: -8}, () => { AdditionalVisualSettings.generateRegionDungeonssList(); $('#dungeonsShortcutModal').modal('show'); }], ]; travelShortcutsToAdd.forEach(([id, name, pos, func]) => { const button = document.createElement('button'); button.id = id; button.textContent = name; button.className = 'btn btn-block btn-success'; button.style = `position: absolute; left: ${pos.left}px; top: ${pos.top}px; width: auto; height: 41px; font-size: 11px;`; button.addEventListener('click', func); document.getElementById('townMap').appendChild(button); }); // Prevent ship modal sequence-breaking document.getElementById('dock-button').setAttribute('data-bind', 'enabled: TownList[GameConstants.DockTowns[player.region]].isUnlocked()'); ko.applyBindings(App.game, document.getElementById('dock-button')); // Create gym and dungeon shortcut modals const modalNames = ['gyms', 'dungeons']; const fragment = new DocumentFragment(); for (const name of modalNames) { const customModal = document.createElement('div'); customModal.setAttribute('class', 'modal noselect fade'); customModal.setAttribute('tabindex', '-1'); customModal.setAttribute('role', 'dialogue'); customModal.setAttribute('id', `${name}ShortcutModal`); customModal.setAttribute('aria-labelledby', `${name}ShortcutModalLabel`); customModal.innerHTML = ``; fragment.appendChild(customModal); } document.getElementById('ShipModal').after(fragment); addGlobalStyle('.pageItemTitle { height:38px }'); addGlobalStyle('#quick-settings, #quick-inventory, #quick-pokedex { height: 36px; background-color: #eee; border: 4px solid #eee; cursor: pointer; image-rendering: pixelated; }'); addGlobalStyle('#quick-pokedex { padding: 2px; }') addGlobalStyle(':is(#quick-settings, #quick-inventory, #quick-pokedex):hover { background-color:#ddd; border: 4px solid #ddd; }'); addGlobalStyle('#shortcutsContainer { display: block !important; }'); addGlobalStyle('.gyms-shortcut-leaders { display: flex; pointer-events: none; position: absolute; height: 36px; top: 0; left: 0; image-rendering: pixelated; }'); addGlobalStyle('.gyms-shortcut-badges { position: absolute; height: 36px; display: flex; top: 0; right: 0; }'); addGlobalStyle('.dungeons-shortcut-costs { position: relative; margin-right: 12px; filter: none !important }'); addGlobalStyle('#dungeons-shortcut-buttons > button:hover { -webkit-animation: bounceBackground 60s linear infinite alternate; animation: bounceBackground 60s linear infinite alternate; }'); addGlobalStyle('#dungeons-shortcut-buttons > button * { z-index: 2 }'); addGlobalStyle('.dungeons-shortcut-overlay { width: 100%; height: 100%; position: absolute; background-color: rgba(0,0,0,0.45); margin-top: -6px; margin-left: -8px; z-index: 1 !important }'); addGlobalStyle('.dungeons-shortcut-info { position: relative; font-weight: bold }'); } static generateRegionGymsList() { const gymsBtns = document.getElementById('gyms-shortcut-buttons'); const gymsHead = document.getElementById('gyms-shortcut-modal-title'); gymsHead.textContent = `Gym Select (${GameConstants.camelCaseToString(GameConstants.Region[player.region])})`; gymsBtns.innerHTML = ''; const fragment = new DocumentFragment(); const regionGyms = Object.values(GymList).filter((gym) => gym.parent?.region === player.region); for (const gym of regionGyms) { const hasBadgeImage = !(BadgeEnums[gym.badgeReward].startsWith('Elite') || BadgeEnums[gym.badgeReward] == 'None'); const badgeImage = (hasBadgeImage ? `assets/images/badges/${BadgeEnums[gym.badgeReward]}.svg` : ''); const btn = document.createElement('button'); btn.setAttribute('style', 'position: relative;'); btn.setAttribute('class', 'btn btn-block btn-success'); btn.addEventListener('click', () => { if (!MapHelper.isTownCurrentLocation(gym.parent.name)) { MapHelper.moveToTown(gym.parent.name); } $("#gymsShortcutModal").modal("hide"); GymRunner.startGym(gym); }); btn.disabled = !(gym.isUnlocked() && gym.parent.isUnlocked()); btn.innerHTML = `
${gym.leaderName}`; fragment.appendChild(btn); } gymsBtns.appendChild(fragment); } static generateRegionDungeonssList() { const dungeonsBtns = document.getElementById('dungeons-shortcut-buttons'); const dungeonsHead = document.getElementById('dungeons-shortcut-modal-title'); dungeonsHead.textContent = `Dungeon Select (${GameConstants.camelCaseToString(GameConstants.Region[player.region])})`; dungeonsBtns.innerHTML = ''; const fragment = new DocumentFragment(); const dungeonTowns = Object.values(TownList).filter((town) => (town.region === player.region && town.constructor.name === 'DungeonTown' && town.dungeon != null)); for (const town of dungeonTowns) { const dungeon = town.dungeon; const dungeonClears = App.game.statistics.dungeonsCleared[GameConstants.getDungeonIndex(dungeon.name)](); const canAffordEntry = App.game.wallet.currencies[GameConstants.Currency.dungeonToken]() >= dungeon.tokenCost; const canAccess = town.isUnlocked() && dungeon.isUnlocked() && canAffordEntry; const btn = document.createElement('button'); btn.setAttribute('style', `position: relative; background-image: url("assets/images/towns/${dungeon.name}.png"); background-position: center;opacity: ${canAccess ? 1 : 0.70}; filter: brightness(${canAccess ? 1 : 0.70});`); btn.setAttribute('class', 'btn btn-block btn-success'); btn.addEventListener('click', () => { if (!MapHelper.isTownCurrentLocation(town.name)) { MapHelper.moveToTown(town.name); } $('#dungeonsShortcutModal').modal('hide'); DungeonRunner.initializeDungeon(dungeon); }); btn.disabled = !canAccess; btn.innerHTML = `
${dungeon.tokenCost.toLocaleString('en-US')}
${dungeon.name}
${dungeonClears.toLocaleString('en-US')} clears
`; fragment.appendChild(btn); } dungeonsBtns.appendChild(fragment); } // Must execute before game loads and applies knockout bindings static addGraphicsBindings() { function selectorWorkaround(element, selector) { try { return element.querySelector(selector); } catch { const [, outer, inner] = selector.match(/(.+):has\((.+)\)/); const innerElem = element.querySelector(`${outer} ${inner}`); return Array.from(element.querySelectorAll(outer)).find(e => e.contains(innerElem)); } } const selectors = { route: { container: '#routeBattleContainer', header: '.pageItemTitle > div:has(> knockout)', pokemon: 'div:has(> knockout[data-bind*="pokemonSpriteTemplate"])', catchIcon: 'div.catchChance', healthbar: 'div.progress.hitpoints', attack: '.pageItemFooter knockout[data-bind*="pokemonAttackTemplate"]', }, gym: { container: '#battleContainer div[data-bind="if: App.game.gameState === GameConstants.GameState.gym"]', header: [ ['h2.pageItemTitle > knockout:has(knockout[data-bind*="pokemonNameTemplate"])', 'before'], ['h2.pageItemTitle > knockout:has(span[data-bind*="pokemonsDefeatedComputable"])', 'after'] ], timer: 'h2.pageItemTitle .timer', pokemon: 'div:has(> knockout[data-bind*="pokemonSpriteTemplate"])', healthbar: 'div.progress.hitpoints', attack: '.pageItemFooter knockout[data-bind*="pokemonAttackTemplate"]', }, dungeon: { container: '#battleContainer div[data-bind="if: App.game.gameState === GameConstants.GameState.dungeon"]', header: [ ['h2.pageItemTitle > knockout:has(knockout[data-bind*="pokemonNameTemplate"])', 'before'], ['h2.pageItemTitle > knockout:has(span[data-bind*="defeatedTrainerPokemon"])', 'after'] ], timer: 'h2.pageItemTitle .timer', images: [ ['h2.pageItemTitle', 'after'], ['h2.pageItemFooter', 'before'] ], attack: '.pageItemFooter knockout[data-bind*="pokemonAttackTemplate"]', }, battleFrontier: { container: '#battleContainer div[data-bind="if: App.game.gameState == GameConstants.GameState.battleFrontier"]', header: [ ['h2.pageItemTitle > knockout:has(knockout[data-bind*="pokemonNameTemplate"])', 'before'], ['h2.pageItemTitle > knockout[data-bind*="pokemonLeftImages"]', 'after'] ], timer: 'h2.pageItemTitle .timer', pokemon: 'div:has(> knockout[data-bind*="pokemonSpriteTemplate"])', healthbar: 'div.progress.hitpoints', }, }; Object.keys(this.graphicsDisabledSettings).forEach(state => { const container = document.querySelector(selectors[state]?.container) if (!container) { console.error(`AVS: could not find ${state} container`); return; } Object.keys(this.graphicsDisabledSettings[state]).forEach(setting => { if (!selectors[state]?.[setting]) { return; } const selector = selectors[state][setting]; const binding = `ko ifnot: AdditionalVisualSettings.graphicsDisabledSettings.${state}.${setting}() && AdditionalVisualSettings.graphicsSettingsActive()`; // Add binding for this setting if (Array.isArray(selector)) { // For binding multiple elements at once, which requires more complicated selecting selector.forEach(([query, order], i) => { const elem = selectorWorkaround(container, query); const commentBinding = i % 2 == 0 ? binding : '/ko'; if (order == 'before') { elem.before(new Comment(commentBinding)); } else { elem.after(new Comment(commentBinding)); } }); } else { const elem = selectorWorkaround(container, selector); // Special case: insert a backup attack-disabled element so formatting doesn't look weird // Do this before applying the main binding to put it outside the binding comments if (setting == 'attack') { const replacementAttack = document.createElement('span'); elem.after(replacementAttack); replacementAttack.outerHTML = `Pokémon Attack: -----`; } // Insert the binding elem.before(new Comment(binding)); elem.after(new Comment('/ko')); } }); }); } static addOptimizeVitamins() { // Add button to vitamin menu // (must execute before game loads and applies knockout bindings) const btn = document.createElement('button'); btn.setAttribute('class', 'btn btn-link btn-sm text-decoration-none align-text-top'); btn.setAttribute('style', 'line-height: 0.6; font-size: 1rem; float: right;'); btn.setAttribute('data-bind', `click: () => { if ($data) { $data.optimizeVitamins() } }, class: (!$data.breeding ? 'text-success' : 'text-muted')`); btn.innerHTML = '⚖'; document.querySelector('#pokemonVitaminExpandedModal tbody[data-bind*="PartyController.getVitaminSortedList"] td').appendChild(btn); // Add optimize-vitamin functions for party pokemon (adapted from wiki) PartyPokemon.prototype.calcBreedingEfficiency = function(vitaminsUsed) { // attack bonus const attackBonusPercent = (GameConstants.BREEDING_ATTACK_BONUS + vitaminsUsed[GameConstants.VitaminType.Calcium]) / 100; const proteinBoost = vitaminsUsed[GameConstants.VitaminType.Protein]; const breedingAttackBonus = (this.baseAttack * attackBonusPercent) + proteinBoost; // egg steps const div = 300; const extraCycles = (vitaminsUsed[GameConstants.VitaminType.Calcium] + vitaminsUsed[GameConstants.VitaminType.Protein]) / 2; const steps = (this.eggCycles + extraCycles) * GameConstants.EGG_CYCLE_MULTIPLIER; const adjustedSteps = (steps <= div ? steps : Math.round(((steps / div) ** (1 - vitaminsUsed[GameConstants.VitaminType.Carbos] / 70)) * div)); // efficiency return (breedingAttackBonus / adjustedSteps) * GameConstants.EGG_CYCLE_MULTIPLIER; } PartyPokemon.prototype.optimizeVitamins = function() { const totalVitamins = (player.highestRegion() + 1) * 5; const carbosUnlocked = player.highestRegion() >= GameConstants.Region.unova; const calciumUnlocked = player.highestRegion() >= GameConstants.Region.hoenn; const prices = GameHelper.enumStrings(GameConstants.VitaminType).map(v => ItemList[v].basePrice); // Add our initial starting efficiency here let optimalVitamins = [0, 0, 0]; let eff = this.calcBreedingEfficiency(optimalVitamins); // Check all max-vitamin combinations for (let carbos = carbosUnlocked * totalVitamins; carbos >= 0; carbos--) { for (let calcium = calciumUnlocked * (totalVitamins - carbos); calcium >= 0; calcium--) { let protein = totalVitamins - (carbos + calcium); let newEff = this.calcBreedingEfficiency([protein, calcium, carbos]); if (newEff >= eff) { const newVitamins = [protein, calcium, carbos]; if (newEff == eff) { // Choose cheaper version const oldPrice = optimalVitamins.reduce((sum, v, i) => (sum + v * prices[i]), 0); const newPrice = newVitamins.reduce((sum, v, i) => (sum + v * prices[i]), 0); if (oldPrice <= newPrice) { continue; } } eff = newEff; optimalVitamins = newVitamins; } } } // Optimally use vitamins GameHelper.enumNumbers(GameConstants.VitaminType).forEach((v) => { if (this.vitaminsUsed[v]()) { this.removeVitamin(v, Infinity); } }); GameHelper.enumNumbers(GameConstants.VitaminType).forEach((v) => { if (v < optimalVitamins.length && optimalVitamins[v] > 0) { this.useVitamin(v, optimalVitamins[v]); } }); } } } /** * Creates container for scripts settings in the settings menu, adding scripts tab if it doesn't exist yet */ function createScriptSettingsContainer(name) { const settingsID = name.replaceAll(/s/g, '').toLowerCase(); var settingsContainer = document.getElementById('settings-scripts-container'); // Create scripts settings tab if it doesn't exist yet if (!settingsContainer) { // Fixes the Scripts nav item getting wrapped to the bottom by increasing the max width of the window document.querySelector('#settingsModal div').style.maxWidth = '850px'; // Create and attach script settings tab link const settingTabs = document.querySelector('#settingsModal ul.nav-tabs'); const li = document.createElement('li'); li.classList.add('nav-item'); li.innerHTML = `Scripts`; settingTabs.appendChild(li); // Create and attach script settings tab contents const tabContent = document.querySelector('#settingsModal .tab-content'); scriptSettings = document.createElement('div'); scriptSettings.classList.add('tab-pane'); scriptSettings.setAttribute('id', 'settings-scripts'); tabContent.appendChild(scriptSettings); settingsContainer = document.createElement('div'); settingsContainer.setAttribute('id', 'settings-scripts-container'); scriptSettings.appendChild(settingsContainer); } // Create settings container const settingsTable = document.createElement('table'); settingsTable.classList.add('table', 'table-striped', 'table-hover', 'm-0'); const header = document.createElement('thead'); header.innerHTML = `${name}`; settingsTable.appendChild(header); const settingsBody = document.createElement('tbody'); settingsBody.setAttribute('id', `settings-scripts-${settingsID}`); settingsTable.appendChild(settingsBody); // Insert settings container in alphabetical order let settingsList = Array.from(settingsContainer.children); let insertBefore = settingsList.find(elem => elem.querySelector('tbody').id > `settings-scripts-${settingsID}`); if (insertBefore) { insertBefore.before(settingsTable); } else { settingsContainer.appendChild(settingsTable); } return settingsBody; } function addGlobalStyle(css) { var head, style; head = document.getElementsByTagName('head')[0]; if (!head) { return; } style = document.createElement('style'); style.type = 'text/css'; style.innerHTML = css; head.appendChild(style); } function loadEpheniaScript(scriptName, initFunction, priorityFunction) { function reportScriptError(scriptName, error) { console.error(`Error while initializing '${scriptName}' userscript:\n${error}`); Notifier.notify({ type: NotificationConstants.NotificationOption.warning, title: scriptName, message: `The '${scriptName}' userscript crashed while loading. Check for updates or disable the script, then restart the game.\n\nReport script issues to the script developer, not to the Pokéclicker team.`, timeout: GameConstants.DAY, }); } const windowObject = !App.isUsingClient ? unsafeWindow : window; // Inject handlers if they don't exist yet if (windowObject.epheniaScriptInitializers === undefined) { windowObject.epheniaScriptInitializers = {}; const oldInit = Preload.hideSplashScreen; var hasInitialized = false; // Initializes scripts once enough of the game has loaded Preload.hideSplashScreen = function (...args) { var result = oldInit.apply(this, args); if (App.game && !hasInitialized) { // Initialize all attached userscripts Object.entries(windowObject.epheniaScriptInitializers).forEach(([scriptName, initFunction]) => { try { initFunction(); } catch (e) { reportScriptError(scriptName, e); } }); hasInitialized = true; } return result; } } // Prevent issues with duplicate script names if (windowObject.epheniaScriptInitializers[scriptName] !== undefined) { console.warn(`Duplicate '${scriptName}' userscripts found!`); Notifier.notify({ type: NotificationConstants.NotificationOption.warning, title: scriptName, message: `Duplicate '${scriptName}' userscripts detected. This could cause unpredictable behavior and is not recommended.`, timeout: GameConstants.DAY, }); let number = 2; while (windowObject.epheniaScriptInitializers[`${scriptName} ${number}`] !== undefined) { number++; } scriptName = `${scriptName} ${number}`; } // Add initializer for this particular script windowObject.epheniaScriptInitializers[scriptName] = initFunction; // Run any functions that need to execute before the game starts if (priorityFunction) { $(document).ready(() => { try { priorityFunction(); } catch (e) { reportScriptError(scriptName, e); // Remove main initialization function windowObject.epheniaScriptInitializers[scriptName] = () => null; } }); } } if (!App.isUsingClient || localStorage.getItem('additionalvisualsettings') === 'true') { if (!App.isUsingClient) { // Necessary for userscript managers unsafeWindow.AdditionalVisualSettings = AdditionalVisualSettings; } loadEpheniaScript('additionalvisualsettings', () => AdditionalVisualSettings.initVisualSettings(), () => AdditionalVisualSettings.initOnLoad()); }