Google Navigation

Navigate through Google Search with a set of hotkeys.

このスクリプトの質問や評価の投稿はこちら通報はこちらへお寄せください。
// ==UserScript==
// @name                Google Navigation
// @name:zh-TW          Google 快覽
// @name:zh-CN          Google 快覽
// @namespace           veringsek
// @match               http://*.google.com/search*
// @match               http://*.google.ad/search*
// @match               http://*.google.ae/search*
// @match               http://*.google.com.af/search*
// @match               http://*.google.com.ag/search*
// @match               http://*.google.com.ai/search*
// @match               http://*.google.al/search*
// @match               http://*.google.am/search*
// @match               http://*.google.co.ao/search*
// @match               http://*.google.com.ar/search*
// @match               http://*.google.as/search*
// @match               http://*.google.at/search*
// @match               http://*.google.com.au/search*
// @match               http://*.google.az/search*
// @match               http://*.google.ba/search*
// @match               http://*.google.com.bd/search*
// @match               http://*.google.be/search*
// @match               http://*.google.bf/search*
// @match               http://*.google.bg/search*
// @match               http://*.google.com.bh/search*
// @match               http://*.google.bi/search*
// @match               http://*.google.bj/search*
// @match               http://*.google.com.bn/search*
// @match               http://*.google.com.bo/search*
// @match               http://*.google.com.br/search*
// @match               http://*.google.bs/search*
// @match               http://*.google.bt/search*
// @match               http://*.google.co.bw/search*
// @match               http://*.google.by/search*
// @match               http://*.google.com.bz/search*
// @match               http://*.google.ca/search*
// @match               http://*.google.cd/search*
// @match               http://*.google.cf/search*
// @match               http://*.google.cg/search*
// @match               http://*.google.ch/search*
// @match               http://*.google.ci/search*
// @match               http://*.google.co.ck/search*
// @match               http://*.google.cl/search*
// @match               http://*.google.cm/search*
// @match               http://*.google.cn/search*
// @match               http://*.google.com.co/search*
// @match               http://*.google.co.cr/search*
// @match               http://*.google.com.cu/search*
// @match               http://*.google.cv/search*
// @match               http://*.google.com.cy/search*
// @match               http://*.google.cz/search*
// @match               http://*.google.de/search*
// @match               http://*.google.dj/search*
// @match               http://*.google.dk/search*
// @match               http://*.google.dm/search*
// @match               http://*.google.com.do/search*
// @match               http://*.google.dz/search*
// @match               http://*.google.com.ec/search*
// @match               http://*.google.ee/search*
// @match               http://*.google.com.eg/search*
// @match               http://*.google.es/search*
// @match               http://*.google.com.et/search*
// @match               http://*.google.fi/search*
// @match               http://*.google.com.fj/search*
// @match               http://*.google.fm/search*
// @match               http://*.google.fr/search*
// @match               http://*.google.ga/search*
// @match               http://*.google.ge/search*
// @match               http://*.google.gg/search*
// @match               http://*.google.com.gh/search*
// @match               http://*.google.com.gi/search*
// @match               http://*.google.gl/search*
// @match               http://*.google.gm/search*
// @match               http://*.google.gr/search*
// @match               http://*.google.com.gt/search*
// @match               http://*.google.gy/search*
// @match               http://*.google.com.hk/search*
// @match               http://*.google.hn/search*
// @match               http://*.google.hr/search*
// @match               http://*.google.ht/search*
// @match               http://*.google.hu/search*
// @match               http://*.google.co.id/search*
// @match               http://*.google.ie/search*
// @match               http://*.google.co.il/search*
// @match               http://*.google.im/search*
// @match               http://*.google.co.in/search*
// @match               http://*.google.iq/search*
// @match               http://*.google.is/search*
// @match               http://*.google.it/search*
// @match               http://*.google.je/search*
// @match               http://*.google.com.jm/search*
// @match               http://*.google.jo/search*
// @match               http://*.google.co.jp/search*
// @match               http://*.google.co.ke/search*
// @match               http://*.google.com.kh/search*
// @match               http://*.google.ki/search*
// @match               http://*.google.kg/search*
// @match               http://*.google.co.kr/search*
// @match               http://*.google.com.kw/search*
// @match               http://*.google.kz/search*
// @match               http://*.google.la/search*
// @match               http://*.google.com.lb/search*
// @match               http://*.google.li/search*
// @match               http://*.google.lk/search*
// @match               http://*.google.co.ls/search*
// @match               http://*.google.lt/search*
// @match               http://*.google.lu/search*
// @match               http://*.google.lv/search*
// @match               http://*.google.com.ly/search*
// @match               http://*.google.co.ma/search*
// @match               http://*.google.md/search*
// @match               http://*.google.me/search*
// @match               http://*.google.mg/search*
// @match               http://*.google.mk/search*
// @match               http://*.google.ml/search*
// @match               http://*.google.com.mm/search*
// @match               http://*.google.mn/search*
// @match               http://*.google.ms/search*
// @match               http://*.google.com.mt/search*
// @match               http://*.google.mu/search*
// @match               http://*.google.mv/search*
// @match               http://*.google.mw/search*
// @match               http://*.google.com.mx/search*
// @match               http://*.google.com.my/search*
// @match               http://*.google.co.mz/search*
// @match               http://*.google.com.na/search*
// @match               http://*.google.com.ng/search*
// @match               http://*.google.com.ni/search*
// @match               http://*.google.ne/search*
// @match               http://*.google.nl/search*
// @match               http://*.google.no/search*
// @match               http://*.google.com.np/search*
// @match               http://*.google.nr/search*
// @match               http://*.google.nu/search*
// @match               http://*.google.co.nz/search*
// @match               http://*.google.com.om/search*
// @match               http://*.google.com.pa/search*
// @match               http://*.google.com.pe/search*
// @match               http://*.google.com.pg/search*
// @match               http://*.google.com.ph/search*
// @match               http://*.google.com.pk/search*
// @match               http://*.google.pl/search*
// @match               http://*.google.pn/search*
// @match               http://*.google.com.pr/search*
// @match               http://*.google.ps/search*
// @match               http://*.google.pt/search*
// @match               http://*.google.com.py/search*
// @match               http://*.google.com.qa/search*
// @match               http://*.google.ro/search*
// @match               http://*.google.ru/search*
// @match               http://*.google.rw/search*
// @match               http://*.google.com.sa/search*
// @match               http://*.google.com.sb/search*
// @match               http://*.google.sc/search*
// @match               http://*.google.se/search*
// @match               http://*.google.com.sg/search*
// @match               http://*.google.sh/search*
// @match               http://*.google.si/search*
// @match               http://*.google.sk/search*
// @match               http://*.google.com.sl/search*
// @match               http://*.google.sn/search*
// @match               http://*.google.so/search*
// @match               http://*.google.sm/search*
// @match               http://*.google.sr/search*
// @match               http://*.google.st/search*
// @match               http://*.google.com.sv/search*
// @match               http://*.google.td/search*
// @match               http://*.google.tg/search*
// @match               http://*.google.co.th/search*
// @match               http://*.google.com.tj/search*
// @match               http://*.google.tl/search*
// @match               http://*.google.tm/search*
// @match               http://*.google.tn/search*
// @match               http://*.google.to/search*
// @match               http://*.google.com.tr/search*
// @match               http://*.google.tt/search*
// @match               http://*.google.com.tw/search*
// @match               http://*.google.co.tz/search*
// @match               http://*.google.com.ua/search*
// @match               http://*.google.co.ug/search*
// @match               http://*.google.co.uk/search*
// @match               http://*.google.com.uy/search*
// @match               http://*.google.co.uz/search*
// @match               http://*.google.com.vc/search*
// @match               http://*.google.co.ve/search*
// @match               http://*.google.vg/search*
// @match               http://*.google.co.vi/search*
// @match               http://*.google.com.vn/search*
// @match               http://*.google.vu/search*
// @match               http://*.google.ws/search*
// @match               http://*.google.rs/search*
// @match               http://*.google.co.za/search*
// @match               http://*.google.co.zm/search*
// @match               http://*.google.co.zw/search*
// @match               http://*.google.cat/search*
// @match               https://*.google.com/search*
// @match               https://*.google.ad/search*
// @match               https://*.google.ae/search*
// @match               https://*.google.com.af/search*
// @match               https://*.google.com.ag/search*
// @match               https://*.google.com.ai/search*
// @match               https://*.google.al/search*
// @match               https://*.google.am/search*
// @match               https://*.google.co.ao/search*
// @match               https://*.google.com.ar/search*
// @match               https://*.google.as/search*
// @match               https://*.google.at/search*
// @match               https://*.google.com.au/search*
// @match               https://*.google.az/search*
// @match               https://*.google.ba/search*
// @match               https://*.google.com.bd/search*
// @match               https://*.google.be/search*
// @match               https://*.google.bf/search*
// @match               https://*.google.bg/search*
// @match               https://*.google.com.bh/search*
// @match               https://*.google.bi/search*
// @match               https://*.google.bj/search*
// @match               https://*.google.com.bn/search*
// @match               https://*.google.com.bo/search*
// @match               https://*.google.com.br/search*
// @match               https://*.google.bs/search*
// @match               https://*.google.bt/search*
// @match               https://*.google.co.bw/search*
// @match               https://*.google.by/search*
// @match               https://*.google.com.bz/search*
// @match               https://*.google.ca/search*
// @match               https://*.google.cd/search*
// @match               https://*.google.cf/search*
// @match               https://*.google.cg/search*
// @match               https://*.google.ch/search*
// @match               https://*.google.ci/search*
// @match               https://*.google.co.ck/search*
// @match               https://*.google.cl/search*
// @match               https://*.google.cm/search*
// @match               https://*.google.cn/search*
// @match               https://*.google.com.co/search*
// @match               https://*.google.co.cr/search*
// @match               https://*.google.com.cu/search*
// @match               https://*.google.cv/search*
// @match               https://*.google.com.cy/search*
// @match               https://*.google.cz/search*
// @match               https://*.google.de/search*
// @match               https://*.google.dj/search*
// @match               https://*.google.dk/search*
// @match               https://*.google.dm/search*
// @match               https://*.google.com.do/search*
// @match               https://*.google.dz/search*
// @match               https://*.google.com.ec/search*
// @match               https://*.google.ee/search*
// @match               https://*.google.com.eg/search*
// @match               https://*.google.es/search*
// @match               https://*.google.com.et/search*
// @match               https://*.google.fi/search*
// @match               https://*.google.com.fj/search*
// @match               https://*.google.fm/search*
// @match               https://*.google.fr/search*
// @match               https://*.google.ga/search*
// @match               https://*.google.ge/search*
// @match               https://*.google.gg/search*
// @match               https://*.google.com.gh/search*
// @match               https://*.google.com.gi/search*
// @match               https://*.google.gl/search*
// @match               https://*.google.gm/search*
// @match               https://*.google.gr/search*
// @match               https://*.google.com.gt/search*
// @match               https://*.google.gy/search*
// @match               https://*.google.com.hk/search*
// @match               https://*.google.hn/search*
// @match               https://*.google.hr/search*
// @match               https://*.google.ht/search*
// @match               https://*.google.hu/search*
// @match               https://*.google.co.id/search*
// @match               https://*.google.ie/search*
// @match               https://*.google.co.il/search*
// @match               https://*.google.im/search*
// @match               https://*.google.co.in/search*
// @match               https://*.google.iq/search*
// @match               https://*.google.is/search*
// @match               https://*.google.it/search*
// @match               https://*.google.je/search*
// @match               https://*.google.com.jm/search*
// @match               https://*.google.jo/search*
// @match               https://*.google.co.jp/search*
// @match               https://*.google.co.ke/search*
// @match               https://*.google.com.kh/search*
// @match               https://*.google.ki/search*
// @match               https://*.google.kg/search*
// @match               https://*.google.co.kr/search*
// @match               https://*.google.com.kw/search*
// @match               https://*.google.kz/search*
// @match               https://*.google.la/search*
// @match               https://*.google.com.lb/search*
// @match               https://*.google.li/search*
// @match               https://*.google.lk/search*
// @match               https://*.google.co.ls/search*
// @match               https://*.google.lt/search*
// @match               https://*.google.lu/search*
// @match               https://*.google.lv/search*
// @match               https://*.google.com.ly/search*
// @match               https://*.google.co.ma/search*
// @match               https://*.google.md/search*
// @match               https://*.google.me/search*
// @match               https://*.google.mg/search*
// @match               https://*.google.mk/search*
// @match               https://*.google.ml/search*
// @match               https://*.google.com.mm/search*
// @match               https://*.google.mn/search*
// @match               https://*.google.ms/search*
// @match               https://*.google.com.mt/search*
// @match               https://*.google.mu/search*
// @match               https://*.google.mv/search*
// @match               https://*.google.mw/search*
// @match               https://*.google.com.mx/search*
// @match               https://*.google.com.my/search*
// @match               https://*.google.co.mz/search*
// @match               https://*.google.com.na/search*
// @match               https://*.google.com.ng/search*
// @match               https://*.google.com.ni/search*
// @match               https://*.google.ne/search*
// @match               https://*.google.nl/search*
// @match               https://*.google.no/search*
// @match               https://*.google.com.np/search*
// @match               https://*.google.nr/search*
// @match               https://*.google.nu/search*
// @match               https://*.google.co.nz/search*
// @match               https://*.google.com.om/search*
// @match               https://*.google.com.pa/search*
// @match               https://*.google.com.pe/search*
// @match               https://*.google.com.pg/search*
// @match               https://*.google.com.ph/search*
// @match               https://*.google.com.pk/search*
// @match               https://*.google.pl/search*
// @match               https://*.google.pn/search*
// @match               https://*.google.com.pr/search*
// @match               https://*.google.ps/search*
// @match               https://*.google.pt/search*
// @match               https://*.google.com.py/search*
// @match               https://*.google.com.qa/search*
// @match               https://*.google.ro/search*
// @match               https://*.google.ru/search*
// @match               https://*.google.rw/search*
// @match               https://*.google.com.sa/search*
// @match               https://*.google.com.sb/search*
// @match               https://*.google.sc/search*
// @match               https://*.google.se/search*
// @match               https://*.google.com.sg/search*
// @match               https://*.google.sh/search*
// @match               https://*.google.si/search*
// @match               https://*.google.sk/search*
// @match               https://*.google.com.sl/search*
// @match               https://*.google.sn/search*
// @match               https://*.google.so/search*
// @match               https://*.google.sm/search*
// @match               https://*.google.sr/search*
// @match               https://*.google.st/search*
// @match               https://*.google.com.sv/search*
// @match               https://*.google.td/search*
// @match               https://*.google.tg/search*
// @match               https://*.google.co.th/search*
// @match               https://*.google.com.tj/search*
// @match               https://*.google.tl/search*
// @match               https://*.google.tm/search*
// @match               https://*.google.tn/search*
// @match               https://*.google.to/search*
// @match               https://*.google.com.tr/search*
// @match               https://*.google.tt/search*
// @match               https://*.google.com.tw/search*
// @match               https://*.google.co.tz/search*
// @match               https://*.google.com.ua/search*
// @match               https://*.google.co.ug/search*
// @match               https://*.google.co.uk/search*
// @match               https://*.google.com.uy/search*
// @match               https://*.google.co.uz/search*
// @match               https://*.google.com.vc/search*
// @match               https://*.google.co.ve/search*
// @match               https://*.google.vg/search*
// @match               https://*.google.co.vi/search*
// @match               https://*.google.com.vn/search*
// @match               https://*.google.vu/search*
// @match               https://*.google.ws/search*
// @match               https://*.google.rs/search*
// @match               https://*.google.co.za/search*
// @match               https://*.google.co.zm/search*
// @match               https://*.google.co.zw/search*
// @match               https://*.google.cat/search*
// @grant               none
// @version             0.3.4
// @author              veringsek
// @description         Navigate through Google Search with a set of hotkeys. 
// @description:zh-TW   在 Google 搜尋結果頁面利用快捷鍵進行快速瀏覽。
// @description:zh-CN   在 Google 搜尋結果頁面利用快捷鍵進行快速瀏覽。
// ==/UserScript==

