Game Search Helper

Adds search buttons on various gaming related websites to search for the game on external sites

// ==UserScript==
// @name         Game Search Helper
// @namespace    https://store.steampowered.com/
// @version      1.3.8
// @license      GPLv3
// @description  Adds search buttons on various gaming related websites to search for the game on external sites
// @author       xdpirate
// @include      /^https\:\/\/www\.(nintendolife|pushsquare|purexbox|timeextension)\.com\/(games|news|reviews|features|guides)\/.*/
// @include      /^https\:\/\/store\.steampowered\.com\/(app|bundle)\/.*/
// @include      /^https\:\/\/www\.metacritic\.com/(browse|game)(\/games\/)?.*/
// @match        https://store.epicgames.com/*
// @match        https://en.wikipedia.org/wiki/*
// @match        https://opencritic.com/*
// @match        https://www.gog.com/*/game/*
// @match        https://www.startpage.com/sp/search*
// @match        https://github.com/xdpirate/GameSearchHelper/blob/main/CustomSearchEngines.md
// @require      https://cdn.jsdelivr.net/npm/sortablejs@1.15.0/Sortable.min.js
// @icon         
// @grant        GM_addStyle
// @grant        GM_getValue
// @grant        GM_setValue
// @run-at       document-end
// ==/UserScript==
if(window.top != window.self) { // Prevent the script from running in frames
    throw `[${GM_info.script.name}] Not running as topmost frame, exiting`;
}

let GSHSettings = GM_getValue("GSHSettings", {
    defaultProviders: {
        Steam: {
            Startpage: {enabled: true},
            Google: {enabled: false},
            Metacritic: {enabled: true},
            OpenCritic: {enabled: true},
            eBay: {enabled: true},
            eBayUK: {enabled: false}
        },
        HookshotMedia: {
            Startpage: {enabled: true},
            Google: {enabled: false},
            Metacritic: {enabled: true},
            OpenCritic: {enabled: true},
            eBay: {enabled: true},
            eBayUK: {enabled: false}
        },
        Metacritic: {
            Startpage: {enabled: true},
            Google: {enabled: false},
            Metacritic: {enabled: false},
            OpenCritic: {enabled: true},
            eBay: {enabled: true},
            eBayUK: {enabled: false}
        },
        EpicGamesStore: {
            Startpage: {enabled: true},
            Google: {enabled: false},
            Metacritic: {enabled: false},
            OpenCritic: {enabled: true},
            eBay: {enabled: true},
            eBayUK: {enabled: false}
        },
        Wikipedia: {
            Startpage: {enabled: true},
            Google: {enabled: false},
            Metacritic: {enabled: false},
            OpenCritic: {enabled: true},
            eBay: {enabled: true},
            eBayUK: {enabled: false}
        },
        OpenCritic: {
            Startpage: {enabled: true},
            Google: {enabled: false},
            Metacritic: {enabled: true},
            OpenCritic: {enabled: false},
            eBay: {enabled: true},
            eBayUK: {enabled: false}
        },
        GOGcom: {
            Startpage: {enabled: true},
            Google: {enabled: false},
            Metacritic: {enabled: true},
            OpenCritic: {enabled: true},
            eBay: {enabled: true},
            eBayUK: {enabled: false}
        }
    },
    customProviders: {},
    OpenCriticHelperEnabled: true,
    StripSpecialCharsEnabled: true
});

let defaultEnabledContextsForNewSearchEngines = {
    Steam: true, 
    HookshotMedia: true, 
    Metacritic: true, 
    EpicGamesStore: true,
    Wikipedia: true,
    OpenCritic: true,
    GOGcom: true
};

// Gracefully add new options if the user already has saved data that doesn't include them
if(GSHSettings.StripSpecialCharsEnabled == null) {
    GSHSettings.StripSpecialCharsEnabled = true;
}

