93Akkord / google-search-extended

// ==UserScript==
// @name        google-search-extended
// @namespace   https://openuserjs.org/users/93Akkord
// @version     0.2.14
// @description Google Search Extended
// @license     MIT
// @author      93Akkord
// @namespace   https://github.com/93Akkord
// @match       https:\/\/*.google.com\/search?*
// @include     /^https?:\/\/(?:www|encrypted|ipv[46])\.google\.[^/]+/(?:$|[#?]|search|webhp)/
// @run-at      document-start
// @grant		unsafeWindow
// @grant       GM.registerMenuCommand
// @grant       GM_registerMenuCommand
// @grant       GM_unregisterMenuCommand
// @grant       GM_xmlhttpRequest
// @grant       GM_getValue
// @grant       GM_setValue
// @grant       GM_listValues
// @connect     *
// @require     https://code.jquery.com/jquery-3.2.1.min.js
// @require     https://cdn.jsdelivr.net/npm/arrive@2.4.1/src/arrive.min.js
// @require     https://openuserjs.org/src/libs/93Akkord/akkd-common.js
// @copyright   2022+, Michael Barros (https://openuserjs.org/users/93Akkord)
// @license     CC-BY-NC-SA-4.0; https://creativecommons.org/licenses/by-nc-sa/4.0/legalcode
// @license     GPL-3.0-or-later; https://www.gnu.org/licenses/gpl-3.0.txt
// @icon        https://www.google.com/favicon.ico
// @updateURL   https://openuserjs.org/meta/93Akkord/google-search-extended.meta.js
// @downloadURL https://openuserjs.org/install/93Akkord/google-search-extended.user.js
// ==/UserScript==

// Add `Available nearby` to any `shopping result`.  Add to following to the url: &tbs=local_avail:1