(function () {
    let css = document.createElement('style');
    css.type = 'text/css';
    css.innerText = `
        :root {
            --google-navigation--button-size: 30px;
            --google-navigation--stroke-color: black;
            --google-navigation--background-color: white;
            --google-navigation--button-opacity: 1;
        }
        
        .google-navigation--button {
            position: absolute;
            left: calc(-5 / 3 * var(--google-navigation--button-size));
            border: 2px var(--google-navigation--stroke-color) solid;
            border-radius: 5px;
            background: var(--google-navigation--background-color);
            box-shadow: 0px 2px var(--google-navigation--stroke-color);
            color: var(--google-navigation--stroke-color) !important;
            opacity: var(--google-navigation--button-opacity);
            text-decoration: none !important;
            width: var(--google-navigation--button-size);
            height: var(--google-navigation--button-size);
            line-height: var(--google-navigation--button-size);
            text-align: center;
            font-size: calc(2 / 3 * var(--google-navigation--button-size));
            user-select: none;
            transition: all 0.1s linear;
        
            animation: spawn 0.2s;
        }
        
        .google-navigation--button.keydown {
            background: var(--google-navigation--stroke-color);
            color: var(--google-navigation--background-color) !important;
            transform: scale(0.8);
        }
        
        .google-navigation--switch {
            opacity: 0;
        }
        
        .google-navigation--switch.keydown {
            opacity: 1;
        }
        
        @keyframes spawn {
            from {
                transform: scale(0);
            }
        }
    `;
    document.head.appendChild(css);
})();