GM_addStyle(`
    .GSHIcon {
        border: none;
        padding-left: 2px;
        padding-right: 2px;
    }

    #GSHToggleButton, .GSHBuiltInHandle, .GSHCustomHandle {
        cursor: pointer;
    }

    #GSHHeader {
        font-size: 18px !important;
    }

    .GSHSteamBundleContainer, .GSHMetacriticListContainer {
        margin-bottom: 20px;
    }

    .GSHSteamBundleContainer {
        margin-left: 190px;
    }

    #GSHLinkbox {
        float: right;
        font-size: 10px;
    }

    #GSHBuiltInSearchEngines, #GSHCustomSearchEngines, #GSHCustomSearchEngineDiv {
        padding: 10px;
        border: 1px solid white;
        line-height: 16px;
        border-radius: 5px;
    }
    
    #GSHCustomSearchEngineDiv {
        margin-top: 20px;
    }

    #GSHWrapper {
        overflow: hidden;
    }

    #GSHBuiltInSearchEngines, #GSHCustomSearchEngines {
        max-width: 500px;
        height: 120px; 
        overflow-y: scroll;
        overflow-x: scroll;
    }

    .GSHSearchContainer {
        float: left;
    }

    #GSHOuterDiv {
        float: left;
        background-color: black;
        color: white;
        padding: 10px;
        border: 1px solid white;
        border-radius: 10px;
        z-index: 2147483647;
        display: block;
        position: absolute;
        top: 50px;
        left: 10px;
        font-family: Helvetica !important;
        font-size: 14px !important;
    }

    #GSHSaveButton, #GSHOptionsDiv {
        margin-top: 10px;
        padding: 5px;
    }

    .GSHModifyCustomButton, #GSHCancelEditButton, #GSHNewCustomSearchButton, #GSHGetEnginesButton, .GSHLink {
        font-size: 12px !important;
        cursor: pointer;
        color: #AAA;
        text-decoration: none;
    }

    #GSHSaveDiv {
        width: 100%;
        margin: auto;
        text-align: center;
    }

    .GSHTextBox {
        width: 100%;
    }

    .GSHHidden {
        display: none;
    }

    .HMIcon {
        display: inline-block;
        border: none;
        margin-left: 4px;
        margin-right: 4px;
    }

    .HMContainer {
        padding: 5px;
        padding-top: 10px;
        background-color: white;
        border-radius: 5px;
        width: 100%;
        text-align: center;
    }

    .GSHLabel {
        font-weight: normal !important;
    }

    .GSHInputButton, .GSHInputField {
        background-color: black !important;
        color: white !important;
        border: 1px solid white !important;
        border-radius: 10px !important;
        position: relative !important;
        opacity: 1 !important;
        pointer-events: auto !important;
    }

    .GSHWPBox, .GSHOCContainer, .GSHGOGContainer {
        margin-bottom: 10px;
    }
`);

// Determine current context (site or groups of sites that behave similarly and can reuse code)
let contexts = {
    GithubCustomSearchEngines: ["github.com"],
    Startpage: ["startpage.com"],
    Steam: ["store.steampowered.com"],
    HookshotMedia: ["nintendolife.com", "pushsquare.com", "purexbox.com", "timeextension.com"],
    Metacritic: ["metacritic.com"],
    EpicGamesStore: ["store.epicgames.com"],
    Wikipedia: ["en.wikipedia.org"],
    OpenCritic: ["opencritic.com"],
    GOGcom: ["gog.com"]
};

let currentContext;
for(const [key, value] of Object.entries(contexts)) {
    if(value.some(v => window.location.hostname.includes(v))) {
        currentContext = key;
    }
}

// Non-"normal" context, i.e. contexts that don't place the actual search elements on the page
if(currentContext == "Startpage") {
    if(GSHSettings.OpenCriticHelperEnabled) {
        if(new URLSearchParams(document.location.search).get("ocgsh")) {
            window.location.href = document.querySelector("a.w-gl__result-title.result-link").href;
        }
    }
    die("Done!");
} else if(currentContext == "GithubCustomSearchEngines") {
    let customSearchEngineTable = document.querySelector(".markdown-body > table");
    if(customSearchEngineTable) {
        let rows = customSearchEngineTable.querySelectorAll("tbody > tr");
        for(let i = 0; i < rows.length; i++) {
            let installLink = rows[i].querySelector("td:nth-child(2) > a");
            let params = new URLSearchParams(installLink.href.substring(12));

            let titleCell = rows[i].querySelector("td:nth-child(1)");
            titleCell.innerHTML = `<img src="${params.get("icon")}" /> ${titleCell.innerHTML}`;

            if(GSHSettings.customProviders[params.get("uuid")]) {
                let newSpan = document.createElement("span");
                newSpan.innerHTML = "Added to GSH";
                installLink.insertAdjacentElement("afterend", newSpan);
                installLink.remove();
            } else {
                installLink.innerHTML = "Add to GSH";
                installLink.onclick = function(e){
                    e.preventDefault();

                    let params = new URLSearchParams(this.href.substring(12));
                    let uniqueID = params.get("uuid");

                    let newCustomProvider = {
                        uniqueID: uniqueID,
                        title: params.get("name"),
                        url: params.get("url"),
                        icon: params.get("icon"),
                        enabled: defaultEnabledContextsForNewSearchEngines
                    };

                    GSHSettings.customProviders[uniqueID] = newCustomProvider;
                    GM_setValue("GSHSettings", GSHSettings);
                    location.reload();
                    return false;
                };
            }
        }
    }
}

// Avoid save data errors when new contexts are added
if(!GSHSettings.defaultProviders[currentContext]) {
    GSHSettings.defaultProviders[currentContext] = {
        Startpage: {enabled: true},
        Google: {enabled: false},
        Metacritic: {enabled: true},
        OpenCritic: {enabled: true},
        eBay: {enabled: true},
        eBayUK: {enabled: false}
    };

    for(const provider in GSHSettings.customProviders) {
        GSHSettings.customProviders[provider].enabled[currentContext] = false;
    }
}

