[Pokeclicker] Additional Visual Settings

Adds additional settings for hiding some visual things to help out with performance. Also, includes various features that help with ease of accessibility.

  1. // ==UserScript==
  2. // @name [Pokeclicker] Additional Visual Settings
  3. // @namespace Pokeclicker Scripts
  4. // @author Ephenia (Credit: Optimatum)
  5. // @description Adds additional settings for hiding some visual things to help out with performance. Also, includes various features that help with ease of accessibility.
  6. // @copyright https://github.com/Ephenia
  7. // @license GPL-3.0 License
  8. // @version 3.0
  9.  
  10. // @homepageURL https://github.com/Ephenia/Pokeclicker-Scripts/
  11. // @supportURL https://github.com/Ephenia/Pokeclicker-Scripts/issues
  12.  
  13. // @match https://www.pokeclicker.com/
  14. // @icon https://www.google.com/s2/favicons?domain=pokeclicker.com
  15. // @grant unsafeWindow
  16. // @run-at document-idle
  17. // ==/UserScript==
  18.  
  19. // TODO disable party attack number + tooltip
  20.  
  21. class AdditionalVisualSettings {
  22. static graphicsDisabledSettings = {
  23. route: {
  24. header: ko.observable(false),
  25. pokemon: ko.observable(false),
  26. catchIcon: ko.observable(false),
  27. healthbar: ko.observable(false),
  28. attack: ko.observable(false),
  29. },
  30. gym: {
  31. header: ko.observable(false),
  32. timer: ko.observable(false),
  33. pokemon: ko.observable(false),
  34. healthbar: ko.observable(false),
  35. attack: ko.observable(false),
  36. },
  37. dungeon: {
  38. header: ko.observable(false),
  39. timer: ko.observable(false),
  40. images: ko.observable(false),
  41. attack: ko.observable(false),
  42. },
  43. battleFrontier: {
  44. header: ko.observable(false),
  45. timer: ko.observable(false),
  46. pokemon: ko.observable(false),
  47. healthbar: ko.observable(false),
  48. },
  49. };
  50. static autoClickerIntegration = ko.observable(JSON.parse(localStorage.getItem('AVSautoClickerIntegration') || 'false'));
  51. // Disable graphics unless autoclicker integration is on and autoclicker is not running
  52. static graphicsSettingsActive = ko.computed({
  53. read: () => !(typeof EnhancedAutoClicker === 'function' && this.autoClickerIntegration() && !EnhancedAutoClicker.autoClickState()),
  54. deferEvaluation: true
  55. });
  56.  
  57. static loadGraphicsSettings() {
  58. try {
  59. const savedSettings = JSON.parse(localStorage.getItem('AVSgraphicsDisabledSettings') || '{}');
  60. Object.keys(this.graphicsDisabledSettings).forEach(state => {
  61. Object.keys(this.graphicsDisabledSettings[state]).forEach(setting => {
  62. if (savedSettings[state]?.[setting] != undefined) {
  63. const val = !!savedSettings[state][setting];
  64. this.graphicsDisabledSettings[state][setting](val);
  65. }
  66. });
  67. });
  68. } catch {
  69. this.saveGraphicsSettings();
  70. }
  71. }
  72.  
  73. static saveGraphicsSettings() {
  74. const settingsToSave = {};
  75. Object.keys(this.graphicsDisabledSettings).forEach(state => {
  76. settingsToSave[state] = {};
  77. Object.keys(this.graphicsDisabledSettings[state]).forEach(setting => {
  78. settingsToSave[state][setting] = this.graphicsDisabledSettings[state][setting]();
  79. });
  80. });
  81. localStorage.setItem('AVSgraphicsDisabledSettings', JSON.stringify(settingsToSave));
  82. }
  83.  
  84. static initOnLoad() {
  85. this.addGraphicsBindings();
  86. this.addOptimizeVitamins();
  87. }
  88.  
  89. static initVisualSettings() {
  90. this.loadGraphicsSettings();
  91.  
  92. // Add shortcut menu icons
  93. const getMenu = document.getElementById('startMenu');
  94. const shortcutsToAdd = [
  95. ['quick-settings', '#settingsModal', ''],
  96. ['quick-inventory', '#showItemsModal', ''],
  97. ['quick-pokedex', '#pokedexModal', ''],
  98. ];
  99. shortcutsToAdd.forEach(([id, modal, source]) => {
  100. const quickElem = document.createElement('img');
  101. quickElem.id = id;
  102. quickElem.src = source;
  103. quickElem.setAttribute('href', modal);
  104. quickElem.setAttribute('data-toggle', 'modal');
  105. getMenu.prepend(quickElem);
  106. });
  107.  
  108. // Add AVS settings options to scripts tab
  109. const settingsBody = createScriptSettingsContainer('Additional Visual Settings');
  110.  
  111. let elem = document.createElement('tr');
  112. elem.innerHTML = `<td class="p-2" colspan="2"><label class="m-0">Disable battle visuals</label></td>`;
  113. settingsBody.appendChild(elem);
  114.  
  115. // Graphics-disabling settings
  116. Object.keys(this.graphicsDisabledSettings).forEach(state => {
  117. elem = document.createElement('tr');
  118. elem.innerHTML = `<th class="p-2 col-md-5" scope="row">${GameConstants.camelCaseToString(state)}</th><td class="p-2" style="display:flex;"></td>`;
  119. let innerElem = elem.querySelector('td');
  120. Object.keys(this.graphicsDisabledSettings[state]).forEach(setting => {
  121. const container = document.createElement('div');
  122. container.className = 'px-3'
  123. container.innerHTML = `${GameConstants.camelCaseToString(setting)} <input id="checkbox-AVS-${state}-${setting}" type="checkbox" class="mx-1"></td>`;
  124. const checkbox = container.querySelector('input');
  125. checkbox.checked = this.graphicsDisabledSettings[state][setting]();
  126. checkbox.addEventListener('change', event => {
  127. this.graphicsDisabledSettings[state][setting](event.target.checked);
  128. this.saveGraphicsSettings();
  129. });
  130. innerElem.appendChild(container);
  131. });
  132. settingsBody.appendChild(elem);
  133. });
  134.  
  135. // EnhancedAutoClicker integration setting, if the script is present
  136. if (typeof EnhancedAutoClicker === 'function') {
  137. elem = document.createElement('tr');
  138. elem.innerHTML = `<td class="p-2" colspan="2"><label class="m-0" for="checkbox-AVS-autoClickerIntegration">Disable graphics only when EnhancedAutoClicker is on</label>` +
  139. `<input id="checkbox-AVS-autoClickerIntegration" type="checkbox" class="mx-2"></td>`;
  140. settingsBody.appendChild(elem);
  141. const checkbox = elem.querySelector('input');
  142. checkbox.checked = this.autoClickerIntegration();
  143. checkbox.addEventListener('change', event => {
  144. this.autoClickerIntegration(event.target.checked);
  145. localStorage.setItem('AVSautoClickerIntegration', this.autoClickerIntegration());
  146. });
  147. }
  148.  
  149. // Create travel shortcut buttons on town map
  150. const travelShortcutsToAdd = [
  151. ['dock-button', 'Dock', {left: 32, top: 0}, MapHelper.openShipModal],
  152. ['gyms-button', 'Gyms', {left: 75, top: -8}, () => { AdditionalVisualSettings.generateRegionGymsList(); $('#gymsShortcutModal').modal('show'); }],
  153. ['dungeons-button', 'Dungeons', {left: 121, top: -8}, () => { AdditionalVisualSettings.generateRegionDungeonssList(); $('#dungeonsShortcutModal').modal('show'); }],
  154. ];
  155.  
  156. travelShortcutsToAdd.forEach(([id, name, pos, func]) => {
  157. const button = document.createElement('button');
  158. button.id = id;
  159. button.textContent = name;
  160. button.className = 'btn btn-block btn-success';
  161. button.style = `position: absolute; left: ${pos.left}px; top: ${pos.top}px; width: auto; height: 41px; font-size: 11px;`;
  162. button.addEventListener('click', func);
  163. document.getElementById('townMap').appendChild(button);
  164. });
  165.  
  166. // Prevent ship modal sequence-breaking
  167. document.getElementById('dock-button').setAttribute('data-bind', 'enabled: TownList[GameConstants.DockTowns[player.region]].isUnlocked()');
  168. ko.applyBindings(App.game, document.getElementById('dock-button'));
  169.  
  170. // Create gym and dungeon shortcut modals
  171. const modalNames = ['gyms', 'dungeons'];
  172. const fragment = new DocumentFragment();
  173. for (const name of modalNames) {
  174. const customModal = document.createElement('div');
  175. customModal.setAttribute('class', 'modal noselect fade');
  176. customModal.setAttribute('tabindex', '-1');
  177. customModal.setAttribute('role', 'dialogue');
  178. customModal.setAttribute('id', `${name}ShortcutModal`);
  179. customModal.setAttribute('aria-labelledby', `${name}ShortcutModalLabel`);
  180. customModal.innerHTML = `<div class="modal-dialog modal-dialog-scrollable modal-dialog-centered modal-sm" role="document">
  181. <div class="modal-content">
  182. <div class="modal-header" style="justify-content: space-around;">
  183. <h5 id="${name}-shortcut-modal-title" class="modal-title"></h5>
  184. <button type="button" class="close" data-dismiss="modal" aria-label="Close">
  185. <span aria-hidden="true">×</span>
  186. </button>
  187. </div>
  188. <div class="modal-body bg-ocean">
  189. <div id="${name}-shortcut-buttons"></div>
  190. </div>
  191. </div>
  192. </div>`;
  193. fragment.appendChild(customModal);
  194. }
  195. document.getElementById('ShipModal').after(fragment);
  196.  
  197. addGlobalStyle('.pageItemTitle { height:38px }');
  198. addGlobalStyle('#quick-settings, #quick-inventory, #quick-pokedex { height: 36px; background-color: #eee; border: 4px solid #eee; cursor: pointer; image-rendering: pixelated; }');
  199. addGlobalStyle('#quick-pokedex { padding: 2px; }')
  200. addGlobalStyle(':is(#quick-settings, #quick-inventory, #quick-pokedex):hover { background-color:#ddd; border: 4px solid #ddd; }');
  201. addGlobalStyle('#shortcutsContainer { display: block !important; }');
  202. addGlobalStyle('.gyms-shortcut-leaders { display: flex; pointer-events: none; position: absolute; height: 36px; top: 0; left: 0; image-rendering: pixelated; }');
  203. addGlobalStyle('.gyms-shortcut-badges { position: absolute; height: 36px; display: flex; top: 0; right: 0; }');
  204. addGlobalStyle('.dungeons-shortcut-costs { position: relative; margin-right: 12px; filter: none !important }');
  205. addGlobalStyle('#dungeons-shortcut-buttons > button:hover { -webkit-animation: bounceBackground 60s linear infinite alternate; animation: bounceBackground 60s linear infinite alternate; }');
  206. addGlobalStyle('#dungeons-shortcut-buttons > button * { z-index: 2 }');
  207. 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 }');
  208. addGlobalStyle('.dungeons-shortcut-info { position: relative; font-weight: bold }');
  209. }
  210.  
  211. static generateRegionGymsList() {
  212. const gymsBtns = document.getElementById('gyms-shortcut-buttons');
  213. const gymsHead = document.getElementById('gyms-shortcut-modal-title');
  214. gymsHead.textContent = `Gym Select (${GameConstants.camelCaseToString(GameConstants.Region[player.region])})`;
  215. gymsBtns.innerHTML = '';
  216. const fragment = new DocumentFragment();
  217. const regionGyms = Object.values(GymList).filter((gym) => gym.parent?.region === player.region);
  218. for (const gym of regionGyms) {
  219. const hasBadgeImage = !(BadgeEnums[gym.badgeReward].startsWith('Elite') || BadgeEnums[gym.badgeReward] == 'None');
  220. const badgeImage = (hasBadgeImage ? `assets/images/badges/${BadgeEnums[gym.badgeReward]}.svg` : '');
  221. const btn = document.createElement('button');
  222. btn.setAttribute('style', 'position: relative;');
  223. btn.setAttribute('class', 'btn btn-block btn-success');
  224. btn.addEventListener('click', () => {
  225. if (!MapHelper.isTownCurrentLocation(gym.parent.name)) {
  226. MapHelper.moveToTown(gym.parent.name);
  227. }
  228. $("#gymsShortcutModal").modal("hide");
  229. GymRunner.startGym(gym);
  230. });
  231. btn.disabled = !(gym.isUnlocked() && gym.parent.isUnlocked());
  232. btn.innerHTML = `<div class="gyms-shortcut-leaders">
  233. <img src="${gym.imagePath}" onerror="{ this.src='assets/images/npcs/specialNPCs/Mysterious Trainer.png'; }">
  234. </div>
  235. <div class="gyms-shortcut-badges">
  236. <img src="${badgeImage}" onerror="{ this.onerror=null; this.style.display='none'; }">
  237. </div>
  238. ${gym.leaderName}`;
  239. fragment.appendChild(btn);
  240. }
  241. gymsBtns.appendChild(fragment);
  242. }
  243.  
  244. static generateRegionDungeonssList() {
  245. const dungeonsBtns = document.getElementById('dungeons-shortcut-buttons');
  246. const dungeonsHead = document.getElementById('dungeons-shortcut-modal-title');
  247. dungeonsHead.textContent = `Dungeon Select (${GameConstants.camelCaseToString(GameConstants.Region[player.region])})`;
  248. dungeonsBtns.innerHTML = '';
  249. const fragment = new DocumentFragment();
  250. const dungeonTowns = Object.values(TownList).filter((town) => (town.region === player.region && town.constructor.name === 'DungeonTown' && town.dungeon != null));
  251. for (const town of dungeonTowns) {
  252. const dungeon = town.dungeon;
  253. const dungeonClears = App.game.statistics.dungeonsCleared[GameConstants.getDungeonIndex(dungeon.name)]();
  254. const canAffordEntry = App.game.wallet.currencies[GameConstants.Currency.dungeonToken]() >= dungeon.tokenCost;
  255. const canAccess = town.isUnlocked() && dungeon.isUnlocked() && canAffordEntry;
  256. const btn = document.createElement('button');
  257. 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});`);
  258. btn.setAttribute('class', 'btn btn-block btn-success');
  259. btn.addEventListener('click', () => {
  260. if (!MapHelper.isTownCurrentLocation(town.name)) {
  261. MapHelper.moveToTown(town.name);
  262. }
  263. $('#dungeonsShortcutModal').modal('hide');
  264. DungeonRunner.initializeDungeon(dungeon);
  265. });
  266. btn.disabled = !canAccess;
  267. btn.innerHTML = `<div class="dungeons-overlay"></div>
  268. <div class="dungeons-shortcut-costs">
  269. <img src="assets/images/currency/dungeonToken.svg" style="height: 24px; width: 24px;">
  270. <span style="font-weight: bold;color: ${canAffordEntry ? 'greenyellow' : '#f04124'}">${dungeon.tokenCost.toLocaleString('en-US')}</span>
  271. </div>
  272. <div class="dungeons-shortcut-info">
  273. <span>${dungeon.name}</span>
  274. <div>${dungeonClears.toLocaleString('en-US')} clears</div>
  275. </div>`;
  276. fragment.appendChild(btn);
  277. }
  278. dungeonsBtns.appendChild(fragment);
  279. }
  280.  
  281. // Must execute before game loads and applies knockout bindings
  282. static addGraphicsBindings() {
  283. function selectorWorkaround(element, selector) {
  284. try {
  285. return element.querySelector(selector);
  286. } catch {
  287. const [, outer, inner] = selector.match(/(.+):has\((.+)\)/);
  288. const innerElem = element.querySelector(`${outer} ${inner}`);
  289. return Array.from(element.querySelectorAll(outer)).find(e => e.contains(innerElem));
  290. }
  291. }
  292.  
  293. const selectors = {
  294. route: {
  295. container: '#routeBattleContainer',
  296. header: '.pageItemTitle > div:has(> knockout)',
  297. pokemon: 'div:has(> knockout[data-bind*="pokemonSpriteTemplate"])',
  298. catchIcon: 'div.catchChance',
  299. healthbar: 'div.progress.hitpoints',
  300. attack: '.pageItemFooter knockout[data-bind*="pokemonAttackTemplate"]',
  301. },
  302. gym: {
  303. container: '#battleContainer div[data-bind="if: App.game.gameState === GameConstants.GameState.gym"]',
  304. header: [
  305. ['h2.pageItemTitle > knockout:has(knockout[data-bind*="pokemonNameTemplate"])', 'before'],
  306. ['h2.pageItemTitle > knockout:has(span[data-bind*="pokemonsDefeatedComputable"])', 'after']
  307. ],
  308. timer: 'h2.pageItemTitle .timer',
  309. pokemon: 'div:has(> knockout[data-bind*="pokemonSpriteTemplate"])',
  310. healthbar: 'div.progress.hitpoints',
  311. attack: '.pageItemFooter knockout[data-bind*="pokemonAttackTemplate"]',
  312. },
  313. dungeon: {
  314. container: '#battleContainer div[data-bind="if: App.game.gameState === GameConstants.GameState.dungeon"]',
  315. header: [
  316. ['h2.pageItemTitle > knockout:has(knockout[data-bind*="pokemonNameTemplate"])', 'before'],
  317. ['h2.pageItemTitle > knockout:has(span[data-bind*="defeatedTrainerPokemon"])', 'after']
  318. ],
  319. timer: 'h2.pageItemTitle .timer',
  320. images: [
  321. ['h2.pageItemTitle', 'after'],
  322. ['h2.pageItemFooter', 'before']
  323. ],
  324. attack: '.pageItemFooter knockout[data-bind*="pokemonAttackTemplate"]',
  325. },
  326. battleFrontier: {
  327. container: '#battleContainer div[data-bind="if: App.game.gameState == GameConstants.GameState.battleFrontier"]',
  328. header: [
  329. ['h2.pageItemTitle > knockout:has(knockout[data-bind*="pokemonNameTemplate"])', 'before'],
  330. ['h2.pageItemTitle > knockout[data-bind*="pokemonLeftImages"]', 'after']
  331. ],
  332. timer: 'h2.pageItemTitle .timer',
  333. pokemon: 'div:has(> knockout[data-bind*="pokemonSpriteTemplate"])',
  334. healthbar: 'div.progress.hitpoints',
  335. },
  336. };
  337.  
  338. Object.keys(this.graphicsDisabledSettings).forEach(state => {
  339. const container = document.querySelector(selectors[state]?.container)
  340. if (!container) {
  341. console.error(`AVS: could not find ${state} container`);
  342. return;
  343. }
  344. Object.keys(this.graphicsDisabledSettings[state]).forEach(setting => {
  345. if (!selectors[state]?.[setting]) {
  346. return;
  347. }
  348. const selector = selectors[state][setting];
  349. const binding = `ko ifnot: AdditionalVisualSettings.graphicsDisabledSettings.${state}.${setting}() && AdditionalVisualSettings.graphicsSettingsActive()`;
  350. // Add binding for this setting
  351. if (Array.isArray(selector)) {
  352. // For binding multiple elements at once, which requires more complicated selecting
  353. selector.forEach(([query, order], i) => {
  354. const elem = selectorWorkaround(container, query);
  355. const commentBinding = i % 2 == 0 ? binding : '/ko';
  356. if (order == 'before') {
  357. elem.before(new Comment(commentBinding));
  358. } else {
  359. elem.after(new Comment(commentBinding));
  360. }
  361. });
  362. } else {
  363. const elem = selectorWorkaround(container, selector);
  364. // Special case: insert a backup attack-disabled element so formatting doesn't look weird
  365. // Do this before applying the main binding to put it outside the binding comments
  366. if (setting == 'attack') {
  367. const replacementAttack = document.createElement('span');
  368. elem.after(replacementAttack);
  369. replacementAttack.outerHTML = `<span style="display: inline;" data-bind="${binding.replace('ko ifnot:', 'if:')}">Pokémon Attack: <span>-----</span></span>`;
  370. }
  371. // Insert the binding
  372. elem.before(new Comment(binding));
  373. elem.after(new Comment('/ko'));
  374. }
  375. });
  376. });
  377. }
  378.  
  379.  
  380. static addOptimizeVitamins() {
  381. // Add button to vitamin menu
  382. // (must execute before game loads and applies knockout bindings)
  383. const btn = document.createElement('button');
  384. btn.setAttribute('class', 'btn btn-link btn-sm text-decoration-none align-text-top');
  385. btn.setAttribute('style', 'line-height: 0.6; font-size: 1rem; float: right;');
  386. btn.setAttribute('data-bind', `click: () => { if ($data) { $data.optimizeVitamins() } }, class: (!$data.breeding ? 'text-success' : 'text-muted')`);
  387. btn.innerHTML = '⚖';
  388. document.querySelector('#pokemonVitaminExpandedModal tbody[data-bind*="PartyController.getVitaminSortedList"] td').appendChild(btn);
  389.  
  390. // Add optimize-vitamin functions for party pokemon (adapted from wiki)
  391. PartyPokemon.prototype.calcBreedingEfficiency = function(vitaminsUsed) {
  392. // attack bonus
  393. const attackBonusPercent = (GameConstants.BREEDING_ATTACK_BONUS + vitaminsUsed[GameConstants.VitaminType.Calcium]) / 100;
  394. const proteinBoost = vitaminsUsed[GameConstants.VitaminType.Protein];
  395. const breedingAttackBonus = (this.baseAttack * attackBonusPercent) + proteinBoost;
  396. // egg steps
  397. const div = 300;
  398. const extraCycles = (vitaminsUsed[GameConstants.VitaminType.Calcium] + vitaminsUsed[GameConstants.VitaminType.Protein]) / 2;
  399. const steps = (this.eggCycles + extraCycles) * GameConstants.EGG_CYCLE_MULTIPLIER;
  400. const adjustedSteps = (steps <= div ? steps : Math.round(((steps / div) ** (1 - vitaminsUsed[GameConstants.VitaminType.Carbos] / 70)) * div));
  401. // efficiency
  402. return (breedingAttackBonus / adjustedSteps) * GameConstants.EGG_CYCLE_MULTIPLIER;
  403. }
  404.  
  405. PartyPokemon.prototype.optimizeVitamins = function() {
  406. const totalVitamins = (player.highestRegion() + 1) * 5;
  407. const carbosUnlocked = player.highestRegion() >= GameConstants.Region.unova;
  408. const calciumUnlocked = player.highestRegion() >= GameConstants.Region.hoenn;
  409. const prices = GameHelper.enumStrings(GameConstants.VitaminType).map(v => ItemList[v].basePrice);
  410. // Add our initial starting efficiency here
  411. let optimalVitamins = [0, 0, 0];
  412. let eff = this.calcBreedingEfficiency(optimalVitamins);
  413. // Check all max-vitamin combinations
  414. for (let carbos = carbosUnlocked * totalVitamins; carbos >= 0; carbos--) {
  415. for (let calcium = calciumUnlocked * (totalVitamins - carbos); calcium >= 0; calcium--) {
  416. let protein = totalVitamins - (carbos + calcium);
  417. let newEff = this.calcBreedingEfficiency([protein, calcium, carbos]);
  418. if (newEff >= eff) {
  419. const newVitamins = [protein, calcium, carbos];
  420. if (newEff == eff) {
  421. // Choose cheaper version
  422. const oldPrice = optimalVitamins.reduce((sum, v, i) => (sum + v * prices[i]), 0);
  423. const newPrice = newVitamins.reduce((sum, v, i) => (sum + v * prices[i]), 0);
  424. if (oldPrice <= newPrice) {
  425. continue;
  426. }
  427. }
  428. eff = newEff;
  429. optimalVitamins = newVitamins;
  430. }
  431. }
  432. }
  433. // Optimally use vitamins
  434. GameHelper.enumNumbers(GameConstants.VitaminType).forEach((v) => {
  435. if (this.vitaminsUsed[v]()) {
  436. this.removeVitamin(v, Infinity);
  437. }
  438. });
  439. GameHelper.enumNumbers(GameConstants.VitaminType).forEach((v) => {
  440. if (v < optimalVitamins.length && optimalVitamins[v] > 0) {
  441. this.useVitamin(v, optimalVitamins[v]);
  442. }
  443. });
  444. }
  445. }
  446. }
  447.  
  448.  
  449. /**
  450. * Creates container for scripts settings in the settings menu, adding scripts tab if it doesn't exist yet
  451. */
  452. function createScriptSettingsContainer(name) {
  453. const settingsID = name.replaceAll(/s/g, '').toLowerCase();
  454. var settingsContainer = document.getElementById('settings-scripts-container');
  455.  
  456. // Create scripts settings tab if it doesn't exist yet
  457. if (!settingsContainer) {
  458. // Fixes the Scripts nav item getting wrapped to the bottom by increasing the max width of the window
  459. document.querySelector('#settingsModal div').style.maxWidth = '850px';
  460. // Create and attach script settings tab link
  461. const settingTabs = document.querySelector('#settingsModal ul.nav-tabs');
  462. const li = document.createElement('li');
  463. li.classList.add('nav-item');
  464. li.innerHTML = `<a class="nav-link" href="#settings-scripts" data-toggle="tab">Scripts</a>`;
  465. settingTabs.appendChild(li);
  466. // Create and attach script settings tab contents
  467. const tabContent = document.querySelector('#settingsModal .tab-content');
  468. scriptSettings = document.createElement('div');
  469. scriptSettings.classList.add('tab-pane');
  470. scriptSettings.setAttribute('id', 'settings-scripts');
  471. tabContent.appendChild(scriptSettings);
  472. settingsContainer = document.createElement('div');
  473. settingsContainer.setAttribute('id', 'settings-scripts-container');
  474. scriptSettings.appendChild(settingsContainer);
  475. }
  476.  
  477. // Create settings container
  478. const settingsTable = document.createElement('table');
  479. settingsTable.classList.add('table', 'table-striped', 'table-hover', 'm-0');
  480. const header = document.createElement('thead');
  481. header.innerHTML = `<tr><th colspan="2">${name}</th></tr>`;
  482. settingsTable.appendChild(header);
  483. const settingsBody = document.createElement('tbody');
  484. settingsBody.setAttribute('id', `settings-scripts-${settingsID}`);
  485. settingsTable.appendChild(settingsBody);
  486.  
  487. // Insert settings container in alphabetical order
  488. let settingsList = Array.from(settingsContainer.children);
  489. let insertBefore = settingsList.find(elem => elem.querySelector('tbody').id > `settings-scripts-${settingsID}`);
  490. if (insertBefore) {
  491. insertBefore.before(settingsTable);
  492. } else {
  493. settingsContainer.appendChild(settingsTable);
  494. }
  495.  
  496. return settingsBody;
  497. }
  498.  
  499. function addGlobalStyle(css) {
  500. var head, style;
  501. head = document.getElementsByTagName('head')[0];
  502. if (!head) { return; }
  503. style = document.createElement('style');
  504. style.type = 'text/css';
  505. style.innerHTML = css;
  506. head.appendChild(style);
  507. }
  508.  
  509. function loadEpheniaScript(scriptName, initFunction, priorityFunction) {
  510. function reportScriptError(scriptName, error) {
  511. console.error(`Error while initializing '${scriptName}' userscript:\n${error}`);
  512. Notifier.notify({
  513. type: NotificationConstants.NotificationOption.warning,
  514. title: scriptName,
  515. 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.`,
  516. timeout: GameConstants.DAY,
  517. });
  518. }
  519. const windowObject = !App.isUsingClient ? unsafeWindow : window;
  520. // Inject handlers if they don't exist yet
  521. if (windowObject.epheniaScriptInitializers === undefined) {
  522. windowObject.epheniaScriptInitializers = {};
  523. const oldInit = Preload.hideSplashScreen;
  524. var hasInitialized = false;
  525.  
  526. // Initializes scripts once enough of the game has loaded
  527. Preload.hideSplashScreen = function (...args) {
  528. var result = oldInit.apply(this, args);
  529. if (App.game && !hasInitialized) {
  530. // Initialize all attached userscripts
  531. Object.entries(windowObject.epheniaScriptInitializers).forEach(([scriptName, initFunction]) => {
  532. try {
  533. initFunction();
  534. } catch (e) {
  535. reportScriptError(scriptName, e);
  536. }
  537. });
  538. hasInitialized = true;
  539. }
  540. return result;
  541. }
  542. }
  543.  
  544. // Prevent issues with duplicate script names
  545. if (windowObject.epheniaScriptInitializers[scriptName] !== undefined) {
  546. console.warn(`Duplicate '${scriptName}' userscripts found!`);
  547. Notifier.notify({
  548. type: NotificationConstants.NotificationOption.warning,
  549. title: scriptName,
  550. message: `Duplicate '${scriptName}' userscripts detected. This could cause unpredictable behavior and is not recommended.`,
  551. timeout: GameConstants.DAY,
  552. });
  553. let number = 2;
  554. while (windowObject.epheniaScriptInitializers[`${scriptName} ${number}`] !== undefined) {
  555. number++;
  556. }
  557. scriptName = `${scriptName} ${number}`;
  558. }
  559. // Add initializer for this particular script
  560. windowObject.epheniaScriptInitializers[scriptName] = initFunction;
  561. // Run any functions that need to execute before the game starts
  562. if (priorityFunction) {
  563. $(document).ready(() => {
  564. try {
  565. priorityFunction();
  566. } catch (e) {
  567. reportScriptError(scriptName, e);
  568. // Remove main initialization function
  569. windowObject.epheniaScriptInitializers[scriptName] = () => null;
  570. }
  571. });
  572. }
  573. }
  574.  
  575. if (!App.isUsingClient || localStorage.getItem('additionalvisualsettings') === 'true') {
  576. if (!App.isUsingClient) {
  577. // Necessary for userscript managers
  578. unsafeWindow.AdditionalVisualSettings = AdditionalVisualSettings;
  579. }
  580. loadEpheniaScript('additionalvisualsettings', () => AdditionalVisualSettings.initVisualSettings(), () => AdditionalVisualSettings.initOnLoad());
  581. }