let GoogleNavigation = {};
GoogleNavigation.GN_KEYDOWN_TIMER_DURATION = 1000;
GoogleNavigation.GN_KEYDOWN_TIMER_CANCELED = 'GN_KEYDOWN_TIMER_CANCELED';
GoogleNavigation.GN_KEYPRESS_SENSITIVITY = 500;
GoogleNavigation.GN_KEYPRESS_CANCELED = 'GN_KEYPRESS_CANCELED';
GoogleNavigation.GN_BUTTON_SIZE = 30;
GoogleNavigation.GN_BUTTON_SIZE_HALF = GoogleNavigation.GN_BUTTON_SIZE / 2;
GoogleNavigation.keydowns = new Set();
GoogleNavigation.tmrKeydown = undefined;
globalThis.GoogleNavigation = GoogleNavigation;

function setCSSVars() {
    document.documentElement.style.setProperty(
        '--google-navigation--button-size',
        `${globalThis.GoogleNavigation.GN_BUTTON_SIZE}px`
    );
    document.documentElement.style.setProperty(
        '--google-navigation--stroke-color',
        globalThis.getComputedStyle(document.body)['color']
    );
    document.documentElement.style.setProperty(
        '--google-navigation--background-color',
        globalThis.getComputedStyle(document.body)['backgroundColor']
    );
}

