NOTICE: By continued use of this site you understand and agree to the binding Terms of Service and Privacy Policy.
// ==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); })();