let providers = [
    { 
        uniqueID: "Startpage",
        enabled: GSHSettings.defaultProviders[currentContext].Startpage.enabled,
        title: "Startpage", 
        url: "https://www.startpage.com/sp/search?query=%search%", 
        icon: ""
    },
    {
        uniqueID: "Google",
        enabled: GSHSettings.defaultProviders[currentContext].Google.enabled,
        title: "Google",
        url: "https://www.google.com/search?q=%search%",
        icon: " "
    },
    {
        uniqueID: "Metacritic",
        enabled: GSHSettings.defaultProviders[currentContext].Metacritic.enabled,
        title: "Metacritic", 
        url: "https://www.metacritic.com/search/game/%searchPlus%/results", 
        icon: ""
    },
    {
        uniqueID: "OpenCritic",
        enabled: GSHSettings.defaultProviders[currentContext].OpenCritic.enabled,
        title: "OpenCritic", 
        url: "https://www.startpage.com/sp/search?query=site:opencritic.com+%search%&ocgsh=1", 
        icon: ""
    },
    {
        uniqueID: "eBay",
        enabled: GSHSettings.defaultProviders[currentContext].eBay.enabled,
        title: "eBay.com", 
        url: "https://www.ebay.com/sch/i.html?_nkw=%search%", 
        icon: ""
    },
    {
        uniqueID: "eBayUK",
        enabled: GSHSettings.defaultProviders[currentContext].eBayUK.enabled,
        title: "eBay United Kingdom", 
        url: "https://www.ebay.co.uk/sch/i.html?_nkw=%search%", 
        icon: ""
    }
];

let providersList = `<div id="GSHBuiltInSortable">`;
for(let i = 0; i < providers.length; i++) {
    providersList += `
        <span id="GSHBuiltinSearchSpan_${providers[i].uniqueID}" data-id-builtin="${i}">
            ${currentContext != "GithubCustomSearchEngines" ? `<span class="GSHBuiltInHandle" title="Drag to reorder search engines">≡</span>
            <input class="GSHInputField" type="checkbox" id="GSHBuiltinSearchCheck_${providers[i].uniqueID}"${providers[i].enabled ? " checked" : ""}>` : ""}
            <label class="GSHLabel" for="GSHBuiltinSearchCheck_${providers[i].uniqueID}" title="${encodeHTMLEntities(providers[i].url)}"><img src="${providers[i].icon}" width="16px" height="16px" /> ${providers[i].title}</label><br />
        </span>
    `;
}
providersList += "</div>";

let customProvidersList = `<div id="GSHCustomSortable">`;
let i = 0;
for(const provider in GSHSettings.customProviders) {
    customProvidersList += `
        <span id="GSHCustomSearchSpan_${GSHSettings.customProviders[provider].uniqueID}" data-id-custom="${provider}">
            ${currentContext != "GithubCustomSearchEngines" ? `<span class="GSHCustomHandle" title="Drag to reorder search engines">≡</span>
            <input class="GSHInputField" type="checkbox" id="GSHCustomSearchCheck_${GSHSettings.customProviders[provider].uniqueID}"${GSHSettings.customProviders[provider].enabled[currentContext] ? " checked" : ""}>` : ""}
            <label class="GSHLabel" for="GSHCustomSearchCheck_${GSHSettings.customProviders[provider].uniqueID}" title="${encodeHTMLEntities(GSHSettings.customProviders[provider].url)}"><img src="${GSHSettings.customProviders[provider].icon}" width="16px" height="16px" /> ${GSHSettings.customProviders[provider].title}</label> 
            <span class="GSHModifyCustomButton" id="GSHEdit_${GSHSettings.customProviders[provider].uniqueID}" title="Edit">[e]</span> 
            <span class="GSHModifyCustomButton" id="GSHDelete_${GSHSettings.customProviders[provider].uniqueID}" title="Delete">[d]</span><br />
        </span>
    `;
    i++;
}
customProvidersList += "</div>";