function makeTemplate() {
    let template = document.createElement('a');
    template.classList.add('google-navigation--button');
    template.commandKeydown = function () {
        this.classList.add('keydown');
        let documentElement = document.documentElement;
        documentElement.scroll({
            top: this.getBoundingClientRect().top + documentElement.scrollTop - window.innerHeight / 2,
            behavior: 'smooth'
        });
        globalThis.GoogleNavigation.tmrKeydown = globalThis.setTimeout(() => {
            globalThis.clearTimeout(globalThis.GoogleNavigation.tmrKeydown);
            globalThis.GoogleNavigation.tmrKeydown = globalThis.GoogleNavigation.GN_KEYDOWN_TIMER_CANCELED;
            this.classList.remove('keydown');
            this.commandCancel();
        }, globalThis.GoogleNavigation.GN_KEYDOWN_TIMER_DURATION);
    };
    template.commandKeyup = function () {
        this.classList.remove('keydown');
        this.commandPress();
    };
    template.commandPress = function () {
        if (document.getElementById('google-navigation--switch-Control').keydown) {
            window.open(this.link, '_blank');
        } else {
            window.open(this.link, '_self');
        }
    };
    template.commandCancel = function () {
        return;
    };
    globalThis.GoogleNavigation.buttonTemplate = template;
}

