- // ==UserScript==
- // @name Geoguessr Team Duels Advanced Options
- // @description Adds extra options to team duel settings.
- // @version 0.2.3
- // @author macca#8949
- // @license MIT
- // @match https://www.geoguessr.com/*
- // @run-at document-start
- // @grant none
- // @require https://greasyfork.org/scripts/460322-geoguessr-styles-scan/code/Geoguessr%20Styles%20Scan.js?version=1151654
- // @namespace https://greasyfork.org/en/scripts/452579-geoguessr-team-duels-advanced-options
- // ==/UserScript==
-
- let cachedGameMode = '';
- let cachedGameOptions = {};
-
- const getGameId = () => {
- const scripts = document.getElementsByTagName('script');
- for (const script of scripts) {
- if (script.src.includes('_buildManifest.js')) {
- return script.src.split('/')[5];
- }
- }
- }
-
- const gameMode = () => {
- if (document.getElementsByClassName(cn('bars_content__')).length < 3) return '';
- const fullText = document.getElementsByClassName(cn('bars_content__'))[2].textContent;
- return fullText.substring(0, fullText.lastIndexOf(' ')).toLowerCase();
- }
-
- async function fetchWithCors(url, method, body) {
- return await fetch(url, {
- "headers": {
- "accept": "*/*",
- "accept-language": "en-US,en;q=0.8",
- "content-type": "application/json",
- "sec-fetch-dest": "empty",
- "sec-fetch-mode": "cors",
- "sec-fetch-site": "same-site",
- "sec-gpc": "1",
- "x-client": "web"
- },
- "referrer": "https://www.geoguessr.com/",
- "referrerPolicy": "strict-origin-when-cross-origin",
- "body": JSON.stringify(body),
- "method": method,
- "mode": "cors",
- "credentials": "same-origin"
- })
- }
-
- window.modifySetting = (e, settingName) => {
- let newValue = e.value;
- if (settingName === 'multiplierIncrement') {
- newValue *= 10;
- newValue = Math.round(newValue);
- } else {
- newValue *= 1; // string to number conversion
- newValue = Math.round(newValue);
- }
-
- // Fetch the game options
- fetchWithCors(`https://www.geoguessr.com/_next/data/${getGameId()}/en/party.json`, "GET")
- .then((response) => response.json())
- .then((data) => {
- let gameOptions = data.pageProps.party.gameSettings;
- gameOptions[settingName] = newValue;
- // Push the updated options
- fetchWithCors(`https://www.geoguessr.com/api/v4/parties/v2/game-settings`, "PUT", gameOptions);
- });
-
- cachedGameMode = gameMode();
- cachedGameOptions[settingName] = e.value;
- }
-
- let optionTextInputInnerHTML = (id, settingName, text, icon, helpText) =>
- `<div class="${cn('numeric-option_wrapper__')} advanced-option-setting"><div class="${cn('numeric-option_icon__')}"><img alt="" loading="lazy" width="48" height="48" decoding="async" data-nimg="1" class="${cn('rule-icons_icon__')}" style="color: transparent;" src="${icon}"></div><div class="${cn('numeric-option_label__')}">${text}</div><div><input type="text" id="${id}" onblur="modifySetting(this, '${settingName}')" style="text-align: center; background: rgba(255,255,255,0.1); color: white; border: none; border-radius: 5px; width: 80px;"></div></div><p>${helpText}</p>`;
-
- function makeCustomTextInput(elt, id, settingName, text, icon, helpText) {
- elt.parentElement.outerHTML = optionTextInputInnerHTML(id, settingName, text, icon, helpText);
- return elt;
- }
-
- function updateValue(inputId, option, gameOptions) {
- if (document.querySelector(inputId)) {
- if (gameMode() == cachedGameMode && option in cachedGameOptions) {
- document.querySelector(inputId).value = cachedGameOptions[option];
- } else {
- if (option == 'multiplierIncrement') {
- document.querySelector(inputId).value = gameOptions[option] / 10;
- } else {
- document.querySelector(inputId).value = gameOptions[option];
- }
- }
- }
- }
-
- let observer = new MutationObserver(async (mutations) => {
- if (document.querySelector('.advanced-option-setting')) return;
-
- await scanStyles();
-
- if (window.location.href.includes('party') && (document.getElementsByClassName(cn('slider-option_slider__')) || document.getElementsByClassName(cn('numeric-option_button__')))) {
- if (gameMode().includes('duels')) {
- let healthEl = document.evaluate('//div[text()="Initial health"]', document, null, XPathResult.FIRST_ORDERED_NODE_TYPE, null).singleNodeValue;
- let multiplierEl = document.evaluate('//div[text()="Multiplier increase"]', document, null, XPathResult.FIRST_ORDERED_NODE_TYPE, null).singleNodeValue;
- let timeAfterGuessEl = document.evaluate('//div[text()="Timer after guess"]', document, null, XPathResult.FIRST_ORDERED_NODE_TYPE, null).singleNodeValue;
- let maxRoundTimeEl = document.evaluate('//div[text()="Max round time"]', document, null, XPathResult.FIRST_ORDERED_NODE_TYPE, null).singleNodeValue;
- if (healthEl) {
- makeCustomTextInput(healthEl, 'health-input', 'initialHealth', 'Initial health', '/_next/image?url=%2F_next%2Fstatic%2Fmedia%2Fheart.3a3fd066.png&w=96&q=75', '');
- }
- if (multiplierEl) {
- makeCustomTextInput(multiplierEl, 'increment-input', 'multiplierIncrement', 'Multiplier increase', '/_next/static/media/multipliers-icon.63803925.svg', '(must be between 0.1 and 10)');
- }
- if (timeAfterGuessEl) {
- makeCustomTextInput(timeAfterGuessEl, 'time-after-guess-input', 'timeAfterGuess', 'Timer after guess', '/_next/image?url=%2F_next%2Fstatic%2Fmedia%2Ftime-limit.8a68c82e.png&w=96&q=75', '(must be between 10 and 300 seconds)');
- }
- if (maxRoundTimeEl) {
- makeCustomTextInput(maxRoundTimeEl, 'max-round-time-input', 'maxRoundTime', 'Max round time', '/_next/image?url=%2F_next%2Fstatic%2Fmedia%2Ftime-limit.8a68c82e.png&w=96&q=75', '(0 for no time limit)');
- }
- } else if (gameMode() == 'city streaks') {
- let gameTimeEl = document.evaluate('//div[text()="Game time"]', document, null, XPathResult.FIRST_ORDERED_NODE_TYPE, null).singleNodeValue;
- if (gameTimeEl) {
- makeCustomTextInput(gameTimeEl, 'time-input', 'duration', 'Game time', '/_next/image?url=%2F_next%2Fstatic%2Fmedia%2Ftime-limit.8a68c82e.png&w=96&q=75', '(must be between 60 and 900 seconds)');
- }
- } else {
- let roundTimeEl = document.evaluate('//div[text()="Round time"]', document, null, XPathResult.FIRST_ORDERED_NODE_TYPE, null).singleNodeValue;
- if (roundTimeEl) {
- let settingName = 'roundTime';
- let helpText = '(must be between 10 and 600 seconds)';
- if (gameMode() == 'bullseye') {
- settingName = 'bullseyeRoundTime';
- helpText = '(must be less than 600 seconds, 0 for no time limit)';
- }
- makeCustomTextInput(roundTimeEl, 'time-input', settingName, 'Round time', '/_next/image?url=%2F_next%2Fstatic%2Fmedia%2Ftime-limit.8a68c82e.png&w=96&q=75', helpText);
- }
- }
-
- fetchWithCors(`https://www.geoguessr.com/_next/data/${getGameId()}/en/party.json`, "GET")
- .then((response) => response.json())
- .then((data) => {
- let gameOptions = data.pageProps.party.gameSettings;
-
- if (gameMode() == 'city streaks') {
- updateValue('#time-input', 'duration', gameOptions);
- } else if (gameMode().includes('duels')) {
- updateValue('#health-input', 'initialHealth', gameOptions);
- updateValue('#increment-input', 'multiplierIncrement', gameOptions);
- updateValue('#time-after-guess-input', 'timeAfterGuess', gameOptions);
- updateValue('#max-round-time-input', 'maxRoundTime', gameOptions);
- } else if (gameMode() == 'bullseye') {
- updateValue('#time-input', 'bullseyeRoundTime', gameOptions);
- } else {
- updateValue('#time-input', 'roundTime', gameOptions);
- }
- });
- }
- });
-
-
- observer.observe(document.body, {
- characterDataOldValue: false,
- subtree: true,
- childList: true,
- characterData: false
- });
-
- document.addEventListener('keydown', (event) => {
- if (event.key == 'Escape' && document.getElementsByClassName(cn('party-modal_heading__'))) {
- document.activeElement.blur();
- }
- });