(async function () {
    const ls = new LocalStorageEx();
    const DEV_TAG = '[akkd]';

    let DEBUG = ls.get('google-search-extended-debug', false);

    // #region Helper Functions

    /**
     * Get difference in days between two dates.
     *
     * @param {Date} a
     * @param {Date} b
     * @returns
     */
    function dateDiffInDays(a, b) {
        let _MS_PER_DAY = 1000 * 60 * 60 * 24;

        // Discard the time and time-zone information.
        let utc1 = Date.UTC(a.getFullYear(), a.getMonth(), a.getDate());
        let utc2 = Date.UTC(b.getFullYear(), b.getMonth(), b.getDate());

        return Math.floor((utc1 - utc2) / _MS_PER_DAY);
    }

    /**
     *
     *
     * @author Michael Barros <michaelcbarros@gmail.com>
     * @param {any[]} msg
     */
    function logDebug(...msg) {
        if (DEBUG) {
            console.debug(DEV_TAG, ...msg);
        }
    }

    /**
     *
     *
     * @author Michael Barros <michaelcbarros@gmail.com>
     * @param {boolean} [init=false]
     */
    function registerDebugMenuCommand(init = false) {
        if (!init) {
            DEBUG = !DEBUG;

            ls.set('google-search-extended-debug', DEBUG);
        }

        let debugMenuCommandID = GM_registerMenuCommand(`Debug ${DEBUG ? 'on' : 'off'}`, () => {
            GM_unregisterMenuCommand(debugMenuCommandID);

            registerDebugMenuCommand();
        });
    }

    /**
     *
     *
     * @author Michael Barros <michaelcbarros@gmail.com>
     * @param {string} text
     * @param {string} [nodeType='div']
     * @returns {HTMLElement}
     */
    function getElementByTextContent(text, nodeType = 'div') {
        let xpath = `//${nodeType}[text()='${text}']`;

        /** @type {HTMLElement} */
        let elem = document.evaluate(xpath, document, null, XPathResult.FIRST_ORDERED_NODE_TYPE, null).singleNodeValue;

        return elem;
    }

    /**
     *
     *
     * @author Michael Barros <michaelcbarros@gmail.com>
     * @param {string} text
     * @param {string} [nodeType='div']
     * @returns {HTMLElement}
     */
    function getElementByTextContentContains(text, nodeType = 'div') {
        let xpath = `//${nodeType}[contains(text(),'${text}')]`;

        /** @type {HTMLElement} */
        let elem = document.evaluate(xpath, document, null, XPathResult.FIRST_ORDERED_NODE_TYPE, null).singleNodeValue;

        return elem;
    }

    // #endregion Helper Functions

    // #region Sites Menu

    // #region Icon Data URIs

    let pypiUri = 'data:image/svg+xml;base64,<svg xmlns="http://www.w3.org/2000/svg" width="65.812" height="58" viewBox="0 0 65.812035 58.000001"><g fill="#a29d86" stroke="#ccc" stroke-width=".214" stroke-linejoin="bevel"><path d="M18.93 18.826l9.323 3.394v10.957l-9.323-3.394z" style="isolation:auto;mix-blend-mode:normal" color="#000" overflow="visible" fill="#e9e9ff"/><path d="M9.47 22.27v10.957l9.46-3.444V18.826z" style="isolation:auto;mix-blend-mode:normal" color="#000" overflow="visible" fill="#353564"/><path d="M9.47 33.227l9.322 3.393 9.46-3.443-9.322-3.394z" style="isolation:auto;mix-blend-mode:normal" color="#000" overflow="visible" fill="#afafde"/><path d="M9.47 22.27l9.322 3.393 9.46-3.443-9.322-3.394z" fill="#f7f7f4"/><path d="M18.792 25.663V36.62l9.46-3.443V22.22z" fill="#fff"/><path d="M9.47 22.27l9.322 3.393V36.62L9.47 33.227z" fill="#efeeea"/><path style="isolation:auto;mix-blend-mode:normal" d="M28.293 11.166l9.323 3.393v10.957l-9.323-3.393z" color="#000" overflow="visible" fill="#e9e9ff"/><path style="isolation:auto;mix-blend-mode:normal" d="M18.833 14.609v10.957l9.46-3.443V11.166z" color="#000" overflow="visible" fill="#353564"/><path style="isolation:auto;mix-blend-mode:normal" d="M18.833 25.566l9.322 3.393 9.461-3.443-9.323-3.393z" color="#000" overflow="visible" fill="#afafde"/><path d="M18.833 14.609l9.322 3.393 9.461-3.443-9.323-3.393z" fill="#f7f7f4"/><path d="M28.155 18.002V28.96l9.461-3.443V14.559z" fill="#fff"/><path d="M18.833 14.609l9.322 3.393V28.96l-9.322-3.393z" fill="#efeeea"/><path d="M9.567 33.297l9.322 3.393v10.957l-9.322-3.393z" style="isolation:auto;mix-blend-mode:normal" color="#000" overflow="visible" fill="#e9e9ff"/><path d="M.106 36.74v10.957l9.46-3.443V33.297z" style="isolation:auto;mix-blend-mode:normal" color="#000" overflow="visible" fill="#353564"/><path d="M.106 47.697L9.43 51.09l9.46-3.443-9.322-3.393z" style="isolation:auto;mix-blend-mode:normal" color="#000" overflow="visible" fill="#afafde"/><path d="M.106 36.74l9.323 3.393 9.46-3.443-9.322-3.393z" fill="#f7f7f4"/><path d="M9.429 40.133V51.09l9.46-3.443V36.69z" fill="#fff"/><path d="M.106 36.74l9.323 3.393V51.09L.106 47.697z" fill="#efeeea"/></g><g fill="#a29d86" stroke="#ccc" stroke-width=".214" stroke-linejoin="bevel"><path style="isolation:auto;mix-blend-mode:normal" d="M18.93 36.702l9.323 3.393v10.957l-9.323-3.393z" color="#000" overflow="visible" fill="#e9e9ff"/><path style="isolation:auto;mix-blend-mode:normal" d="M9.47 40.145v10.957l9.46-3.443V36.702z" color="#000" overflow="visible" fill="#353564"/><path style="isolation:auto;mix-blend-mode:normal" d="M9.47 51.102l9.322 3.393 9.46-3.443-9.322-3.393z" color="#000" overflow="visible" fill="#afafde"/><path d="M9.47 40.145l9.322 3.393 9.46-3.443-9.322-3.393z" fill="#f7f7f4"/><path d="M18.792 43.538v10.957l9.46-3.443V40.095z" fill="#fff"/><path d="M9.47 40.145l9.322 3.393v10.957L9.47 51.102z" fill="#efeeea"/></g><g fill="#a29d86" stroke="#ccc" stroke-width=".214" stroke-linejoin="bevel"><path d="M18.93 25.636l9.323 3.393v10.957l-9.323-3.393z" style="isolation:auto;mix-blend-mode:normal" color="#000" overflow="visible" fill="#e9e9ff"/><path d="M9.47 29.08v10.956l9.46-3.443V25.636z" style="isolation:auto;mix-blend-mode:normal" color="#000" overflow="visible" fill="#353564"/><path d="M9.47 40.036l9.322 3.394 9.46-3.444-9.322-3.393z" style="isolation:auto;mix-blend-mode:normal" color="#000" overflow="visible" fill="#afafde"/><path d="M9.47 29.08l9.322 3.393 9.46-3.444-9.322-3.393z" fill="#f7f7f4"/><path d="M18.792 32.473V43.43l9.46-3.444V29.03z" fill="#fff"/><path d="M9.47 29.08l9.322 3.393V43.43L9.47 40.036z" fill="#efeeea"/></g><g fill="#a29d86" stroke="#ccc" stroke-width=".214" stroke-linejoin="bevel"><path style="isolation:auto;mix-blend-mode:normal" d="M56.383 29.892l9.323 3.393v10.957l-9.323-3.393z" color="#000" overflow="visible" fill="#e9e9ff"/><path style="isolation:auto;mix-blend-mode:normal" d="M46.923 33.335v10.957l9.46-3.443V29.892z" color="#000" overflow="visible" fill="#353564"/><path style="isolation:auto;mix-blend-mode:normal" d="M46.923 44.292l9.322 3.394 9.46-3.444-9.322-3.393z" color="#000" overflow="visible" fill="#afafde"/><path d="M46.923 33.335l9.322 3.394 9.46-3.444-9.322-3.393z" fill="#f7f7f4"/><path d="M56.245 36.729v10.957l9.46-3.444V33.285z" fill="#fff"/><path d="M46.923 33.335l9.322 3.394v10.957l-9.322-3.394z" fill="#efeeea"/></g><g transform="translate(-1683.66 -513.275) scale(.65177)"><g fill="#a29d86" stroke="#ccc" stroke-width=".328" stroke-linejoin="bevel"><path style="isolation:auto;mix-blend-mode:normal" d="M2655.349 838.594l14.303 5.206v16.811l-14.303-5.206z" color="#000" overflow="visible" fill="#e9e9ff"/><path style="isolation:auto;mix-blend-mode:normal" d="M2640.834 843.877v16.811l14.515-5.283v-16.81z" color="#000" overflow="visible" fill="#353564"/><path style="isolation:auto;mix-blend-mode:normal" d="M2640.834 860.688l14.303 5.206 14.515-5.283-14.303-5.206z" color="#000" overflow="visible" fill="#afafde"/><path d="M2640.834 843.877l14.303 5.206 14.515-5.283-14.303-5.206z" fill="#f7f7f4"/><path d="M2655.137 849.083v16.811l14.515-5.283v-16.81z" fill="#ffd242"/><path d="M2640.834 843.877l14.303 5.206v16.811l-14.303-5.206z" fill="#efeeea"/></g><circle transform="skewY(-20) scale(.9397 1)" cy="1823.899" cx="2835.218" r="2.967" fill="#fff"/></g><g fill="#a29d86" stroke="#ccc" stroke-width=".214" stroke-linejoin="bevel"><path d="M37.657 36.702l9.322 3.393v10.957l-9.322-3.393z" style="isolation:auto;mix-blend-mode:normal" color="#000" overflow="visible" fill="#e9e9ff"/><path d="M28.196 40.145v10.957l9.46-3.443V36.702z" style="isolation:auto;mix-blend-mode:normal" color="#000" overflow="visible" fill="#353564"/><path d="M28.196 51.102l9.323 3.393 9.46-3.443-9.322-3.393z" style="isolation:auto;mix-blend-mode:normal" color="#000" overflow="visible" fill="#afafde"/><path d="M28.196 40.145l9.323 3.393 9.46-3.443-9.322-3.393z" fill="#f7f7f4"/><path d="M37.519 43.538v10.957l9.46-3.443V40.095z" fill="#ffd242"/><path d="M28.196 40.145l9.323 3.393v10.957l-9.323-3.393z" fill="#efeeea"/></g><g fill="#a29d86" stroke="#ccc" stroke-width=".214" stroke-linejoin="bevel"><path style="isolation:auto;mix-blend-mode:normal" d="M28.293 40.107l9.323 3.393v10.957l-9.323-3.393z" color="#000" overflow="visible" fill="#e9e9ff"/><path style="isolation:auto;mix-blend-mode:normal" d="M18.833 43.55v10.957l9.46-3.443V40.107z" color="#000" overflow="visible" fill="#353564"/><path style="isolation:auto;mix-blend-mode:normal" d="M18.833 54.507l9.322 3.393 9.46-3.443-9.322-3.393z" color="#000" overflow="visible" fill="#afafde"/><path d="M18.833 43.55l9.322 3.393 9.46-3.443-9.322-3.393z" fill="#f7f7f4"/><path d="M28.155 46.943V57.9l9.46-3.443V43.5z" fill="#fff"/><path d="M18.833 43.55l9.322 3.393V57.9l-9.322-3.393z" fill="#efeeea"/></g><g fill="#a29d86" stroke="#ccc" stroke-width=".214" stroke-linejoin="bevel"><path d="M56.383 18.826l9.323 3.394v10.957l-9.323-3.394z" style="isolation:auto;mix-blend-mode:normal" color="#000" overflow="visible" fill="#e9e9ff"/><path d="M46.923 22.27v10.957l9.46-3.444V18.826z" style="isolation:auto;mix-blend-mode:normal" color="#000" overflow="visible" fill="#353564"/><path d="M46.923 33.227l9.322 3.393 9.46-3.443-9.322-3.394z" style="isolation:auto;mix-blend-mode:normal" color="#000" overflow="visible" fill="#afafde"/><path d="M46.923 22.27l9.322 3.393 9.46-3.443-9.322-3.394z" fill="#f7f7f4"/><path d="M56.245 25.663V36.62l9.46-3.443V22.22z" fill="#ffd242"/><path d="M46.923 22.27l9.322 3.393V36.62l-9.322-3.393z" fill="#efeeea"/></g><g fill="#a29d86" stroke="#ccc" stroke-width=".214" stroke-linejoin="bevel"><path style="isolation:auto;mix-blend-mode:normal" d="M56.383 7.76l9.323 3.394V22.11l-9.323-3.393z" color="#000" overflow="visible" fill="#e9e9ff"/><path style="isolation:auto;mix-blend-mode:normal" d="M46.923 11.204v10.957l9.46-3.443V7.76z" color="#000" overflow="visible" fill="#353564"/><path style="isolation:auto;mix-blend-mode:normal" d="M46.923 22.161l9.322 3.393 9.46-3.443-9.322-3.393z" color="#000" overflow="visible" fill="#afafde"/><path d="M46.923 11.204l9.322 3.393 9.46-3.443-9.322-3.393z" fill="#ffc91d"/><path d="M56.245 14.597v10.957l9.46-3.443V11.154z" fill="#ffd242"/><path d="M46.923 11.204l9.322 3.393v10.957l-9.322-3.393z" fill="#efeeea"/></g><g fill="#a29d86" stroke="#ccc" stroke-width=".214" stroke-linejoin="bevel"><path style="isolation:auto;mix-blend-mode:normal" d="M47.02 22.231l9.322 3.393v10.957l-9.322-3.393z" color="#000" overflow="visible" fill="#e9e9ff"/><path style="isolation:auto;mix-blend-mode:normal" d="M37.56 25.675v10.957l9.46-3.444V22.231z" color="#000" overflow="visible" fill="#353564"/><path style="isolation:auto;mix-blend-mode:normal" d="M37.56 36.632l9.322 3.393 9.46-3.444-9.322-3.393z" color="#000" overflow="visible" fill="#afafde"/><path d="M37.56 25.675l9.322 3.393 9.46-3.444-9.322-3.393z" fill="#f7f7f4"/><path d="M46.882 29.068v10.957l9.46-3.444V25.624z" fill="#ffd242"/><path d="M37.56 25.675l9.322 3.393v10.957l-9.323-3.393z" fill="#efeeea"/></g><g fill="#a29d86" stroke="#ccc" stroke-width=".214" stroke-linejoin="bevel"><path d="M47.02 11.166l9.322 3.393v10.957l-9.322-3.393z" style="isolation:auto;mix-blend-mode:normal" color="#000" overflow="visible" fill="#e9e9ff"/><path d="M37.56 14.609v10.957l9.46-3.443V11.166z" style="isolation:auto;mix-blend-mode:normal" color="#000" overflow="visible" fill="#353564"/><path d="M37.56 25.566l9.322 3.393 9.46-3.443-9.322-3.393z" style="isolation:auto;mix-blend-mode:normal" color="#000" overflow="visible" fill="#afafde"/><path d="M37.56 14.609l9.322 3.393 9.46-3.443-9.322-3.393z" fill="#f7f7f4"/><path d="M46.882 18.002V28.96l9.46-3.443V14.559z" fill="#3775a9"/><path d="M37.56 14.609l9.322 3.393V28.96l-9.323-3.393z" fill="#efeeea"/></g><g fill="#a29d86" stroke="#ccc" stroke-width=".214" stroke-linejoin="bevel"><path style="isolation:auto;mix-blend-mode:normal" d="M47.02.1l9.322 3.393V14.45l-9.322-3.393z" color="#000" overflow="visible" fill="#e9e9ff"/><path style="isolation:auto;mix-blend-mode:normal" d="M37.56 3.543V14.5l9.46-3.443V.1z" color="#000" overflow="visible" fill="#353564"/><path style="isolation:auto;mix-blend-mode:normal" d="M37.56 14.5l9.322 3.393 9.46-3.443-9.322-3.393z" color="#000" overflow="visible" fill="#afafde"/><path d="M37.56 3.543l9.322 3.393 9.46-3.443L47.02.1z" fill="#2f6491"/><path d="M46.882 6.936v10.957l9.46-3.443V3.493z" fill="#3775a9"/><path d="M37.56 3.543l9.322 3.393v10.957L37.559 14.5z" fill="#efeeea"/></g><g fill="#a29d86" stroke="#ccc" stroke-width=".214" stroke-linejoin="bevel"><path d="M37.657 25.636l9.322 3.393v10.957l-9.322-3.393z" style="isolation:auto;mix-blend-mode:normal" color="#000" overflow="visible" fill="#e9e9ff"/><path d="M28.196 29.08v10.956l9.46-3.443V25.636z" style="isolation:auto;mix-blend-mode:normal" color="#000" overflow="visible" fill="#353564"/><path d="M28.196 40.036l9.323 3.394 9.46-3.444-9.322-3.393z" style="isolation:auto;mix-blend-mode:normal" color="#000" overflow="visible" fill="#afafde"/><path d="M28.196 29.08l9.323 3.393 9.46-3.444-9.322-3.393z" fill="#f7f7f4"/><path d="M37.519 32.473V43.43l9.46-3.444V29.03z" fill="#ffd242"/><path d="M28.196 29.08l9.323 3.393V43.43l-9.323-3.394z" fill="#efeeea"/></g><g fill="#a29d86" stroke="#ccc" stroke-width=".214" stroke-linejoin="bevel"><path style="isolation:auto;mix-blend-mode:normal" d="M37.657 14.57l9.322 3.394V28.92l-9.322-3.394z" color="#000" overflow="visible" fill="#e9e9ff"/><path style="isolation:auto;mix-blend-mode:normal" d="M28.196 18.014V28.97l9.46-3.444V14.57z" color="#000" overflow="visible" fill="#353564"/><path style="isolation:auto;mix-blend-mode:normal" d="M28.196 28.97l9.323 3.394 9.46-3.443-9.322-3.394z" color="#000" overflow="visible" fill="#afafde"/><path d="M28.196 18.014l9.323 3.393 9.46-3.443-9.322-3.394z" fill="#f7f7f4"/><path d="M37.519 21.407v10.957l9.46-3.443V17.964z" fill="#3775a9"/><path d="M28.196 18.014l9.323 3.393v10.957l-9.323-3.393z" fill="#efeeea"/></g><g fill="#a29d86" stroke="#ccc" stroke-width=".214" stroke-linejoin="bevel"><path style="isolation:auto;mix-blend-mode:normal" d="M28.293 29.04l9.323 3.394v10.957l-9.323-3.393z" color="#000" overflow="visible" fill="#e9e9ff"/><path style="isolation:auto;mix-blend-mode:normal" d="M18.833 32.484v10.957l9.46-3.443V29.04z" color="#000" overflow="visible" fill="#353564"/><path style="isolation:auto;mix-blend-mode:normal" d="M18.833 43.441l9.322 3.393 9.46-3.443-9.322-3.393z" color="#000" overflow="visible" fill="#afafde"/><path d="M18.833 32.484l9.322 3.393 9.46-3.443-9.322-3.393z" fill="#f7f7f4"/><path d="M28.155 35.877v10.957l9.46-3.443V32.434z" fill="#3775a9"/><path d="M18.833 32.484l9.322 3.393v10.957l-9.322-3.393z" fill="#2f6491"/></g><g fill="#a29d86" stroke="#ccc" stroke-width=".214" stroke-linejoin="bevel"><path d="M28.293 17.975l9.323 3.393v10.957l-9.323-3.393z" style="isolation:auto;mix-blend-mode:normal" color="#000" overflow="visible" fill="#e9e9ff"/><path d="M18.833 21.419v10.957l9.46-3.444V17.975z" style="isolation:auto;mix-blend-mode:normal" color="#000" overflow="visible" fill="#353564"/><path d="M18.833 32.376l9.322 3.393 9.46-3.444-9.322-3.393z" style="isolation:auto;mix-blend-mode:normal" color="#000" overflow="visible" fill="#afafde"/><path d="M18.833 21.419l9.322 3.393 9.46-3.444-9.322-3.393z" fill="#2f6491"/><path d="M28.155 24.812v10.957l9.46-3.444V21.368z" fill="#3775a9"/><path d="M18.833 21.419l9.322 3.393v10.957l-9.322-3.393z" fill="#2f6491"/></g><g transform="translate(-1683.66 -513.275) scale(.65177)"><g fill="#a29d86" stroke="#ccc" stroke-width=".328" stroke-linejoin="bevel"><path style="isolation:auto;mix-blend-mode:normal" d="M2640.983 792.885l14.304 5.206v16.811l-14.304-5.206z" color="#000" overflow="visible" fill="#e9e9ff"/><path style="isolation:auto;mix-blend-mode:normal" d="M2626.468 798.168v16.811l14.515-5.283v-16.811z" color="#000" overflow="visible" fill="#353564"/><path style="isolation:auto;mix-blend-mode:normal" d="M2626.468 814.979l14.304 5.206 14.515-5.283-14.304-5.206z" color="#000" overflow="visible" fill="#afafde"/><path d="M2626.468 798.168l14.304 5.206 14.515-5.283-14.304-5.206z" fill="#2f6491"/><path d="M2640.772 803.374v16.811l14.515-5.283v-16.811z" fill="#3775a9"/><path d="M2626.468 798.168l14.304 5.206v16.811l-14.304-5.206z" fill="#2f6491"/></g><circle transform="skewY(-20) scale(.9397 1)" cy="1772.922" cx="2816.017" r="2.967" fill="#fff"/></g></svg>';

    // #endregion Icon Data URIs

    class SitesMenu {
        constructor(init = true) {
            this.akkdSitesButtonId = 'akkd-sites-button';

            if (init) {
                this.init();
            }
        }

        init() {
            $(document).arrive(`form[action='/search']>div>div:first-child`, async (elem) => {
                this._createButtonCss();

                // this.container = $('.sKb6pb');
                this.container = [this._getButtonContainer()];
                this.sitesButton = this._createSitesButton();

                this.sitesButton.hide();

                // Append to main container if using mobile
                if (window.orientation != undefined) {
                    $(this.container[0]).append(this.sitesButton);
                } else {
                    $(this.container[0]).append(this.sitesButton);
                }

                let benchmark = new Benchmark({ logger: logDebug });
                let benchmarkName = 'menuItems creation';

                benchmark.start(benchmarkName);

                let icons = [this._getFavIcon('stackoverflow.com'), this._getFavIcon('gist.github.com'), this._getFavIcon('reddit.com'), this._getFavIcon('codepen.io'), this._getFavIcon('codesandbox.io'), this._getFavIcon('npmjs.com'), this._getFavIcon('forum.xda-developers.com')];

                // Pre-fetching icons all at once. This is faster if it's a fresh run.
                icons = await Promise.all(icons);

                let menuItems = [
                    //
                    this._createMenuItem('Clear', 'akkd-sites-clear', `<img alt="" src="data:image/gif;base64,R0lGODlhAQABAAD/ACwAAAAAAQABAAACADs=" />`),
                    this._createMenuItem('stackoverflow', 'akkd-sites-stackoverflow', `<img alt="" src="${await this._getFavIcon('stackoverflow.com')}" />`, 'stackoverflow.com'),
                    this._createMenuItem('gist', 'akkd-sites-gist', `<img alt="" src="${await this._getFavIcon('gist.github.com')}" />`, 'gist.github.com'),
                    this._createMenuItem('reddit', 'akkd-sites-reddit', `<img alt="" src="${await this._getFavIcon('reddit.com')}" />`, 'reddit.com'),
                    this._createMenuItem('codepen', 'akkd-sites-codepen', `<img alt="" src="${await this._getFavIcon('codepen.io')}" />`, 'codepen.io'),
                    this._createMenuItem('codesandbox', 'akkd-sites-codesandbox', `<img alt="" src="${await this._getFavIcon('codesandbox.io')}" />`, 'codesandbox.io'),
                    this._createMenuItem('npmjs', 'akkd-sites-npmjs', `<img alt="" src="${await this._getFavIcon('npmjs.com')}" />`, 'npmjs.com'),
                    this._createMenuItem('pypi', 'akkd-sites-pypi', `<img alt="" src="${pypiUri}" />`, 'pypi.org'),
                    this._createMenuItem('xda', 'akkd-sites-xda', `<img alt="" src="${await this._getFavIcon('forum.xda-developers.com')}" />`, 'forum.xda-developers.com'),
                ];

                benchmark.stop(benchmarkName);

                this._createSitesMenu(menuItems);

                this.menuContainer.hide();

                this.sitesButton.on('click', (ev) => {
                    ev.preventDefault();

                    this.show();
                });

                this._positionButtonContainer();

                this._setupResizeObserver();

                this.menu
                    .on('mousedown', (ev) => {
                        ev.preventDefault();
                    })
                    .on('blur', (ev) => {
                        this.menuContainer.hide();
                    });

                $(document.body).append(this.hiddenMenuContainer);

                this.sitesButton.show();
            });
        }

        _setupResizeObserver(maxTries = 10) {
            let tries = 0;

            let _worker = () => {
                tries++;

                try {
                    // this.resizeObserver = new ResizeObserver(this._positionButtonContainer).observe($('.logo')[0].nextElementSibling);
                    this.resizeObserver = new ResizeObserver(this._positionButtonContainer).observe($(`form[action='/search'][role='search']`).find(`button`)[0].parentElement);
                } catch (error) {
                    if (tries < maxTries) {
                        setTimeout(_worker, 100);
                    }
                }
            };

            _worker();
        }

        show() {
            let { left, top } = this._getLeftAndTop();

            this.menuContainer[0].style.top = `${top}px`;
            this.menuContainer[0].style.left = `${left}px`;

            this.menuContainer.show();

            /** @type {HTMLElement} */ (this.menu[0]).focus();
        }

        removeSitesButton() {
            $(`#${this.akkdSitesButtonId}`).remove();
        }

        _positionButtonContainer() {
            let tries = 0;

            let _worker = () => {
                tries++;

                try {
                    /** @type {HTMLElement} */
                    let buttonContainerElem = $('.akkd-button-container')[0];

                    /** @type {HTMLElement} */
                    // let searchBarElem = $('.logo')[0].nextElementSibling;
                    let searchBarElem = $(`div[jsname='RNNXgb']`)[0];

                    let searchBarWidth = $(searchBarElem).rect().width;
                    let searchBarHeight = $(searchBarElem).rect().height;

                    buttonContainerElem.style.height = `${searchBarHeight}px`;
                    buttonContainerElem.style.left = `${searchBarWidth + 40}px`;
                    buttonContainerElem.style.top = '0px'; // `${searchBarHeight * -1}px`;
                } catch (error) {
                    if (tries < maxTries) {
                        setTimeout(_worker, 100);
                    }
                }
            };

            _worker();
        }

        _getLeftAndTop() {
            /** @type {number} */
            let left = this.sitesButton.rect().left + getWindow().scrollX - this.sitesButton.width() / 2.7;

            left = left - Math.max(0, left + this.menuContainer.actual('width') + 20 - $(document.body).rect().width);

            /** @type {number} */
            let top = this.sitesButton.rect().top + getWindow().scrollY + $(this.sitesButton.children()[0]).rect().height + 1000;

            return {
                left,
                top,
            };
        }

        _createButtonCss() {
            let styleElem = document.createElement('style');

            styleElem.innerHTML = `html:not(.zAoYTe) [href]{outline:0}.akkd-button-container-2{display:inline-block;margin-left:6px;vertical-align:top}.akkd-button-container-2:first-of-type{margin-left:0}.akkd-button-container-2{height:100%}.akkd-button-container-3{box-sizing:border-box;flex-direction:row;-webkit-transition:background-color .1s;align-items:center;backdrop-filter:blur(4px);background-color:#fff;border:1px solid #dadce0;border-radius:20px;display:flex;height:100%;min-width:48px;padding:0 14px;text-align:center}.akkd-button-container-3:not(:focus-visible){outline:0}.akkd-button-container-3,.akkd-button-container-3:active,.akkd-button-container-3:hover,.akkd-button-container-3:link,.akkd-button-container-3:visited{color:#202124;text-decoration:none}.akkd-button-container-3:not(.OvGWO):hover{background-color:#f1f3f4;border-color:#dadce0}.akkd-button-text{font-family:Google Sans,Roboto,arial,sans-serif;font-size:14px;min-width:0;text-align:center}.akkd-button-text::first-letter{text-transform:uppercase}.akkd-button-svg{fill:#4285f4;height:18px;margin-right:6px;width:18px}.akkd-button-container-4{padding-left:12px}.akkd-button-container{display:inline-block;height:40px;margin-left:6px;vertical-align:top;position:absolute;top:-43px;left:705px}@font-face{font-family:'Google Sans';font-style:normal;font-weight:400;font-display:optional;src:url(//fonts.gstatic.com/s/googlesans/v14/4UaGrENHsxJlGDuGo1OIlL3Kwp5MKg.woff2) format('woff2');unicode-range:U+0301,U+0400-045F,U+0490-0491,U+04B0-04B1,U+2116}@font-face{font-family:'Google Sans';font-style:normal;font-weight:400;font-display:optional;src:url(//fonts.gstatic.com/s/googlesans/v14/4UaGrENHsxJlGDuGo1OIlL3Nwp5MKg.woff2) format('woff2');unicode-range:U+0370-03FF}@font-face{font-family:'Google Sans';font-style:normal;font-weight:400;font-display:optional;src:url(//fonts.gstatic.com/s/googlesans/v14/4UaGrENHsxJlGDuGo1OIlL3Bwp5MKg.woff2) format('woff2');unicode-range:U+0102-0103,U+0110-0111,U+0128-0129,U+0168-0169,U+01A0-01A1,U+01AF-01B0,U+1EA0-1EF9,U+20AB}@font-face{font-family:'Google Sans';font-style:normal;font-weight:400;font-display:optional;src:url(//fonts.gstatic.com/s/googlesans/v14/4UaGrENHsxJlGDuGo1OIlL3Awp5MKg.woff2) format('woff2');unicode-range:U+0100-024F,U+0259,U+1E00-1EFF,U+2020,U+20A0-20AB,U+20AD-20CF,U+2113,U+2C60-2C7F,U+A720-A7FF}@font-face{font-family:'Google Sans';font-style:normal;font-weight:400;font-display:optional;src:url(//fonts.gstatic.com/s/googlesans/v14/4UaGrENHsxJlGDuGo1OIlL3Owp4.woff2) format('woff2');unicode-range:U+0000-00FF,U+0131,U+0152-0153,U+02BB-02BC,U+02C6,U+02DA,U+02DC,U+2000-206F,U+2074,U+20AC,U+2122,U+2191,U+2193,U+2212,U+2215,U+FEFF,U+FFFD}@font-face{font-family:Roboto;font-style:normal;font-weight:400;font-display:optional;src:url(//fonts.gstatic.com/s/roboto/v18/KFOmCnqEu92Fr1Mu72xKOzY.woff2) format('woff2');unicode-range:U+0460-052F,U+1C80-1C88,U+20B4,U+2DE0-2DFF,U+A640-A69F,U+FE2E-FE2F}@font-face{font-family:Roboto;font-style:normal;font-weight:400;font-display:optional;src:url(//fonts.gstatic.com/s/roboto/v18/KFOmCnqEu92Fr1Mu5mxKOzY.woff2) format('woff2');unicode-range:U+0301,U+0400-045F,U+0490-0491,U+04B0-04B1,U+2116}@font-face{font-family:Roboto;font-style:normal;font-weight:400;font-display:optional;src:url(//fonts.gstatic.com/s/roboto/v18/KFOmCnqEu92Fr1Mu7mxKOzY.woff2) format('woff2');unicode-range:U+1F00-1FFF}@font-face{font-family:Roboto;font-style:normal;font-weight:400;font-display:optional;src:url(//fonts.gstatic.com/s/roboto/v18/KFOmCnqEu92Fr1Mu4WxKOzY.woff2) format('woff2');unicode-range:U+0370-03FF}@font-face{font-family:Roboto;font-style:normal;font-weight:400;font-display:optional;src:url(//fonts.gstatic.com/s/roboto/v18/KFOmCnqEu92Fr1Mu7WxKOzY.woff2) format('woff2');unicode-range:U+0102-0103,U+0110-0111,U+0128-0129,U+0168-0169,U+01A0-01A1,U+01AF-01B0,U+1EA0-1EF9,U+20AB}@font-face{font-family:Roboto;font-style:normal;font-weight:400;font-display:optional;src:url(//fonts.gstatic.com/s/roboto/v18/KFOmCnqEu92Fr1Mu7GxKOzY.woff2) format('woff2');unicode-range:U+0100-024F,U+0259,U+1E00-1EFF,U+2020,U+20A0-20AB,U+20AD-20CF,U+2113,U+2C60-2C7F,U+A720-A7FF}@font-face{font-family:Roboto;font-style:normal;font-weight:400;font-display:optional;src:url(//fonts.gstatic.com/s/roboto/v18/KFOmCnqEu92Fr1Mu4mxK.woff2) format('woff2');unicode-range:U+0000-00FF,U+0131,U+0152-0153,U+02BB-02BC,U+02C6,U+02DA,U+02DC,U+2000-206F,U+2074,U+20AC,U+2122,U+2191,U+2193,U+2212,U+2215,U+FEFF,U+FFFD}@font-face{font-family:Roboto;font-style:normal;font-weight:700;font-display:optional;src:url(//fonts.gstatic.com/s/roboto/v18/KFOlCnqEu92Fr1MmWUlfCRc4EsA.woff2) format('woff2');unicode-range:U+0460-052F,U+1C80-1C88,U+20B4,U+2DE0-2DFF,U+A640-A69F,U+FE2E-FE2F}@font-face{font-family:Roboto;font-style:normal;font-weight:700;font-display:optional;src:url(//fonts.gstatic.com/s/roboto/v18/KFOlCnqEu92Fr1MmWUlfABc4EsA.woff2) format('woff2');unicode-range:U+0301,U+0400-045F,U+0490-0491,U+04B0-04B1,U+2116}@font-face{font-family:Roboto;font-style:normal;font-weight:700;font-display:optional;src:url(//fonts.gstatic.com/s/roboto/v18/KFOlCnqEu92Fr1MmWUlfCBc4EsA.woff2) format('woff2');unicode-range:U+1F00-1FFF}@font-face{font-family:Roboto;font-style:normal;font-weight:700;font-display:optional;src:url(//fonts.gstatic.com/s/roboto/v18/KFOlCnqEu92Fr1MmWUlfBxc4EsA.woff2) format('woff2');unicode-range:U+0370-03FF}@font-face{font-family:Roboto;font-style:normal;font-weight:700;font-display:optional;src:url(//fonts.gstatic.com/s/roboto/v18/KFOlCnqEu92Fr1MmWUlfCxc4EsA.woff2) format('woff2');unicode-range:U+0102-0103,U+0110-0111,U+0128-0129,U+0168-0169,U+01A0-01A1,U+01AF-01B0,U+1EA0-1EF9,U+20AB}@font-face{font-family:Roboto;font-style:normal;font-weight:700;font-display:optional;src:url(//fonts.gstatic.com/s/roboto/v18/KFOlCnqEu92Fr1MmWUlfChc4EsA.woff2) format('woff2');unicode-range:U+0100-024F,U+0259,U+1E00-1EFF,U+2020,U+20A0-20AB,U+20AD-20CF,U+2113,U+2C60-2C7F,U+A720-A7FF}@font-face{font-family:Roboto;font-style:normal;font-weight:700;font-display:optional;src:url(//fonts.gstatic.com/s/roboto/v18/KFOlCnqEu92Fr1MmWUlfBBc4.woff2) format('woff2');unicode-range:U+0000-00FF,U+0131,U+0152-0153,U+02BB-02BC,U+02C6,U+02DA,U+02DC,U+2000-206F,U+2074,U+20AC,U+2122,U+2191,U+2193,U+2212,U+2215,U+FEFF,U+FFFD}.akkd-button-container-3.akkd-button-container-4:hover{cursor:pointer;user-select:none}.akkd-menu-item-text-01{margin-right:5px;vertical-align:text-bottom}.akkd-menu-item-text-02{display:inline-block;fill:currentColor}.akkd-menu-item-text-02 svg{display:block;height:100%;width:100%}.akkd-hide-focus-ring{outline:0}.akkd-hide-focus-ring:hover{background-color:rgba(0,0,0,.1)}.akkd-hidden-menu-container{z-index:1001;position:absolute;top:-1000px}.akkd-menu{background-color:#fff}.akkd-menu akkd-menu-item{display:block;font-size:14px;line-height:23px;white-space:nowrap}.akkd-menu akkd-menu-item a{display:block;padding-top:4px;padding-bottom:4px;cursor:pointer}.akkd-menu akkd-menu-item a,.akkd-menu akkd-menu-item a:hover,.akkd-menu akkd-menu-item a:visited{text-decoration:inherit;color:inherit}.akkd-menu-style-01 akkd-menu-item{color:#5f6368}.akkd-menu-style-01 akkd-menu-item a{line-height:16px;padding-top:8px;padding-bottom:8px}.akkd-menu-container{display:block;position:absolute;border-radius:8px;box-shadow:0 2px 10px 0 rgba(0,0,0,.2)}.akkd-menu-style-02{border:none;display:block;outline:0}.akkd-menu-style-02{border-radius:8px}.akkd-menu-style-02{padding:5px 0}.akkd-menu-item{display:block;position:relative}.akkd-menu-item-inner-01{overflow:hidden;padding:0 16px;text-overflow:ellipsis;vertical-align:middle;white-space:nowrap}.akkd-menu-item-inner-01.akkd-menu-item-inner-02{padding:0}.akkd-menu-item-inner-02>*{padding:0 16px}.akkd-menu-item:hover{cursor:pointer}.akkd-menu-item{cursor:default}.akkd-menu-item-inner-01{line-height:23px}.akkd-menu-item:active{background-color:rgba(0,0,0,.1)}.akkd-menu-item-text-01.akkd-menu-item-text-02>img{height:16px;width:16px}.akkd-btn .GOE98c,.akkd-btn a,.akkd-btn.hdtb-msel,.t2vtad{color:#5f6368;text-decoration:none;display:inline-block;padding:0 12px;padding:8px 16px 8px 16px;padding:17px 12px 11px 10px}.akkd-btn{margin:-5px 0 0 15px;display:inline-block;z-index:99999}.akkd-btn:hover{cursor:pointer}.akkd-btn a:active{color:#1a73e8}.akkd-btn.hdtb-msel{color:#1a73e8}.cCvmNd .akkd-btn.hdtb-msel{border-bottom:none}.akkd-btn.hdtb-msel:hover{cursor:pointer}.akkd-btn.hdtb-msel:active{background:0 0}.akkd-btn a{color:#5f6368}@media (max-width:1300px){div[jsname=RNNXgb]{width:538px}}.o6juZc{margin-left:0}`;

            document.head.appendChild(styleElem);

            let shouldUseDarkmode = window.matchMedia('(prefers-color-scheme: dark)').matches;

            if (shouldUseDarkmode && this._isDarkMode()) {
                let styleElemDark = document.createElement('style');

                styleElemDark.innerHTML = `.akkd-button-container-3 {
    background-color: #202124;
    border: 1px solid #3c4043;
    color: #f1f3f4;
}

.akkd-button-container-3:hover {
    background-color: #303134 !important;
    border: 1px solid #3c4043 !important;
    color: #f1f3f4 !important;
}

.akkd-menu {
    background-color: #202124;
}

.akkd-hide-focus-ring:hover {
    background-color: rgba(255,255,255,0.1)
}

.akkd-menu-style-01 akkd-menu-item {
    color: #bdc1c6;
}
`;

                document.head.appendChild(styleElemDark);
            }
        }

        _isDarkMode() {
            try {
                return document.querySelector('#logo > img').src.includes('light');
            } catch (error) {
                let metaElems = document.getElementsByTagName('meta');
                let metas = {};

                for (let i = 0; i < metaElems.length; i++) {
                    let metaElem = metaElems[i];

                    if (metaElem.name.trim() == '') {
                        continue;
                    }

                    let val = metaElem.getAttribute('content');

                    try {
                        val = JSON.parse(val);
                    } catch (error) {}

                    metas[metaElem.name] = val;
                }

                return 'color-scheme' in metas && metas['color-scheme'] === 'dark';
            }
        }

        /**
         *
         *
         * @author Michael Barros <michaelcbarros@gmail.com>
         * @returns {JQuery<JQuery.Node[]>}
         * @memberof SitesMenu
         */
        _createSitesButton() {
            // hdtb-mitem
            let htmlOld = `<div id="${this.akkdSitesButtonId}" class="discuss-btn hdtb-imb akkd-btn">
        <a class="q qs" href="#">
            <span class="MbEPDb z1asCe SaPW2b" style="height: 16px; line-height: 16px; width: 16px; margin-left: -20px">
                <svg focusable="false" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24">
                    <path d="M12 8c1.1 0 2-.9 2-2s-.9-2-2-2-2 .9-2 2 .9 2 2 2zm0 2c-1.1 0-2 .9-2 2s.9 2 2 2 2-.9 2-2-.9-2-2-2zm0 6c-1.1 0-2 .9-2 2s.9 2 2 2 2-.9 2-2-.9-2-2-2z"></path>
                </svg>
            </span>Sites
        </a>
        </div>`;

            let html = `<div class="akkd-button-container" id="${this.akkdSitesButtonId}">
    <div class="akkd-button-container-2">
        <div class="akkd-button-container-3 akkd-button-container-4">
            <svg class="akkd-button-svg" focusable="false" viewBox="0 0 24 24">
                <path d="M12 8c1.1 0 2-.9 2-2s-.9-2-2-2-2 .9-2 2 .9 2 2 2zm0 2c-1.1 0-2 .9-2 2s.9 2 2 2 2-.9 2-2-.9-2-2-2zm0 6c-1.1 0-2 .9-2 2s.9 2 2 2 2-.9 2-2-.9-2-2-2z"></path>
            </svg>
            <div class="akkd-button-text">Sites</div>
        </div>
    </div>
</div>`;

            let sitesButton = $($.parseHTML(html));

            return sitesButton;
        }

        /**
         *
         *
         * @author Michael Barros <michaelcbarros@gmail.com>
         * @returns {HTMLElement}
         * @memberof SitesMenu
         */
        _getButtonContainerOld() {
            let buttonContainer = getElementByTextContent('All', 'span').parentElement.parentElement.parentElement.children[1]; // .children[0].children[0]

            return buttonContainer;
        }

        /**
         *
         *
         * @author Michael Barros <michaelcbarros@gmail.com>
         * @returns {HTMLElement}
         * @memberof SitesMenu
         */
        _getButtonContainer() {
            /** @type {HTMLElement} */
            let buttonContainer = $(`form[action='/search']>div>div:first-child`)[0]; // $('div.eTnfyc')[0];

            // buttonContainer.style.position = 'absolute';

            return buttonContainer;
        }

        /**
         *
         *
         * @author Michael Barros <michaelcbarros@gmail.com>
         * @param {JQuery<JQuery.Node[]>[]} menuItems
         * @returns
         * @memberof SitesMenu
         */
        _createSitesMenu(menuItems) {
            let html = `<div class="akkd-hidden-menu-container">
    <div class="akkd-menu-container" style="z-index: 1;">
        <akkd-menu class="akkd-menu akkd-menu-style-01 akkd-menu-style-02" tabindex="0">

        </akkd-menu>
    </div>
    </div>`;

            this.hiddenMenuContainer = $($.parseHTML(html));
            this.menuContainer = $(this.hiddenMenuContainer[0].children[0]);
            this.menu = $(this.hiddenMenuContainer[0].children[0].children[0]);

            for (let i = 0; i < menuItems.length; i++) {
                let menuItem = menuItems[i];

                this.menu.append(menuItem);
            }

            getWindow().menuContainer = this.menuContainer;

            return {
                hiddenMenuContainer: this.hiddenMenuContainer,
                menuContainer: this.menuContainer,
                menu: this.menu,
            };
        }

        /**
         *
         *
         * @author Michael Barros <michaelcbarros@gmail.com>
         * @param {string} caption
         * @param {string} id
         * @param {string} svg
         * @param {string} url
         * @returns {JQuery<JQuery.Node[]>}
         * @memberof SitesMenu
         */
        _createMenuItem(caption, id, svg, url) {
            onclick = onclick || function () {};

            let html = `<akkd-menu-item id="${id}" class=".akkd-menu-item akkd-hide-focus-ring">
    <div class="akkd-menu-item-inner-01 akkd-menu-item-inner-02">
        <a tabindex="-1">
            <span class="akkd-menu-item-text-01 akkd-menu-item-text-02" style="height: 16px; width: 16px">
                ${svg}
            </span>${caption}
        </a>
    </div>
    </akkd-menu-item>`;

            let menuItem = $($.parseHTML(html));
            let self = this;

            menuItem.on('click', function (ev) {
                ev.preventDefault();

                let searchElem = self._getSearchInputElem();

                let value = url != null ? `${searchElem.value.replace(/site:.*$/g, '').trim()} site:${url}` : `${searchElem.value.replace(/site:.*$/g, '').trim()}`;

                searchElem.value = value;

                $(searchElem).closest('form')[0].submit();

                $(this).parent().parent().hide();
            });

            return menuItem;
        }

        /**
         *
         *
         * @author Michael Barros <michaelcbarros@gmail.com>
         * @param {string} url
         * @returns {string | null}
         * @memberof SitesMenu
         */
        _getCachedFavIcon(url) {
            let iconStorage = ls.get('icons', {});

            let now = new Date();

            try {
                let dateSaved = new Date(iconStorage[url].refreshDate);
                let diffDays = dateDiffInDays(now, dateSaved);

                if (diffDays > 15) {
                    return null;
                } else {
                    return iconStorage[url].icon;
                }
            } catch (error) {
                return null;
            }
        }

        /**
         *
         *
         * @author Michael Barros <michaelcbarros@gmail.com>
         * @param {string} url
         * @returns {Promise<string>}
         * @memberof SitesMenu
         */
        async _getFavIcon(url) {
            // Remove subdomain if present
            url = url.trim().replace(/^(?:[a-z]+\:\/{2})?(?:[\w-]+\.([\w-]+\.\w+))$/, '$1');

            let _url = `https://www.google.com/s2/favicons?domain=${url}`;

            let icon = this._getCachedFavIcon(url);

            if (icon) {
                return icon;
            }

            return new Promise(async (resolve, reject) => {
                let res = await GM_fetch(_url, {
                    // credentials: 'include',
                    headers: {
                        accept: 'text/html,application/xhtml+xml,application/xml;q=0.9,image/webp,image/apng,*/*;q=0.8,application/signed-exchange;v=b3;q=0.9',
                        'accept-language': 'en-US,en;q=0.9',
                    },
                    body: null,
                    method: 'GET',
                    // mode: 'cors',
                });

                let blob = await res.blob();

                let reader = new FileReader();

                reader.onload = function () {
                    let iconStorage = ls.get('icons', {});

                    iconStorage[url] = {
                        refreshDate: new Date().toISOString(),
                        icon: this.result,
                    };

                    ls.set('icons', iconStorage);

                    resolve(this.result);
                };

                reader.onerror = function (ev) {
                    console.error(DEV_TAG, 'Error:', ev);
                };

                reader.readAsDataURL(blob);
            });
        }

        /**
         *
         *
         * @author Michael Barros <michaelcbarros@gmail.com>
         * @returns {HTMLInputElement}
         * @memberof SitesMenu
         */
        _getSearchInputElem() {
            /** @type {HTMLInputElement} */
            let searchElem = $('[name="q"]')[0];

            return searchElem;
        }

        /**
         *
         *
         * @author Michael Barros <michaelcbarros@gmail.com>
         * @returns {HTMLElement}
         * @memberof SitesMenu
         */
        _getGoogleSearchButton() {
            let googleSearchButton = $('.Tg7LZd')[0];

            return googleSearchButton;
        }
    }

    function initSitesMenu() {
        let sitesMenu = new SitesMenu();
    }

    // #endregion Sites Menu

    /**
     *
     *
     * @author Michael Barros <michaelcbarros@gmail.com>
     */
    async function init() {
        // when, options, func, args) {
        const DEFAULT_OPTIONS = {
            use_vanilla: false,
        };

        let when = arguments[0];
        let options = typeof arguments[1] == 'object' ? arguments[1] : {};
        let func = typeof arguments[1] == 'object' ? arguments[2] : arguments[1];
        let args = typeof arguments[1] == 'object' ? arguments[3] : arguments[2];

        options = Object.assign(DEFAULT_OPTIONS, options);

        async function runCallback() {
            if (args && args.length > 0) {
                await func(...args);
            } else {
                await func();
            }
        }

        if (when == 'start') {
            await runCallback();
        } else if (when == 'ready') {
            if (!options.use_vanilla) {
                $(document).ready(async (e) => {
                    await runCallback();
                });
            } else {
                document.addEventListener('DOMContentLoaded', async (e) => {
                    await runCallback();
                });
            }
        } else if (when == 'loaded') {
            if (!options.use_vanilla) {
                $(document).on('readystatechange', async (e) => {
                    if (e.target.readyState == 'complete') {
                        await runCallback();
                    }
                });
            } else {
                document.addEventListener('readystatechange', async (e) => {
                    if (e.target.readyState === 'complete') {
                        await runCallback();
                    }
                });
            }
        }
    }

    registerDebugMenuCommand(true);
    exposeGlobalVariables([
        { name: 'GM_xmlhttpRequest', value: GM_xmlhttpRequest },
        { name: 'GM_getValue', value: GM_getValue },
        { name: 'GM_setValue', value: GM_setValue },
        { name: 'GM_listValues', value: GM_listValues },
        { name: 'SitesMenu', value: SitesMenu },
        { name: 'LocalStorageEx', value: LocalStorageEx },
        { name: 'ls', value: ls },

        { name: '$', value: $ },
    ]);

    await init('start', initSitesMenu);
})();