function getClonedButton(key, link) {
    let string = key;
    if (Array.isArray(key)) {
        [key, string] = key;
    }
    let cloned = globalThis.GoogleNavigation.buttonTemplate.cloneNode();
    cloned.commandKeydown = globalThis.GoogleNavigation.buttonTemplate.commandKeydown;
    cloned.commandKeyup = globalThis.GoogleNavigation.buttonTemplate.commandKeyup;
    cloned.commandPress = globalThis.GoogleNavigation.buttonTemplate.commandPress;
    cloned.commandCancel = globalThis.GoogleNavigation.buttonTemplate.commandCancel;
    cloned.innerHTML = string;
    cloned.classList.add(`google-navigation--button-${key}`);
    if (link) {
        cloned.link = link;
        cloned.setAttribute('link', link);
    }
    return cloned;
}

function plantButtons() {
    let search = document.getElementById('search');
    let rcnt = document.getElementById('rcnt');
    let links = [...document.getElementsByClassName('LC20lb')];
    let num = 0;
    for (let link of links) {
        if (getCollapseSectionParent(link)) continue;
        if (num >= 10) break;
        let href = link.parentElement.href;
        let cloned;
        cloned = getClonedButton(num, href);
        cloned.style.top = `${link.offsetTop + link.clientHeight / 2 - GoogleNavigation.GN_BUTTON_SIZE_HALF}px`;
        link.insertBefore(cloned, link.children[0]);

        let node = link;
        let notInRcnt = false;
        do {
            if (!(node instanceof Element)) {
                notInRcnt = true;
                break;
            }
            let style = window.getComputedStyle(node);
            if (style['overflow'] === 'hidden') {
                node.style.overflow = 'visible';
            }
            if (style['contain'] === 'layout paint') {
                node.style.contain = 'unset';
            }
            node = node.parentElement;
        } while (node !== rcnt);
        if (notInRcnt) continue;
        
        setButtonEnter(document.activeElement === document.body);
        num += 1;
    }

    if (search) {
        let cloned = getClonedButton(['Control', 'CTRL']);
        cloned.id = 'google-navigation--switch-Control';
        cloned.classList.add('google-navigation--switch');
        cloned.style.left = '-150px';
        cloned.style.width = '80px';
        cloned.style.fontWeight = 'bold';
        cloned.keydown = false;
        cloned.commandKeydown = function () {
            this.tmrKeypress = globalThis.setTimeout(() => {
                this.tmrKeypress = globalThis.GoogleNavigation.GN_KEYPRESS_CANCELED;
            }, globalThis.GoogleNavigation.GN_KEYPRESS_SENSITIVITY);
        };
        cloned.commandKeyup = function () {
            if (this.tmrKeypress === globalThis.GoogleNavigation.GN_KEYPRESS_CANCELED) {
                this.tmrKeypress = undefined;
            } else if (this.tmrKeypress) {
                this.keydown = !this.keydown;
                if (this.keydown) {
                    this.classList.add('keydown');
                } else {
                    this.classList.remove('keydown');
                }
            }
        };
        search.insertBefore(cloned, search.children[0]);
    }

    let btnWiki = document.getElementsByClassName('ruhjFe')[0];
    if (btnWiki) {
        let cloned = getClonedButton(['p', 'P'], btnWiki.href);
        let card = getWidgetParent(btnWiki);
        if (!card) {
            card = document.getElementById('rhs');
            cloned.style.left = `calc(2 / 3 * var(--google-navigation--button-size) + 100%)`;
        }
        card.style.position = 'relative';
        card.insertBefore(cloned, card.children[0]);
    }

    let translatorWidgetMain = document.getElementById('tw-main');
    let translatorWidgetSourceTextTA = document.getElementById('tw-source-text-ta');
    if (translatorWidgetMain && translatorWidgetSourceTextTA) {
        let cloned = getClonedButton(['t', 'T']);
        let rectMain = translatorWidgetMain.getBoundingClientRect();
        let rectText = translatorWidgetSourceTextTA.getBoundingClientRect();
        let top = (rectText.bottom + rectText.top) / 2 - rectMain.top - GoogleNavigation.GN_BUTTON_SIZE_HALF;
        cloned.style.top = `${top}px`;
        cloned.commandPress = function () {
            document.getElementById('tw-source-text-ta').focus();
        };
        translatorWidgetMain.style.position = 'relative';
        translatorWidgetMain.insertBefore(cloned, translatorWidgetMain.children[0]);
    }

    let calculatorWidget = document.getElementById('cwos')?.parentElement?.parentElement;
    if (calculatorWidget) {
        let calculatorWidgetParent = getWidgetParent(calculatorWidget);
        let cloned = getClonedButton(['t', 'T']);
        let top = calculatorWidgetParent.clientHeight / 2 - globalThis.GoogleNavigation.GN_BUTTON_SIZE_HALF;
        cloned.style.top = `${top}px`;
        cloned.commandPress = function () {
            document.getElementById('cwos').parentElement.parentElement.focus();
        };
        calculatorWidgetParent.insertBefore(cloned, calculatorWidgetParent.children[0]);
    }

    let pnprev = document.getElementById('pnprev');
    if (pnprev) {
        let cloned = getClonedButton([',', ','], pnprev.href);
        cloned.style.top = '-3px';
        cloned.style.left = '-40px';
        pnprev.style.position = 'relative';
        pnprev.insertBefore(cloned, pnprev.children[0]);
    }
    let pnnext = document.getElementById('pnnext');
    if (pnnext) {
        let cloned = getClonedButton(['.', '‧'], pnnext.href);
        cloned.style.top = '-3px';
        cloned.style.left = 'calc(100% + 5px)';
        pnnext.style.position = 'relative';
        pnnext.insertBefore(cloned, pnnext.children[0]);
    }
}