let newBox = document.createElement("div");
newBox.innerHTML = `
    <div id="GSHOuterDiv">
        <span id="GSHToggleButton" title="${GM_info.script.name}">🔍</span>
        <div id="GSHInnerDiv" class="GSHHidden">
            <div id="GSHLinkbox">[ <a href="https://github.com/xdpirate/GameSearchHelper" class="GSHLink" target="_blank" title="Visit GSH's Github repository">Github</a> | <a href="https://greasyfork.org/en/scripts/441809-game-search-helper/" class="GSHLink" target="_blank" title="Visit GSH's Greasy Fork page">Greasy Fork</a> ]</div>
            <div id="GSHHeader">${GM_info.script.name} v${GM_info.script.version}</div>
            <div id="GSHSettingsContent">
                Current context: <b>${currentContext}</b><br /><br />

                <div id="GSHWrapper">
                    <div class="GSHSearchContainer">
                        Built-in search engines:<br />
                        <div id="GSHBuiltInSearchEngines">
                            ${providersList}
                        </div>
                        ${currentContext != "GithubCustomSearchEngines" ? `Select: <span id="GSHBuiltInSelectAllButton" class="GSHLink">[all]</span> <span id="GSHBuiltInSelectNoneButton" class="GSHLink">[none]</span> <span id="GSHBuiltInSelectInvertButton" class="GSHLink">[invert]</span>` : ""}
                    </div>

                    <div class="GSHSearchContainer">
                        Custom search engines: <span id="GSHNewCustomSearchButton" title="Add a new custom search engine">[new]</span> <a id="GSHGetEnginesButton" href="https://github.com/xdpirate/GameSearchHelper/blob/main/CustomSearchEngines.md" target="_blank" title="Get some premade custom search engines from the Github repository">[get]</a><br />
                        <div id="GSHCustomSearchEngines">
                            ${customProvidersList}
                        </div>
                        ${currentContext != "GithubCustomSearchEngines" ? `Select: <span id="GSHCustomSelectAllButton" class="GSHLink">[all]</span> <span id="GSHCustomSelectNoneButton" class="GSHLink">[none]</span> <span id="GSHCustomSelectInvertButton" class="GSHLink">[invert]</span>` : ""}
                    </div>
                </div>

                <div id="GSHCustomSearchEngineDiv" class="GSHHidden">
                    <span id="GSHSearchEngineEditorHeader"></span> <span id="GSHCancelEditButton" title="Cancel editing this custom search engine">[cancel]</span><br /><br />
                    
                    Display name:<br />
                    <input type="text" class="GSHTextBox GSHInputField" id="GSHDisplayNameInput" title="The display name of the custom search engine"></input>
                    <br /><br />

                    Search URL <span title="Substitute the search term with %search% in the URL. If the site you are searching requires spaces to be replaced with plus-signs, use %searchPlus% instead.">[help]</span>:<br />
                    <input type="text" class="GSHTextBox GSHInputField" id="GSHSearchURLInput" title="The search URL of the custom search engine"></input>
                    <br /><br />

                    Icon (16x16, URL or <a href="https://en.wikipedia.org/wiki/Data_URI_scheme" class="GSHLink" target="_blank">image data URI</a>):<br />
                    <input type="text" class="GSHTextBox GSHInputField" id="GSHIconInput" title="The URL or data URI of the custom search engine's icon"></input>

                    <input type="hidden" id="GSHUniqueIDInput"></input>
                    <input type="hidden" id="GSHModifyMode" value="none"></input>
                </div>

                <div id="GSHOptionsDiv">
                    <input type="checkbox" class="GSHInputField" id="GSHOpenCriticHelperCheckbox"${GSHSettings.OpenCriticHelperEnabled ? " checked" : ""}>
                    <label class="GSHLabel" for="GSHOpenCriticHelperCheckbox">Open Startpage proxy results with one click</label> <sup><a href="https://github.com/xdpirate/GameSearchHelper/blob/main/README.md#options" class="GSHLink" target="_blank" title="Read about this option in the README on Github">?</a></sup><br />
                    <input type="checkbox" class="GSHInputField" id="GSHStripSpecialCharsCheckbox"${GSHSettings.StripSpecialCharsEnabled ? " checked" : ""}>
                    <label class="GSHLabel" for="GSHStripSpecialCharsCheckbox">Strip non-ASCII characters from search term (&trade;, &copy;, etc)</label><br />
                </div>

                <div id="GSHSaveDiv">
                    <input type="button" class="GSHInputButton" value="Save and reload" id="GSHSaveButton" title="Save your settings and reload the page for them to take effect"></input>
                </div>
            </div>
        <div>
    </div>
`;

document.body.append(newBox);

if(currentContext != "GithubCustomSearchEngines") {
    let builtInSortable = new Sortable(document.getElementById("GSHBuiltInSortable"), {
        group: "GSHBuiltInSearchEnginesSort",
        handle: ".GSHBuiltInHandle",
        dataIdAttr: 'data-id-builtin',
        store: {
            get: function (sortable) {
                var order = GM_getValue(`${sortable.options.group.name}_${currentContext}`, false);
                return order ? order.split('|') : [];
            },
            set: function (sortable) {
                var order = sortable.toArray();
                GM_setValue(`${sortable.options.group.name}_${currentContext}`, order.join('|'));
            },
        }
    });
    
    let customSortable = new Sortable(document.getElementById("GSHCustomSortable"), {
        group: "GSHCustomSearchEnginesSort",
        handle: ".GSHCustomHandle",
        dataIdAttr: 'data-id-custom',
        store: {
            get: function (sortable) {
                var order = GM_getValue(`${sortable.options.group.name}_${currentContext}`, false);
                return order ? order.split('|') : [];
            },
            set: function (sortable) {
                var order = sortable.toArray();
                GM_setValue(`${sortable.options.group.name}_${currentContext}`, order.join('|'));
            },
        }
    });

    document.getElementById("GSHBuiltInSelectAllButton").onclick = function() {
        let builtinSearchCheckboxes = document.querySelectorAll(`input[id^="GSHBuiltinSearchCheck_"]`);
        for(let i = 0; i < builtinSearchCheckboxes.length; i++) {
            builtinSearchCheckboxes[i].checked = true;
        }
    };

    document.getElementById("GSHBuiltInSelectNoneButton").onclick = function() {
        let builtinSearchCheckboxes = document.querySelectorAll(`input[id^="GSHBuiltinSearchCheck_"]`);
        for(let i = 0; i < builtinSearchCheckboxes.length; i++) {
            builtinSearchCheckboxes[i].checked = false;
        }
    };

    document.getElementById("GSHBuiltInSelectInvertButton").onclick = function() {
        let builtinSearchCheckboxes = document.querySelectorAll(`input[id^="GSHBuiltinSearchCheck_"]`);
        for(let i = 0; i < builtinSearchCheckboxes.length; i++) {
            builtinSearchCheckboxes[i].checked = !builtinSearchCheckboxes[i].checked;
        }
    };

    document.getElementById("GSHCustomSelectAllButton").onclick = function() {
        let customSearchCheckboxes = document.querySelectorAll(`input[id^="GSHCustomSearchCheck_"]`);
        for(let i = 0; i < customSearchCheckboxes.length; i++) {
            customSearchCheckboxes[i].checked = true;
        }
    };

    document.getElementById("GSHCustomSelectNoneButton").onclick = function() {
        let customSearchCheckboxes = document.querySelectorAll(`input[id^="GSHCustomSearchCheck_"]`);
        for(let i = 0; i < customSearchCheckboxes.length; i++) {
            customSearchCheckboxes[i].checked = false;
        }
    };

    document.getElementById("GSHCustomSelectInvertButton").onclick = function() {
        let customSearchCheckboxes = document.querySelectorAll(`input[id^="GSHCustomSearchCheck_"]`);
        for(let i = 0; i < customSearchCheckboxes.length; i++) {
            customSearchCheckboxes[i].checked = !customSearchCheckboxes[i].checked;
        }
    };
}

document.getElementById("GSHSaveButton").onclick = function() {
    saveData();
    location.reload();
};

document.getElementById("GSHToggleButton").onclick = function() {
    document.getElementById('GSHInnerDiv').classList.toggle('GSHHidden');
};

document.getElementById("GSHNewCustomSearchButton").onclick = function() {
    let uniqueID = crypto.randomUUID();
    document.getElementById("GSHSearchEngineEditorHeader").innerHTML = "New custom search engine";
    document.getElementById("GSHDisplayNameInput").value = "";
    document.getElementById("GSHSearchURLInput").value = "https://example.com/?s=%search%";
    document.getElementById("GSHIconInput").value = "";

    document.getElementById("GSHUniqueIDInput").value = uniqueID;
    document.getElementById("GSHModifyMode").value = "add";

    document.getElementById("GSHCustomSearchEngineDiv").classList.remove("GSHHidden");
};

let editButtons = document.querySelectorAll(`span[id^="GSHEdit_`);
for(let i = 0; i < editButtons.length; i++) {
    editButtons[i].onclick = function() {
        let uniqueID = this.id.match(/^GSHEdit_(.+)$/)[1];
        document.getElementById("GSHSearchEngineEditorHeader").innerHTML = "Editing " + GSHSettings.customProviders[uniqueID].title;
        document.getElementById("GSHDisplayNameInput").value = GSHSettings.customProviders[uniqueID].title;
        document.getElementById("GSHSearchURLInput").value = GSHSettings.customProviders[uniqueID].url;
        document.getElementById("GSHIconInput").value = GSHSettings.customProviders[uniqueID].icon;

        document.getElementById("GSHUniqueIDInput").value = uniqueID;
        document.getElementById("GSHModifyMode").value = "edit";

        document.getElementById("GSHCustomSearchEngineDiv").classList.remove("GSHHidden");
    };
}

let deleteButtons = document.querySelectorAll(`span[id^="GSHDelete_`);
for(let i = 0; i < deleteButtons.length; i++) {
    deleteButtons[i].onclick = function() {
        let uniqueID = this.id.match(/^GSHDelete_(.+)$/)[1];
        if(confirm("Are you sure you want to delete this custom search engine?\nName: " + GSHSettings.customProviders[uniqueID].title + "\nUnique ID: " + uniqueID)) {
            saveData();
            delete GSHSettings.customProviders[uniqueID];
            GM_setValue("GSHSettings", GSHSettings);
            location.reload();
        }
    };
}

document.getElementById("GSHCancelEditButton").onclick = function() {
    document.getElementById("GSHCustomSearchEngineDiv").classList.add("GSHHidden");
    document.getElementById("GSHModifyMode").value = "none";
};