function isCommanding() {
    let target = document.activeElement;
    return !(['INPUT', 'TEXTAREA'].includes(target.tagName) || target.classList.contains('jlkklc'));
}

function cssVar(name) {
    return globalThis.getComputedStyle(document.documentElement).getPropertyValue(name);
}

function getParent(element, condition) {
    let node = element;
    do {
        if (condition(node)) return node;
        node = node.parentElement;
    } while (node !== document.body);
    return null;
}

function getWidgetParent(element) {
    return getParent(element, node => {
        let computedStyle = globalThis.getComputedStyle(node);
        let borderWidth = computedStyle.borderWidth;
        return borderWidth !== '' && !(/^0\D*/.test(borderWidth));
    });
}

function getCollapseSectionParent(element) {
    return getParent(element, node => node.getAttribute('jscontroller') === 'Da4hkd');
}

function assignEventListeners() {
    let onFocusChanged = function (ev) {
        document.documentElement.style.setProperty(
            '--google-navigation--button-opacity',
            isCommanding() ? 1 : 0.5
        );
        setButtonEnter(document.activeElement === document.body);
    };
    window.addEventListener('focus', onFocusChanged, true);
    window.addEventListener('blur', onFocusChanged, true);

    document.body.addEventListener('keydown', function (ev) {
        if (globalThis.GoogleNavigation.keydowns.has(ev.key)) return;
        globalThis.GoogleNavigation.keydowns.add(ev.key);
        if (isCommanding()) {
            let button = document.getElementsByClassName(`google-navigation--button-${ev.key}`)[0];
            if (button) {
                button.commandKeydown();
            }
        }
    });

    document.body.addEventListener('keyup', function (ev) {
        globalThis.GoogleNavigation.keydowns.delete(ev.key);
        if (globalThis.GoogleNavigation.tmrKeydown === globalThis.GoogleNavigation.GN_KEYDOWN_TIMER_CANCELED) {
            globalThis.GoogleNavigation.tmrKeydown = undefined;
            return;
        }
        globalThis.clearTimeout(globalThis.GoogleNavigation.tmrKeydown);
        globalThis.GoogleNavigation.tmrKeydown = undefined;
        if (isCommanding()) {
            let button = document.getElementsByClassName(`google-navigation--button-${ev.key}`)[0];
            if (button) {
                button.commandKeyup();
            }
        }
    });
}

function setButtonEnter(enabled) {
    let btn0 = document.getElementsByClassName('google-navigation--button-0')[0];
    if (enabled) {
        btn0.innerHTML = '⮨';
        btn0.classList.add('google-navigation--button-Enter');
    } else {
        btn0.innerHTML = '0';
        btn0.classList.remove('google-navigation--button-Enter');
    }
}

setCSSVars();
makeTemplate();
plantButtons();
assignEventListeners();