let game = {
    set name(name) {
        this._name = name.trim();
    },
    get name() {
        if(GSHSettings.StripSpecialCharsEnabled) {
            return encodeURIComponent(this._name.replace(/[^\x00-\x7F]/g, "").replace(/  /g, " "));
        } else {
            return encodeURIComponent(this._name);
        }
    },
    get namePlus() {
        if(GSHSettings.StripSpecialCharsEnabled) {
            return encodeURIComponent(this._name.replace(/[^\x00-\x7F]/g, "").replace(/  /g, " ")).replace(/%20/g, "+");
        } else {
            return encodeURIComponent(this._name).replace(/%20/g, "+");
        }
    },
    get namePlusAlphanumericOnly() {
        return encodeURIComponent(this._name.replace(/[^A-Za-z0-9 ]/g, "")).replace(/%20/g, "+");
    }
};

// "Normal" contexts where search elements will be placed on the page
if(currentContext == "Steam") {
    if(window.location.href.includes("/app/")) {
        let headerElement = document.getElementById("appHubAppName");

        if(headerElement) {
            game.name = headerElement.innerHTML;
            addGSHBox(game, headerElement, "", "GSHIcon");
        } else {
            die("Could not find header element");
        }
    } else if(window.location.href.includes("/bundle/")) {
        let multiGames = document.querySelectorAll("div.bundle_package_item");
        if(multiGames) {
            for(let i = 0; i < multiGames.length; i++) {
                let containerElement = multiGames[i].querySelector("div.tab_item_content");
                game.name = containerElement.querySelector("div.tab_item_name").innerHTML;

                let newAdjElem = multiGames[i].insertAdjacentElement("afterend", document.createElement("div"));
                addGSHBox(game, newAdjElem, "GSHSteamBundleContainer", "GSHIcon");
            }    
        }
    }
} else if(currentContext == "HookshotMedia") {
    let gameOverview, headerElement;
    if(window.location.href.includes("/games/")) {
        headerElement = document.querySelector("div.info");

        if(headerElement) {
            game.name = headerElement.querySelector("h1.title > a").innerHTML.match(/^(.+) \(.+\)$/)[1];
            addGSHBox(game, headerElement, "HMContainer", "HMIcon");
        } else {
            die("Could not find header element");            
        }
    } else {
        gameOverview = document.getElementById("game-overview");

        if(gameOverview) {
            headerElement = gameOverview.querySelector("header.widget-header");
            game.name = gameOverview.querySelector("div.body > div.items > div.item > p.definition > a").innerHTML;
            addGSHBox(game, headerElement, "HMContainer", "HMIcon"); 
        } else {
            // No single game overview found, is there a "featured games" multi-game section?
            let multiGames = document.querySelector("section.block.related-games");
            if(multiGames) {
                let games = multiGames.querySelectorAll("li.item");
                for(let i = 0; i < games.length; i++) {
                    game.name = games[i].querySelector("h2.heading > a").innerHTML.match(/(.*?)</)[1].trim();
                    addGSHBox(game, games[i], "HMContainer", "HMIcon");
                }
            } else {
                die("Could not find game overview or featured games");
            }
        }
    }
} else if(currentContext == "Metacritic") {
    if(window.location.href == "https://www.metacritic.com/game" || window.location.href.match(/\/game\/(?!legacy)([A-Za-z0-9\-]+)$/)) {
        // Game home page or any system's game home page, /game/legacy is excluded because it acts like a browse page
        let mainBox = document.getElementById("main");
        if(mainBox) {
            let observer = new MutationObserver(function(event) {
                let bodyBox = document.getElementById("main").querySelector("div.body_wrap.has_genre_nav");
                if(bodyBox) {
                    if(!bodyBox.querySelector(".GSHIcon")) { // Avoids infinite recursion
                        let multiGames = bodyBox.querySelectorAll(`table.clamp-list > tbody > tr:not(.spacer)`);
                        if(multiGames) {
                            for(let i = 0; i < multiGames.length; i++) {
                                let headerElement = multiGames[i].querySelector("td.clamp-summary-wrap > a.title");
                                game.name = headerElement.querySelector("h3").innerHTML;

                                let newAdjElem = headerElement.insertAdjacentElement("afterend", document.createElement("div"));
                                addGSHBox(game, newAdjElem, "GSHMetacriticListContainer", "GSHIcon");
                            }
                        }
                    }
                }
            });
        
            observer.observe(mainBox, {subtree: true, childList: true});

            let multiGames = document.querySelectorAll(`table.clamp-list > tbody > tr:not(.spacer)`);
            if(multiGames) {
                for(let i = 0; i < multiGames.length; i++) {
                    let headerElement = multiGames[i].querySelector("td.clamp-summary-wrap > a.title");
                    if(headerElement) {
                        game.name = headerElement.querySelector("h3").innerHTML;

                        let newAdjElem = headerElement.insertAdjacentElement("afterend", document.createElement("div"));
                        addGSHBox(game, newAdjElem, "GSHMetacriticListContainer", "GSHIcon");
                    }
                }
            } else {
                die("Could not find game elements");
            }
        }
    } else if(window.location.href.includes("/browse/games/") || window.location.href.endsWith("/game/legacy")) {
        // Any game list / browse page
        let multiGames = document.querySelectorAll(`table.clamp-list > tbody > tr:not(.spacer)`);
        if(multiGames) {
            for(let i = 0; i < multiGames.length; i++) {
                let headerElement = multiGames[i].querySelector("td.clamp-summary-wrap > a.title");
                game.name = headerElement.querySelector("h3").innerHTML;

                let newAdjElem = headerElement.insertAdjacentElement("afterend", document.createElement("div"));
                addGSHBox(game, newAdjElem, "GSHMetacriticListContainer", "GSHIcon");
            }
        } else {
            die("Could not find game elements");
        }
    } else if(window.location.href.match(/\/game\/([A-Za-z0-9\-]+)\/([A-Za-z0-9\-]+)$/)) {
        // Any specific game page
        let titleElement = document.querySelector("div.product_title");
        if(titleElement) {
            game.name = titleElement.querySelector("a > h1").innerHTML;
            addGSHBox(game, titleElement, "", "GSHIcon");
        } else {
            die("Could not get title element");
        }
    }
} else if(currentContext == "EpicGamesStore") {
    let observer = new MutationObserver(function(event) {
        if(document.documentElement) {
            let titleElement = document.querySelector(`div[data-component="PDPTitleHeader"] > span[data-component="Text"]`);
            if(titleElement) {
                if(titleElement.GSHDetected != true) {
                    titleElement.GSHDetected = true;
                    game.name = titleElement.innerHTML;
                    addGSHBox(game, titleElement.insertAdjacentElement("afterend", document.createElement("div")), "", "GSHIcon");
                }
            }
        }
    });
    
    observer.observe(document.documentElement, {subtree: true, childList: true});
} else if(currentContext == "Wikipedia") {
    let blacklistedCategories = ["fictional video games", "nintendo hardware", "video game locations"];
    let categories = document.querySelectorAll("div#catlinks > div#mw-normal-catlinks > ul > li > a");
    
    let isGame = false;
    for(let i = 0; i < categories.length; i++) {
        let category = categories[i].innerHTML.toLowerCase();
        if(blacklistedCategories.includes(category) || category.includes("characters in video games") || category.includes("about video games")) {
            isGame = false;
            break;
        }

        if(category.includes("video games") || category.includes("video game franchises")) {
            isGame = true;
        }
    }

    if(isGame) {
        let headerElement = document.querySelector("h1#firstHeading");
        game.name = headerElement.querySelector("i").innerHTML;
        addGSHBox(game, headerElement.insertAdjacentElement("afterend", document.createElement("div")), "GSHWPBox", "GSHIcon");
    }
} else if(currentContext == "OpenCritic") {
    let observer = new MutationObserver(function(event) {
        if(document.documentElement) {
            let titleElement = document.querySelector("h1.mb-0");
            if(titleElement) {
                if(titleElement.GSHDetected != true) {
                    titleElement.GSHDetected = true;
                    game.name = titleElement.innerHTML;
                    addGSHBox(game, titleElement.insertAdjacentElement("afterend", document.createElement("div")), "GSHOCContainer", "GSHIcon");
                }

                let titleUpdateObserver = new MutationObserver(function(event) {
                    let titleElement = document.querySelector("h1.mb-0");
                    document.getElementsByClassName("GSHOCContainer")[0].parentNode.remove();
                    game.name = titleElement.innerHTML;
                    addGSHBox(game, titleElement.insertAdjacentElement("afterend", document.createElement("div")), "GSHOCContainer", "GSHIcon");
                });

                titleUpdateObserver.observe(titleElement, {subtree: true, characterData: true});
            }
        }
    });

    observer.observe(document.documentElement, {subtree: true, childList: true});
} else if(currentContext == "GOGcom") {
    let titleElement = document.querySelector(`div.productcard-basics > h1.productcard-basics__title`);
    if(titleElement) {
        titleElement.GSHDetected = true;
        game.name = titleElement.innerHTML;
        addGSHBox(game, titleElement.insertAdjacentElement("afterend", document.createElement("div")), "GSHGOGContainer", "GSHIcon");
    }
}

saveData();

function addGSHBox(game, containerElement, boxClass, iconClass) {
    let GSHBox = document.createElement("div");
    if(boxClass !== "") {
        GSHBox.classList.add(boxClass);
    }

    let builtinSortOrder = GM_getValue(`GSHBuiltInSearchEnginesSort_${currentContext}`, false);
    if(!builtinSortOrder) {
        builtinSortOrder = [0, 1, 2, 3, 4, 5];
    } else {
        builtinSortOrder = builtinSortOrder.split("|");
    }

    for(let i = 0; i < providers.length; i++) {
        if(providers[builtinSortOrder[i]].enabled) {
            let searchLink = document.createElement("span");
            searchLink.classList.add(iconClass);

            let title = providers[builtinSortOrder[i]].title;
            let searchURL = providers[builtinSortOrder[i]].url.replace("%search%", game.name).replace("%searchPlus%", game.namePlus).replace("%searchPlusAlpha%", game.namePlusAlphanumericOnly);
            let icon = providers[builtinSortOrder[i]].icon;

            searchLink.innerHTML = `
                <a target="_blank" title="${title}" href="${searchURL}"><img src="${icon}" width="16px" height="16px" /></a>
            `;

            GSHBox.append(searchLink);
        }
    }

    let i = 0;
    let customProvidersArray = [];
    for(const provider in GSHSettings.customProviders) {
        customProvidersArray[i] = provider;
        i++;
    }

    let customSortOrder = GM_getValue(`GSHCustomSearchEnginesSort_${currentContext}`, false);
    if(customSortOrder) {
        customSortOrder = customSortOrder.split("|");
    }

    for(let i = 0; i < customProvidersArray.length; i++) {
        if(customSortOrder) {
            provider = customSortOrder[i];
        } else {
            provider = customProvidersArray[i];
        }

        if(GSHSettings.customProviders[provider]) {
            if(GSHSettings.customProviders[provider].enabled[currentContext]) {
                let searchLink = document.createElement("span");
                searchLink.classList.add(iconClass);

                let title = GSHSettings.customProviders[provider].title;
                let searchURL = GSHSettings.customProviders[provider].url.replace("%search%", game.name).replace("%searchPlus%", game.namePlus).replace("%searchPlusAlpha%", game.namePlusAlphanumericOnly);
                let icon = GSHSettings.customProviders[provider].icon;

                searchLink.innerHTML = `
                    <a target="_blank" title="${title}" href="${searchURL}"><img src="${icon}" width="16px" height="16px" /></a>
                `;

                GSHBox.append(searchLink);
            }
        }
        
    }
    
    containerElement.append(GSHBox);
}

function saveData() {
    if(currentContext != "GithubCustomSearchEngines") {
        let builtinSearchCheckboxes = document.querySelectorAll(`input[id^="GSHBuiltinSearchCheck_"]`);
        for(let i = 0; i < builtinSearchCheckboxes.length; i++) {
            let uniqueID = builtinSearchCheckboxes[i].id.match(/GSHBuiltinSearchCheck_(.+)$/)[1];
            GSHSettings.defaultProviders[currentContext][uniqueID].enabled = builtinSearchCheckboxes[i].checked;
        }

        let customSearchCheckboxes = document.querySelectorAll(`input[id^="GSHCustomSearchCheck_"]`);
        for(let i = 0; i < customSearchCheckboxes.length; i++) {
            let uniqueID = customSearchCheckboxes[i].id.match(/GSHCustomSearchCheck_(.+)$/)[1];
            GSHSettings.customProviders[uniqueID].enabled[currentContext] = customSearchCheckboxes[i].checked;
        }
    }
    

    if(!document.getElementById("GSHCustomSearchEngineDiv").classList.contains("GSHHidden")) {
        let modifyMode = document.getElementById("GSHModifyMode").value;
        if(modifyMode == "add") {
            let uniqueID = document.getElementById("GSHUniqueIDInput").value;

            let newCustomProvider = {
                uniqueID: uniqueID,
                title: document.getElementById("GSHDisplayNameInput").value,
                url: document.getElementById("GSHSearchURLInput").value,
                icon: document.getElementById("GSHIconInput").value,
                enabled: defaultEnabledContextsForNewSearchEngines
            };

            GSHSettings.customProviders[uniqueID] = newCustomProvider;
        } else if(modifyMode == "edit") {
            let uniqueID = document.getElementById("GSHUniqueIDInput").value;

            GSHSettings.customProviders[uniqueID].title = document.getElementById("GSHDisplayNameInput").value;
            GSHSettings.customProviders[uniqueID].url = document.getElementById("GSHSearchURLInput").value;
            GSHSettings.customProviders[uniqueID].icon = document.getElementById("GSHIconInput").value;
        }

    }

    GSHSettings.OpenCriticHelperEnabled = document.getElementById("GSHOpenCriticHelperCheckbox").checked;
    GSHSettings.StripSpecialCharsEnabled = document.getElementById("GSHStripSpecialCharsCheckbox").checked;

    GM_setValue("GSHSettings", GSHSettings);
}

function encodeHTMLEntities(text) { // modified from SO answer to include the full range
    text = text.replace(/[\u0000-\u9999<>\&]/g, function(i) {
        return '&#'+i.charCodeAt(0)+';';
    });

    return text;
}

function die(errorMsg) {
    throw `[${GM_info.script.name}] ${errorMsg} (${currentContext})`;   
}