Skip to content

Instantly share code, notes, and snippets.

@Explosion-Scratch
Last active March 26, 2025 17:38
Show Gist options
  • Save Explosion-Scratch/bd1d9fc6840fa9997c4a02648aa3e64a to your computer and use it in GitHub Desktop.
Save Explosion-Scratch/bd1d9fc6840fa9997c4a02648aa3e64a to your computer and use it in GitHub Desktop.
Web Search Navigator Userscript
// ==UserScript==
// @name Web Search Navigator
// @version 0.5.2
// @description Keyboard shortcuts for Google search, YouTube, Startpage, Brave Search, Google Scholar, Github, Gitlab, and Amazon.
// @author Web Search Navigator
// @iconURL data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAABAAAAAQEAYAAABPYyMiAAAABGdBTUEAALGPC/xhBQAAACBjSFJNAAB6JgAAgIQAAPoAAACA6AAAdTAAAOpgAAA6mAAAF3CculE8AAAABmJLR0T///////8JWPfcAAAAB3RJTUUH4goDFzU3zLh1KgAABfpJREFUSMe1lWtQVecZhZ/32/scDgdB4HiFgKihgjbiddC2RKI1GkOisbYkRpPaasOUJJ2AbVqNQ7zGCxoTpY4TMwaimEQNNYmoVWtHra2tiUxVIt7QAt4KgnDg3PbeX3/UzFhn+jPPv3dmrZn1412zhAc49PThxMPnwF/mX9W5CDLJzMycAa4JrslmHmBgYAAa/T9GQQCIJVbiwC9+f8cNqNxYOaryPPSY7ivosRDmN8yfUXzsftsDaK211rjvneHzc+pyL6S5lnk2eyo8n5hBGmmkka/RD0SQ/0aQbLIZQ8adta0LWgusycNfGLYq67NIzj2VKSIigvWNzXwwQPuFjnP+GsJ2ib3Qfj19U8qvU0pTPnkrSnbJUanr+QtChAhpxf9jF2HqeC9xtq/Cd+LOntrA1/r8j97w+Tb7dvmO1LY8KDdHuZb0bG0C3aLPsBuMrUbIXALaq706Or3GWG28bZQ9U8IHVMiHKoIfv+4EIkSIABYOgEyVLBSoIlkiw0B9Kn9jCTlGvmpU+7b39Q72XvROqm0Z+fvFz98pAhkn70k/UJ5cb1vseBh6MK0mZhrUjGz6WH4Oxg4p4qSTKRMlWo7Z09RClU0DKFG71TNgrDUGq7+A0WzMUcfB8joT1Xo4MeBKMgKnZzes1nH29M7doUa72PnhX+fV/1RiYWhT2ve69QTPeG8wdiqYkfGWsgeWbrh85vZYfT4qo3jF7hjJc9qnZQ+db1zqH/tq9/GGzjKelRFMcSJgtpgVZhaYe43exhsg63meFvD3Dd3SJfDFxDNJdibYUZZhj1Gjne3hlQmD57lPjrtRZubl7nCfNb7rXFS+SK4d0b1D7TJ50zsfB7tr/eKvRq8JzIZrG1s9xvuQubb3IKcNJszIrHe1A8d1NYdx6CMPkY6iNz3wAeNIR8BqszfyL6hva5muM9D4nSN2EKlrun44sA6uprev834O/RYlBO1JUP72P34b/Qcw9Sy2E0+5VencYBYvyl226cUQHGwVql6ghqovzcmgKlSq7EXpL3SjvnvfF1k4aDBzzIdJgSFDkmfKKMR5yVlvL4bLhbeqg1+B2ij5eg9YW5wSOkHPopI/809l9leH9V7sxOXeKXoiZO1JMqyzcHbm9X2SAV2XQqkRQJ6gSl8GeVMK5CnAg4EDxN6r7Bzmyqugd7Bb/xGuj7rzSPhl6LbB/Utb4NH4Ad0jFZBwKrpB/xjMNHVIV2ObkWX2EnkZmk7ePaBKIfKsfULyobM4fJR3YNftU/GhQ5Ce1uuVsA9GlqStiqoGz2NR/cwI4OcWFyBgBbqCxXCqo35/MBv2J9UmymJIvpyYEj0JkssjzcyBttOBJimFyDJ7qRSC6S4zkvQCGOjzTba3wqWC5jHGu/D4dwZFhYfD9eXtHmMpdL0Qmms9BKeX1//dPxdi5nluKgOcsfZOOxE6hgQuWp+BP8/qcH8I5uaoKZ6BkBuX/v1gC8gUdvIUNH/QWaXOgXuT0VcvADMkVop8BOdib8aYLrh5qmOnaobO6+EOaYC2wkCs6gH/XufvUGlglKlSvRzo1Id0CrCIz81yoEnKo2eBle8Mk/UQeC28QsbCyaRrta4MCL1lbZAQSJF8RQuElPW6fARmVKk5Qufxs6Fm0mprJnjTmvsZClLjE35nm3DzaPtKtRXi90e/q9+EAat8T9qPghPWB5kBPEwM1aCUTOA1uHKsJWyMgqRtcbkyFfqkxvZxjsO1La0pRiukp/bE3gwHV9cNd+9luOTcXFMV7LqaP6ip16f2Slecvyv0tKRywmu51+oECNZGfiPHwFVgiC6Bbk+6H9d3QT/HNuLvm4JKZtEGnX8KH5E4CK+0u2QpeIa5SvUPoCsQfkXaoFvfqAP6KmPqkm/nG4sjKTLi0vKc0GP9CttvB0fKHLPK9ZIqokyft/KcuzwHhkdVsQ8cRzcwDKwaZxfzgBwGEL6vjse4ghvMR9R0toCKkv6cBjvgTOMJMA+oBHZAZIOzhkLJiOvl+VK/b/1ERusVdjAMYopXEr+ZRL5dBLSlu3QL/AeVQYYk5qYwaQAAACV0RVh0ZGF0ZTpjcmVhdGUAMjAxOC0xMC0wM1QyMDo1Mzo1NSswMzowMPz2q7oAAAAldEVYdGRhdGU6bW9kaWZ5ADIwMTgtMTAtMDNUMjA6NTM6NTUrMDM6MDCNqxMGAAAAAElFTkSuQmCC
// @match *://www.google.com/search*
// @match *://www.google.ad/search*
// @match *://www.google.ae/search*
// @match *://www.google.com.af/search*
// @match *://www.google.com.ag/search*
// @match *://www.google.com.ai/search*
// @match *://www.google.al/search*
// @match *://www.google.am/search*
// @match *://www.google.co.ao/search*
// @match *://www.google.com.ar/search*
// @match *://www.google.as/search*
// @match *://www.google.at/search*
// @match *://www.google.com.au/search*
// @match *://www.google.az/search*
// @match *://www.google.ba/search*
// @match *://www.google.com.bd/search*
// @match *://www.google.be/search*
// @match *://www.google.bf/search*
// @match *://www.google.bg/search*
// @match *://www.google.com.bh/search*
// @match *://www.google.bi/search*
// @match *://www.google.bj/search*
// @match *://www.google.com.bn/search*
// @match *://www.google.com.bo/search*
// @match *://www.google.com.br/search*
// @match *://www.google.bs/search*
// @match *://www.google.bt/search*
// @match *://www.google.co.bw/search*
// @match *://www.google.by/search*
// @match *://www.google.com.bz/search*
// @match *://www.google.ca/search*
// @match *://www.google.cd/search*
// @match *://www.google.cf/search*
// @match *://www.google.cg/search*
// @match *://www.google.ch/search*
// @match *://www.google.ci/search*
// @match *://www.google.co.ck/search*
// @match *://www.google.cl/search*
// @match *://www.google.cm/search*
// @match *://www.google.cn/search*
// @match *://www.google.com.co/search*
// @match *://www.google.co.cr/search*
// @match *://www.google.com.cu/search*
// @match *://www.google.cv/search*
// @match *://www.google.com.cy/search*
// @match *://www.google.cz/search*
// @match *://www.google.de/search*
// @match *://www.google.dj/search*
// @match *://www.google.dk/search*
// @match *://www.google.dm/search*
// @match *://www.google.com.do/search*
// @match *://www.google.dz/search*
// @match *://www.google.com.ec/search*
// @match *://www.google.ee/search*
// @match *://www.google.com.eg/search*
// @match *://www.google.es/search*
// @match *://www.google.com.et/search*
// @match *://www.google.fi/search*
// @match *://www.google.com.fj/search*
// @match *://www.google.fm/search*
// @match *://www.google.fr/search*
// @match *://www.google.ga/search*
// @match *://www.google.ge/search*
// @match *://www.google.gg/search*
// @match *://www.google.com.gh/search*
// @match *://www.google.com.gi/search*
// @match *://www.google.gl/search*
// @match *://www.google.gm/search*
// @match *://www.google.gp/search*
// @match *://www.google.gr/search*
// @match *://www.google.com.gt/search*
// @match *://www.google.gy/search*
// @match *://www.google.com.hk/search*
// @match *://www.google.hn/search*
// @match *://www.google.hr/search*
// @match *://www.google.ht/search*
// @match *://www.google.hu/search*
// @match *://www.google.co.id/search*
// @match *://www.google.ie/search*
// @match *://www.google.co.il/search*
// @match *://www.google.im/search*
// @match *://www.google.co.in/search*
// @match *://www.google.iq/search*
// @match *://www.google.is/search*
// @match *://www.google.it/search*
// @match *://www.google.je/search*
// @match *://www.google.com.jm/search*
// @match *://www.google.jo/search*
// @match *://www.google.co.jp/search*
// @match *://www.google.co.ke/search*
// @match *://www.google.com.kh/search*
// @match *://www.google.ki/search*
// @match *://www.google.kg/search*
// @match *://www.google.co.kr/search*
// @match *://www.google.com.kw/search*
// @match *://www.google.kz/search*
// @match *://www.google.la/search*
// @match *://www.google.com.lb/search*
// @match *://www.google.li/search*
// @match *://www.google.lk/search*
// @match *://www.google.co.ls/search*
// @match *://www.google.lt/search*
// @match *://www.google.lu/search*
// @match *://www.google.lv/search*
// @match *://www.google.com.ly/search*
// @match *://www.google.co.ma/search*
// @match *://www.google.md/search*
// @match *://www.google.me/search*
// @match *://www.google.mg/search*
// @match *://www.google.mk/search*
// @match *://www.google.ml/search*
// @match *://www.google.com.mm/search*
// @match *://www.google.mn/search*
// @match *://www.google.ms/search*
// @match *://www.google.com.mt/search*
// @match *://www.google.mu/search*
// @match *://www.google.mv/search*
// @match *://www.google.mw/search*
// @match *://www.google.com.mx/search*
// @match *://www.google.com.my/search*
// @match *://www.google.co.mz/search*
// @match *://www.google.com.na/search*
// @match *://www.google.com.nf/search*
// @match *://www.google.com.ng/search*
// @match *://www.google.com.ni/search*
// @match *://www.google.ne/search*
// @match *://www.google.nl/search*
// @match *://www.google.no/search*
// @match *://www.google.com.np/search*
// @match *://www.google.nr/search*
// @match *://www.google.nu/search*
// @match *://www.google.co.nz/search*
// @match *://www.google.com.om/search*
// @match *://www.google.com.pa/search*
// @match *://www.google.com.pe/search*
// @match *://www.google.com.pg/search*
// @match *://www.google.com.ph/search*
// @match *://www.google.com.pk/search*
// @match *://www.google.pl/search*
// @match *://www.google.pn/search*
// @match *://www.google.com.pr/search*
// @match *://www.google.ps/search*
// @match *://www.google.pt/search*
// @match *://www.google.com.py/search*
// @match *://www.google.com.qa/search*
// @match *://www.google.ro/search*
// @match *://www.google.ru/search*
// @match *://www.google.rw/search*
// @match *://www.google.com.sa/search*
// @match *://www.google.com.sb/search*
// @match *://www.google.sc/search*
// @match *://www.google.se/search*
// @match *://www.google.com.sg/search*
// @match *://www.google.sh/search*
// @match *://www.google.si/search*
// @match *://www.google.sk/search*
// @match *://www.google.com.sl/search*
// @match *://www.google.sn/search*
// @match *://www.google.so/search*
// @match *://www.google.sm/search*
// @match *://www.google.sr/search*
// @match *://www.google.st/search*
// @match *://www.google.com.sv/search*
// @match *://www.google.td/search*
// @match *://www.google.tg/search*
// @match *://www.google.co.th/search*
// @match *://www.google.com.tj/search*
// @match *://www.google.tk/search*
// @match *://www.google.tl/search*
// @match *://www.google.tm/search*
// @match *://www.google.tn/search*
// @match *://www.google.to/search*
// @match *://www.google.com.tr/search*
// @match *://www.google.tt/search*
// @match *://www.google.com.tw/search*
// @match *://www.google.co.tz/search*
// @match *://www.google.com.ua/search*
// @match *://www.google.co.ug/search*
// @match *://www.google.co.uk/search*
// @match *://www.google.com.uy/search*
// @match *://www.google.co.uz/search*
// @match *://www.google.com.vc/search*
// @match *://www.google.co.ve/search*
// @match *://www.google.vg/search*
// @match *://www.google.co.vi/search*
// @match *://www.google.com.vn/search*
// @match *://www.google.vu/search*
// @match *://www.google.ws/search*
// @match *://www.google.rs/search*
// @match *://www.google.co.za/search*
// @match *://www.google.co.zm/search*
// @match *://www.google.co.zw/search*
// @match *://www.google.cat/search*
// @match https://*/*
// @match https://search.brave.com/*
// @match https://startpage.com/*
// @match https://www.startpage.com/*
// @match https://www.youtube.com/*
// @match https://github.com/*
// @match https://gitlab.com/*
// @match https://www.github.com/*
// @match https://www.amazon.com/*
// @match https://www.amazon.cn/*
// @match https://www.amazon.in/*
// @match https://www.amazon.co.jp/*
// @match https://www.amazon.co.uk/*
// @match https://www.amazon.ca/*
// @match https://www.amazon.fr/*
// @match https://www.amazon.de/*
// @match https://www.amazon.it/*
// @match https://www.amazon.es/*
// @match https://www.amazon.com.au/*
// @match https://www.amazon.com.mx/*
// @match https://www.amazon.com.br/*
// @match https://www.amazon.nl/*
// @match https://scholar.google.ad/*
// @match https://scholar.google.ae/*
// @match https://scholar.google.al/*
// @match https://scholar.google.am/*
// @match https://scholar.google.as/*
// @match https://scholar.google.at/*
// @match https://scholar.google.az/*
// @match https://scholar.google.ba/*
// @match https://scholar.google.be/*
// @match https://scholar.google.bf/*
// @match https://scholar.google.bg/*
// @match https://scholar.google.bi/*
// @match https://scholar.google.bj/*
// @match https://scholar.google.bs/*
// @match https://scholar.google.bt/*
// @match https://scholar.google.by/*
// @match https://scholar.google.ca/*
// @match https://scholar.google.cat/*
// @match https://scholar.google.cd/*
// @match https://scholar.google.cf/*
// @match https://scholar.google.cg/*
// @match https://scholar.google.ch/*
// @match https://scholar.google.ci/*
// @match https://scholar.google.cl/*
// @match https://scholar.google.cm/*
// @match https://scholar.google.cn/*
// @match https://scholar.google.co.ao/*
// @match https://scholar.google.co.bw/*
// @match https://scholar.google.co.ck/*
// @match https://scholar.google.co.cr/*
// @match https://scholar.google.co.id/*
// @match https://scholar.google.co.il/*
// @match https://scholar.google.co.in/*
// @match https://scholar.google.co.jp/*
// @match https://scholar.google.co.ke/*
// @match https://scholar.google.co.kr/*
// @match https://scholar.google.co.ls/*
// @match https://scholar.google.co.ma/*
// @match https://scholar.google.co.mz/*
// @match https://scholar.google.co.nz/*
// @match https://scholar.google.co.th/*
// @match https://scholar.google.co.tz/*
// @match https://scholar.google.co.ug/*
// @match https://scholar.google.co.uk/*
// @match https://scholar.google.co.uz/*
// @match https://scholar.google.co.ve/*
// @match https://scholar.google.co.vi/*
// @match https://scholar.google.co.za/*
// @match https://scholar.google.co.zm/*
// @match https://scholar.google.co.zw/*
// @match https://scholar.google.com.af/*
// @match https://scholar.google.com.ag/*
// @match https://scholar.google.com.ai/*
// @match https://scholar.google.com.ar/*
// @match https://scholar.google.com.au/*
// @match https://scholar.google.com.bd/*
// @match https://scholar.google.com.bh/*
// @match https://scholar.google.com.bn/*
// @match https://scholar.google.com.bo/*
// @match https://scholar.google.com.br/*
// @match https://scholar.google.com.bz/*
// @match https://scholar.google.com.co/*
// @match https://scholar.google.com.cu/*
// @match https://scholar.google.com.cy/*
// @match https://scholar.google.com.do/*
// @match https://scholar.google.com.ec/*
// @match https://scholar.google.com.eg/*
// @match https://scholar.google.com.et/*
// @match https://scholar.google.com.fj/*
// @match https://scholar.google.com.gh/*
// @match https://scholar.google.com.gi/*
// @match https://scholar.google.com.gt/*
// @match https://scholar.google.com.hk/*
// @match https://scholar.google.com.jm/*
// @match https://scholar.google.com.kh/*
// @match https://scholar.google.com.kw/*
// @match https://scholar.google.com.lb/*
// @match https://scholar.google.com.ly/*
// @match https://scholar.google.com.mm/*
// @match https://scholar.google.com.mt/*
// @match https://scholar.google.com.mx/*
// @match https://scholar.google.com.my/*
// @match https://scholar.google.com.na/*
// @match https://scholar.google.com.nf/*
// @match https://scholar.google.com.ng/*
// @match https://scholar.google.com.ni/*
// @match https://scholar.google.com.np/*
// @match https://scholar.google.com.om/*
// @match https://scholar.google.com.pa/*
// @match https://scholar.google.com.pe/*
// @match https://scholar.google.com.pg/*
// @match https://scholar.google.com.ph/*
// @match https://scholar.google.com.pk/*
// @match https://scholar.google.com.pr/*
// @match https://scholar.google.com.py/*
// @match https://scholar.google.com.qa/*
// @match https://scholar.google.com.sa/*
// @match https://scholar.google.com.sb/*
// @match https://scholar.google.com.sg/*
// @match https://scholar.google.com.sl/*
// @match https://scholar.google.com.sv/*
// @match https://scholar.google.com.tj/*
// @match https://scholar.google.com.tr/*
// @match https://scholar.google.com.tw/*
// @match https://scholar.google.com.ua/*
// @match https://scholar.google.com.uy/*
// @match https://scholar.google.com.vc/*
// @match https://scholar.google.com.vn/*
// @match https://scholar.google.com/*
// @match https://scholar.google.cv/*
// @match https://scholar.google.cz/*
// @match https://scholar.google.de/*
// @match https://scholar.google.dj/*
// @match https://scholar.google.dk/*
// @match https://scholar.google.dm/*
// @match https://scholar.google.dz/*
// @match https://scholar.google.ee/*
// @match https://scholar.google.es/*
// @match https://scholar.google.fi/*
// @match https://scholar.google.fm/*
// @match https://scholar.google.fr/*
// @match https://scholar.google.ga/*
// @match https://scholar.google.ge/*
// @match https://scholar.google.gg/*
// @match https://scholar.google.gl/*
// @match https://scholar.google.gm/*
// @match https://scholar.google.gp/*
// @match https://scholar.google.gr/*
// @match https://scholar.google.gy/*
// @match https://scholar.google.hn/*
// @match https://scholar.google.hr/*
// @match https://scholar.google.ht/*
// @match https://scholar.google.hu/*
// @match https://scholar.google.ie/*
// @match https://scholar.google.im/*
// @match https://scholar.google.iq/*
// @match https://scholar.google.is/*
// @match https://scholar.google.it/*
// @match https://scholar.google.je/*
// @match https://scholar.google.jo/*
// @match https://scholar.google.kg/*
// @match https://scholar.google.ki/*
// @match https://scholar.google.kz/*
// @match https://scholar.google.la/*
// @match https://scholar.google.li/*
// @match https://scholar.google.lk/*
// @match https://scholar.google.lt/*
// @match https://scholar.google.lu/*
// @match https://scholar.google.lv/*
// @match https://scholar.google.md/*
// @match https://scholar.google.me/*
// @match https://scholar.google.mg/*
// @match https://scholar.google.mk/*
// @match https://scholar.google.ml/*
// @match https://scholar.google.mn/*
// @match https://scholar.google.ms/*
// @match https://scholar.google.mu/*
// @match https://scholar.google.mv/*
// @match https://scholar.google.mw/*
// @match https://scholar.google.ne/*
// @match https://scholar.google.nl/*
// @match https://scholar.google.no/*
// @match https://scholar.google.nr/*
// @match https://scholar.google.nu/*
// @match https://scholar.google.pl/*
// @match https://scholar.google.pn/*
// @match https://scholar.google.ps/*
// @match https://scholar.google.pt/*
// @match https://scholar.google.ro/*
// @match https://scholar.google.rs/*
// @match https://scholar.google.ru/*
// @match https://scholar.google.rw/*
// @match https://scholar.google.sc/*
// @match https://scholar.google.se/*
// @match https://scholar.google.sh/*
// @match https://scholar.google.si/*
// @match https://scholar.google.sk/*
// @match https://scholar.google.sm/*
// @match https://scholar.google.sn/*
// @match https://scholar.google.so/*
// @match https://scholar.google.sr/*
// @match https://scholar.google.st/*
// @match https://scholar.google.td/*
// @match https://scholar.google.tg/*
// @match https://scholar.google.tk/*
// @match https://scholar.google.tl/*
// @match https://scholar.google.tm/*
// @match https://scholar.google.tn/*
// @match https://scholar.google.to/*
// @match https://scholar.google.tt/*
// @match https://scholar.google.vg/*
// @match https://scholar.google.vu/*
// @match https://scholar.google.ws/*
// ==/UserScript==
globalThis.IS_USERSCRIPT = true;
const USE_GM = false;
const PREFIX = 'userscript-polyfill';
globalThis._localStorage_browser_polyfill = {
get: async (...args) => {
args = args.filter(Boolean);
debugger;
console.log('[localStorage] Get: ', ...args);
const out = {};
for (const k of args) {
out[k] = USE_GM ? GM_getValue(k) : localStorage[`${PREFIX}_${k}`];
}
return out;
},
set: async (...args) => {
debugger;
console.log('[localStorage] Set: ', ...args);
},
clear: async () => {
debugger;
console.log('[localStorage] Clear');
},
};
globalThis._browser_userscript_polyfill = {
runtime: {
sendMessage: (msg) => {
if (msg.type === 'tabsCreate') {
window.open(msg.options.url, '_blank');
}
},
id: '093889f3-43be-45e3-bc5a-e257e75b466d',
},
storage: {sync: globalThis._localStorage_browser_polyfill, local: globalThis._localStorage_browser_polyfill},
permissions: {
remove: () => {},
add: () => {},
request: () => {},
getAll: () => ({}),
},
};
console.log(globalThis.browser, _browser_userscript_polyfill);
Object.assign(globalThis, {browser: globalThis._browser_userscript_polyfill, chrome: globalThis._browser_userscript_polyfill});
(function(a,b){if("function"==typeof define&&define.amd)define("webextension-polyfill",["module"],b);else if("undefined"!=typeof exports)b(module);else{var c={exports:{}};b(c),a.browser=c.exports}})("undefined"==typeof globalThis?"undefined"==typeof self?this:self:globalThis,function(a){"use strict";if(!(globalThis.chrome&&globalThis.chrome.runtime&&globalThis.chrome.runtime.id))throw new Error("This script should only be loaded in a browser extension.");if(!(globalThis.browser&&globalThis.browser.runtime&&globalThis.browser.runtime.id)){a.exports=(a=>{const b={alarms:{clear:{minArgs:0,maxArgs:1},clearAll:{minArgs:0,maxArgs:0},get:{minArgs:0,maxArgs:1},getAll:{minArgs:0,maxArgs:0}},bookmarks:{create:{minArgs:1,maxArgs:1},get:{minArgs:1,maxArgs:1},getChildren:{minArgs:1,maxArgs:1},getRecent:{minArgs:1,maxArgs:1},getSubTree:{minArgs:1,maxArgs:1},getTree:{minArgs:0,maxArgs:0},move:{minArgs:2,maxArgs:2},remove:{minArgs:1,maxArgs:1},removeTree:{minArgs:1,maxArgs:1},search:{minArgs:1,maxArgs:1},update:{minArgs:2,maxArgs:2}},browserAction:{disable:{minArgs:0,maxArgs:1,fallbackToNoCallback:!0},enable:{minArgs:0,maxArgs:1,fallbackToNoCallback:!0},getBadgeBackgroundColor:{minArgs:1,maxArgs:1},getBadgeText:{minArgs:1,maxArgs:1},getPopup:{minArgs:1,maxArgs:1},getTitle:{minArgs:1,maxArgs:1},openPopup:{minArgs:0,maxArgs:0},setBadgeBackgroundColor:{minArgs:1,maxArgs:1,fallbackToNoCallback:!0},setBadgeText:{minArgs:1,maxArgs:1,fallbackToNoCallback:!0},setIcon:{minArgs:1,maxArgs:1},setPopup:{minArgs:1,maxArgs:1,fallbackToNoCallback:!0},setTitle:{minArgs:1,maxArgs:1,fallbackToNoCallback:!0}},browsingData:{remove:{minArgs:2,maxArgs:2},removeCache:{minArgs:1,maxArgs:1},removeCookies:{minArgs:1,maxArgs:1},removeDownloads:{minArgs:1,maxArgs:1},removeFormData:{minArgs:1,maxArgs:1},removeHistory:{minArgs:1,maxArgs:1},removeLocalStorage:{minArgs:1,maxArgs:1},removePasswords:{minArgs:1,maxArgs:1},removePluginData:{minArgs:1,maxArgs:1},settings:{minArgs:0,maxArgs:0}},commands:{getAll:{minArgs:0,maxArgs:0}},contextMenus:{remove:{minArgs:1,maxArgs:1},removeAll:{minArgs:0,maxArgs:0},update:{minArgs:2,maxArgs:2}},cookies:{get:{minArgs:1,maxArgs:1},getAll:{minArgs:1,maxArgs:1},getAllCookieStores:{minArgs:0,maxArgs:0},remove:{minArgs:1,maxArgs:1},set:{minArgs:1,maxArgs:1}},devtools:{inspectedWindow:{eval:{minArgs:1,maxArgs:2,singleCallbackArg:!1}},panels:{create:{minArgs:3,maxArgs:3,singleCallbackArg:!0},elements:{createSidebarPane:{minArgs:1,maxArgs:1}}}},downloads:{cancel:{minArgs:1,maxArgs:1},download:{minArgs:1,maxArgs:1},erase:{minArgs:1,maxArgs:1},getFileIcon:{minArgs:1,maxArgs:2},open:{minArgs:1,maxArgs:1,fallbackToNoCallback:!0},pause:{minArgs:1,maxArgs:1},removeFile:{minArgs:1,maxArgs:1},resume:{minArgs:1,maxArgs:1},search:{minArgs:1,maxArgs:1},show:{minArgs:1,maxArgs:1,fallbackToNoCallback:!0}},extension:{isAllowedFileSchemeAccess:{minArgs:0,maxArgs:0},isAllowedIncognitoAccess:{minArgs:0,maxArgs:0}},history:{addUrl:{minArgs:1,maxArgs:1},deleteAll:{minArgs:0,maxArgs:0},deleteRange:{minArgs:1,maxArgs:1},deleteUrl:{minArgs:1,maxArgs:1},getVisits:{minArgs:1,maxArgs:1},search:{minArgs:1,maxArgs:1}},i18n:{detectLanguage:{minArgs:1,maxArgs:1},getAcceptLanguages:{minArgs:0,maxArgs:0}},identity:{launchWebAuthFlow:{minArgs:1,maxArgs:1}},idle:{queryState:{minArgs:1,maxArgs:1}},management:{get:{minArgs:1,maxArgs:1},getAll:{minArgs:0,maxArgs:0},getSelf:{minArgs:0,maxArgs:0},setEnabled:{minArgs:2,maxArgs:2},uninstallSelf:{minArgs:0,maxArgs:1}},notifications:{clear:{minArgs:1,maxArgs:1},create:{minArgs:1,maxArgs:2},getAll:{minArgs:0,maxArgs:0},getPermissionLevel:{minArgs:0,maxArgs:0},update:{minArgs:2,maxArgs:2}},pageAction:{getPopup:{minArgs:1,maxArgs:1},getTitle:{minArgs:1,maxArgs:1},hide:{minArgs:1,maxArgs:1,fallbackToNoCallback:!0},setIcon:{minArgs:1,maxArgs:1},setPopup:{minArgs:1,maxArgs:1,fallbackToNoCallback:!0},setTitle:{minArgs:1,maxArgs:1,fallbackToNoCallback:!0},show:{minArgs:1,maxArgs:1,fallbackToNoCallback:!0}},permissions:{contains:{minArgs:1,maxArgs:1},getAll:{minArgs:0,maxArgs:0},remove:{minArgs:1,maxArgs:1},request:{minArgs:1,maxArgs:1}},runtime:{getBackgroundPage:{minArgs:0,maxArgs:0},getPlatformInfo:{minArgs:0,maxArgs:0},openOptionsPage:{minArgs:0,maxArgs:0},requestUpdateCheck:{minArgs:0,maxArgs:0},sendMessage:{minArgs:1,maxArgs:3},sendNativeMessage:{minArgs:2,maxArgs:2},setUninstallURL:{minArgs:1,maxArgs:1}},sessions:{getDevices:{minArgs:0,maxArgs:1},getRecentlyClosed:{minArgs:0,maxArgs:1},restore:{minArgs:0,maxArgs:1}},storage:{local:{clear:{minArgs:0,maxArgs:0},get:{minArgs:0,maxArgs:1},getBytesInUse:{minArgs:0,maxArgs:1},remove:{minArgs:1,maxArgs:1},set:{minArgs:1,maxArgs:1}},managed:{get:{minArgs:0,maxArgs:1},getBytesInUse:{minArgs:0,maxArgs:1}},sync:{clear:{minArgs:0,maxArgs:0},get:{minArgs:0,maxArgs:1},getBytesInUse:{minArgs:0,maxArgs:1},remove:{minArgs:1,maxArgs:1},set:{minArgs:1,maxArgs:1}}},tabs:{captureVisibleTab:{minArgs:0,maxArgs:2},create:{minArgs:1,maxArgs:1},detectLanguage:{minArgs:0,maxArgs:1},discard:{minArgs:0,maxArgs:1},duplicate:{minArgs:1,maxArgs:1},executeScript:{minArgs:1,maxArgs:2},get:{minArgs:1,maxArgs:1},getCurrent:{minArgs:0,maxArgs:0},getZoom:{minArgs:0,maxArgs:1},getZoomSettings:{minArgs:0,maxArgs:1},goBack:{minArgs:0,maxArgs:1},goForward:{minArgs:0,maxArgs:1},highlight:{minArgs:1,maxArgs:1},insertCSS:{minArgs:1,maxArgs:2},move:{minArgs:2,maxArgs:2},query:{minArgs:1,maxArgs:1},reload:{minArgs:0,maxArgs:2},remove:{minArgs:1,maxArgs:1},removeCSS:{minArgs:1,maxArgs:2},sendMessage:{minArgs:2,maxArgs:3},setZoom:{minArgs:1,maxArgs:2},setZoomSettings:{minArgs:1,maxArgs:2},update:{minArgs:1,maxArgs:2}},topSites:{get:{minArgs:0,maxArgs:0}},webNavigation:{getAllFrames:{minArgs:1,maxArgs:1},getFrame:{minArgs:1,maxArgs:1}},webRequest:{handlerBehaviorChanged:{minArgs:0,maxArgs:0}},windows:{create:{minArgs:0,maxArgs:1},get:{minArgs:1,maxArgs:2},getAll:{minArgs:0,maxArgs:1},getCurrent:{minArgs:0,maxArgs:1},getLastFocused:{minArgs:0,maxArgs:1},remove:{minArgs:1,maxArgs:1},update:{minArgs:2,maxArgs:2}}};if(0===Object.keys(b).length)throw new Error("api-metadata.json has not been included in browser-polyfill");class c extends WeakMap{constructor(a,b=void 0){super(b),this.createItem=a}get(a){return this.has(a)||this.set(a,this.createItem(a)),super.get(a)}}const d=a=>a&&"object"==typeof a&&"function"==typeof a.then,e=(b,c)=>(...d)=>{a.runtime.lastError?b.reject(new Error(a.runtime.lastError.message)):c.singleCallbackArg||1>=d.length&&!1!==c.singleCallbackArg?b.resolve(d[0]):b.resolve(d)},f=a=>1==a?"argument":"arguments",g=(a,b)=>function(c,...d){if(d.length<b.minArgs)throw new Error(`Expected at least ${b.minArgs} ${f(b.minArgs)} for ${a}(), got ${d.length}`);if(d.length>b.maxArgs)throw new Error(`Expected at most ${b.maxArgs} ${f(b.maxArgs)} for ${a}(), got ${d.length}`);return new Promise((f,g)=>{if(b.fallbackToNoCallback)try{c[a](...d,e({resolve:f,reject:g},b))}catch(e){console.warn(`${a} API method doesn't seem to support the callback parameter, `+"falling back to call it without a callback: ",e),c[a](...d),b.fallbackToNoCallback=!1,b.noCallback=!0,f()}else b.noCallback?(c[a](...d),f()):c[a](...d,e({resolve:f,reject:g},b))})},h=(a,b,c)=>new Proxy(b,{apply(b,d,e){return c.call(d,a,...e)}});let i=Function.call.bind(Object.prototype.hasOwnProperty);const j=(a,b={},c={})=>{let d=Object.create(null),e=Object.create(a);return new Proxy(e,{has(b,c){return c in a||c in d},get(e,f){if(f in d)return d[f];if(!(f in a))return;let k=a[f];if("function"==typeof k){if("function"==typeof b[f])k=h(a,a[f],b[f]);else if(i(c,f)){let b=g(f,c[f]);k=h(a,a[f],b)}else k=k.bind(a);}else if("object"==typeof k&&null!==k&&(i(b,f)||i(c,f)))k=j(k,b[f],c[f]);else if(i(c,"*"))k=j(k,b[f],c["*"]);else return Object.defineProperty(d,f,{configurable:!0,enumerable:!0,get(){return a[f]},set(b){a[f]=b}}),k;return d[f]=k,k},set(b,c,e){return c in d?d[c]=e:a[c]=e,!0},defineProperty(a,b,c){return Reflect.defineProperty(d,b,c)},deleteProperty(a,b){return Reflect.deleteProperty(d,b)}})},k=a=>({addListener(b,c,...d){b.addListener(a.get(c),...d)},hasListener(b,c){return b.hasListener(a.get(c))},removeListener(b,c){b.removeListener(a.get(c))}}),l=new c(a=>"function"==typeof a?function(b){const c=j(b,{},{getContent:{minArgs:0,maxArgs:0}});a(c)}:a),m=new c(a=>"function"==typeof a?function(b,c,e){let f,g,h=!1,i=new Promise(a=>{f=function(b){h=!0,a(b)}});try{g=a(b,c,f)}catch(a){g=Promise.reject(a)}const j=!0!==g&&d(g);if(!0!==g&&!j&&!h)return!1;const k=a=>{a.then(a=>{e(a)},a=>{let b;b=a&&(a instanceof Error||"string"==typeof a.message)?a.message:"An unexpected error occurred",e({__mozWebExtensionPolyfillReject__:!0,message:b})}).catch(a=>{console.error("Failed to send onMessage rejected reply",a)})};return j?k(g):k(i),!0}:a),n=({reject:b,resolve:c},d)=>{a.runtime.lastError?a.runtime.lastError.message==="The message port closed before a response was received."?c():b(new Error(a.runtime.lastError.message)):d&&d.__mozWebExtensionPolyfillReject__?b(new Error(d.message)):c(d)},o=(a,b,c,...d)=>{if(d.length<b.minArgs)throw new Error(`Expected at least ${b.minArgs} ${f(b.minArgs)} for ${a}(), got ${d.length}`);if(d.length>b.maxArgs)throw new Error(`Expected at most ${b.maxArgs} ${f(b.maxArgs)} for ${a}(), got ${d.length}`);return new Promise((a,b)=>{const e=n.bind(null,{resolve:a,reject:b});d.push(e),c.sendMessage(...d)})},p={devtools:{network:{onRequestFinished:k(l)}},runtime:{onMessage:k(m),onMessageExternal:k(m),sendMessage:o.bind(null,"sendMessage",{minArgs:1,maxArgs:3})},tabs:{sendMessage:o.bind(null,"sendMessage",{minArgs:2,maxArgs:3})}},q={clear:{minArgs:1,maxArgs:1},get:{minArgs:1,maxArgs:1},set:{minArgs:1,maxArgs:1}};return b.privacy={network:{"*":q},services:{"*":q},websites:{"*":q}},j(a,p,b)})(chrome)}else a.exports=globalThis.browser});
//# sourceMappingURL=browser-polyfill.min.js.map
// webextension-polyfill v.0.12.0 (https://github.com/mozilla/webextension-polyfill)
/* This Source Code Form is subject to the terms of the Mozilla Public
* License, v. 2.0. If a copy of the MPL was not distributed with this
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
/* mousetrap v1.6.5 craig.is/killing/mice */
(function(q,u,c){function v(a,b,g){a.addEventListener?a.addEventListener(b,g,!1):a.attachEvent("on"+b,g)}function z(a){if("keypress"==a.type){var b=String.fromCharCode(a.which);a.shiftKey||(b=b.toLowerCase());return b}return n[a.which]?n[a.which]:r[a.which]?r[a.which]:String.fromCharCode(a.which).toLowerCase()}function F(a){var b=[];a.shiftKey&&b.push("shift");a.altKey&&b.push("alt");a.ctrlKey&&b.push("ctrl");a.metaKey&&b.push("meta");return b}function w(a){return"shift"==a||"ctrl"==a||"alt"==a||
"meta"==a}function A(a,b){var g,d=[];var e=a;"+"===e?e=["+"]:(e=e.replace(/\+{2}/g,"+plus"),e=e.split("+"));for(g=0;g<e.length;++g){var m=e[g];B[m]&&(m=B[m]);b&&"keypress"!=b&&C[m]&&(m=C[m],d.push("shift"));w(m)&&d.push(m)}e=m;g=b;if(!g){if(!p){p={};for(var c in n)95<c&&112>c||n.hasOwnProperty(c)&&(p[n[c]]=c)}g=p[e]?"keydown":"keypress"}"keypress"==g&&d.length&&(g="keydown");return{key:m,modifiers:d,action:g}}function D(a,b){return null===a||a===u?!1:a===b?!0:D(a.parentNode,b)}function d(a){function b(a){a=
a||{};var b=!1,l;for(l in p)a[l]?b=!0:p[l]=0;b||(x=!1)}function g(a,b,t,f,g,d){var l,E=[],h=t.type;if(!k._callbacks[a])return[];"keyup"==h&&w(a)&&(b=[a]);for(l=0;l<k._callbacks[a].length;++l){var c=k._callbacks[a][l];if((f||!c.seq||p[c.seq]==c.level)&&h==c.action){var e;(e="keypress"==h&&!t.metaKey&&!t.ctrlKey)||(e=c.modifiers,e=b.sort().join(",")===e.sort().join(","));e&&(e=f&&c.seq==f&&c.level==d,(!f&&c.combo==g||e)&&k._callbacks[a].splice(l,1),E.push(c))}}return E}function c(a,b,c,f){k.stopCallback(b,
b.target||b.srcElement,c,f)||!1!==a(b,c)||(b.preventDefault?b.preventDefault():b.returnValue=!1,b.stopPropagation?b.stopPropagation():b.cancelBubble=!0)}function e(a){"number"!==typeof a.which&&(a.which=a.keyCode);var b=z(a);b&&("keyup"==a.type&&y===b?y=!1:k.handleKey(b,F(a),a))}function m(a,g,t,f){function h(c){return function(){x=c;++p[a];clearTimeout(q);q=setTimeout(b,1E3)}}function l(g){c(t,g,a);"keyup"!==f&&(y=z(g));setTimeout(b,10)}for(var d=p[a]=0;d<g.length;++d){var e=d+1===g.length?l:h(f||
A(g[d+1]).action);n(g[d],e,f,a,d)}}function n(a,b,c,f,d){k._directMap[a+":"+c]=b;a=a.replace(/\s+/g," ");var e=a.split(" ");1<e.length?m(a,e,b,c):(c=A(a,c),k._callbacks[c.key]=k._callbacks[c.key]||[],g(c.key,c.modifiers,{type:c.action},f,a,d),k._callbacks[c.key][f?"unshift":"push"]({callback:b,modifiers:c.modifiers,action:c.action,seq:f,level:d,combo:a}))}var k=this;a=a||u;if(!(k instanceof d))return new d(a);k.target=a;k._callbacks={};k._directMap={};var p={},q,y=!1,r=!1,x=!1;k._handleKey=function(a,
d,e){var f=g(a,d,e),h;d={};var k=0,l=!1;for(h=0;h<f.length;++h)f[h].seq&&(k=Math.max(k,f[h].level));for(h=0;h<f.length;++h)f[h].seq?f[h].level==k&&(l=!0,d[f[h].seq]=1,c(f[h].callback,e,f[h].combo,f[h].seq)):l||c(f[h].callback,e,f[h].combo);f="keypress"==e.type&&r;e.type!=x||w(a)||f||b(d);r=l&&"keydown"==e.type};k._bindMultiple=function(a,b,c){for(var d=0;d<a.length;++d)n(a[d],b,c)};v(a,"keypress",e);v(a,"keydown",e);v(a,"keyup",e)}if(q){var n={8:"backspace",9:"tab",13:"enter",16:"shift",17:"ctrl",
18:"alt",20:"capslock",27:"esc",32:"space",33:"pageup",34:"pagedown",35:"end",36:"home",37:"left",38:"up",39:"right",40:"down",45:"ins",46:"del",91:"meta",93:"meta",224:"meta"},r={106:"*",107:"+",109:"-",110:".",111:"/",186:";",187:"=",188:",",189:"-",190:".",191:"/",192:"`",219:"[",220:"\\",221:"]",222:"'"},C={"~":"`","!":"1","@":"2","#":"3",$:"4","%":"5","^":"6","&":"7","*":"8","(":"9",")":"0",_:"-","+":"=",":":";",'"':"'","<":",",">":".","?":"/","|":"\\"},B={option:"alt",command:"meta","return":"enter",
escape:"esc",plus:"+",mod:/Mac|iPod|iPhone|iPad/.test(navigator.platform)?"meta":"ctrl"},p;for(c=1;20>c;++c)n[111+c]="f"+c;for(c=0;9>=c;++c)n[c+96]=c.toString();d.prototype.bind=function(a,b,c){a=a instanceof Array?a:[a];this._bindMultiple.call(this,a,b,c);return this};d.prototype.unbind=function(a,b){return this.bind.call(this,a,function(){},b)};d.prototype.trigger=function(a,b){if(this._directMap[a+":"+b])this._directMap[a+":"+b]({},a);return this};d.prototype.reset=function(){this._callbacks={};
this._directMap={};return this};d.prototype.stopCallback=function(a,b){if(-1<(" "+b.className+" ").indexOf(" mousetrap ")||D(b,this.target))return!1;if("composedPath"in a&&"function"===typeof a.composedPath){var c=a.composedPath()[0];c!==a.target&&(b=c)}return"INPUT"==b.tagName||"SELECT"==b.tagName||"TEXTAREA"==b.tagName||b.isContentEditable};d.prototype.handleKey=function(){return this._handleKey.apply(this,arguments)};d.addKeycodes=function(a){for(var b in a)a.hasOwnProperty(b)&&(n[b]=a[b]);p=null};
d.init=function(){var a=d(u),b;for(b in a)"_"!==b.charAt(0)&&(d[b]=function(b){return function(){return a[b].apply(a,arguments)}}(b))};d.init();q.Mousetrap=d;"undefined"!==typeof module&&module.exports&&(module.exports=d);"function"===typeof define&&define.amd&&define(function(){return d})}})("undefined"!==typeof window?window:null,"undefined"!==typeof window?document:null);
(function(a){var c={},d=a.prototype.stopCallback;a.prototype.stopCallback=function(e,b,a,f){return this.paused?!0:c[a]||c[f]?!1:d.call(this,e,b,a)};a.prototype.bindGlobal=function(a,b,d){this.bind(a,b,d);if(a instanceof Array)for(b=0;b<a.length;b++)c[a[b]]=!0;else c[a]=!0};a.init()})(Mousetrap);
const DEFAULT_CSS = `/* NOTE:
*
* - Using !important is needed for some styles because otherwise they get
* overriden by the search engine stylesheets
* - Using outline works better than border sometimes because creating the
* border can move other elements, for example the page numbers are moved in
* Google Scholar when highlighting the prev/next buttons.
*/
:root {
--result-outline: 1px solid black;
}
@media (prefers-color-scheme: dark) {
:root {
--result-outline: 1px solid #aaaaaa;
}
}
html[dark], [dark] {
--result-outline: 1px solid #aaaaaa;
}
.wsn-google-focused-link {
position: relative;
/* This is required for the arrow to appear when navigating sub-results, see
* also: https://github.com/infokiller/web-search-navigator/issues/357 */
overflow: visible !important;
}
.wsn-google-focused-link::before,
.wsn-google-focused-map::before,
.wsn-gitlab-focused-link::before,
.wsn-brave-search-focused-link::before,
.wsn-startpage-focused-link::before {
content: "\u25BA";
margin-right: 25px;
left: -25px;
position: absolute;
}
.wsn-brave-search-focused-news {
position: relative;
}
.wsn-brave-search-focused-news::before {
content: "\u25BA";
top: 5px;
left: -45px;
position: absolute;
}
.wsn-google-focused-image {
outline: var(--result-outline) !important;
/* Images are less visible with a thin outline */
outline-width: 2px;
}
.wsn-google-focused-card,
.wsn-brave-search-focused-card,
.wsn-google-focused-job-card {
border: var(--result-outline) !important;
}
.wsn-google-focused-map,
.wsn-google-card-item,
.wsn-gitlab-focused-group-row {
outline: var(--result-outline) !important;
}
.wsn-google-focused-memex-result {
border: var(--result-outline) !important;
box-sizing: border-box;
-moz-box-sizing: border-box;
-webkit-box-sizing: border-box;
}
/* Startpage has dark themes where a black outline won't be visible */
.wsn-startpage-focused-link {
outline: 1px solid #435a69 !important;
outline-offset: 3px;
}
.wsn-youtube-focused-video {
outline: var(--result-outline) !important;
outline-offset: 1px;
}
.wsn-youtube-focused-grid-video {
border: var(--result-outline) !important;
}
.wsn-google-scholar-next-page {
/* Using outline works better than border for the Scholar previous/next
* buttons because border moves the page numbers a bit. */
outline: var(--result-outline) !important;
}
.wsn-amazon-focused-item {
outline: var(--result-outline) !important;
outline-offset: 3px;
}
.wsn-amazon-focused-cart-item,
.wsn-amazon-focused-carousel-item {
border: var(--result-outline) !important;
}
.wsn-github-focused-item,
.wsn-github-focused-pagination {
outline: var(--result-outline) !important;
outline-offset: 2px;
}
/* This rule is only used when the "hide outline" option is enabled, and is used
* to disable the website's default search result outlining */
.wsn-no-outline,
.wsn-no-outline:focus {
outline: none;
}`;
const DEFAULT_KEYBINDINGS = {
nextKey: ['down', 'j'],
previousKey: ['up', 'k'],
navigatePreviousResultPage: ['left', 'h'],
navigateNextResultPage: ['right', 'l'],
navigateKey: ['return', 'space'],
navigateNewTabBackgroundKey: ['ctrl+return', 'command+return', 'ctrl+space'],
navigateNewTabKey: [
'ctrl+shift+return',
'command+shift+return',
'ctrl+shift+space',
],
navigateSearchTab: ['a', 's'],
navigateImagesTab: ['i'],
navigateVideosTab: ['v'],
navigateMapsTab: ['m'],
navigateNewsTab: ['n'],
navigateShoppingTab: ['alt+s'],
navigateBooksTab: ['b'],
navigateFlightsTab: ['alt+l'],
navigateFinancialTab: ['f'],
focusSearchInput: ['/', 'escape'],
navigateShowAll: ['z z'],
navigateShowHour: ['z h'],
navigateShowDay: ['z d'],
navigateShowWeek: ['z w'],
navigateShowMonth: ['z m'],
navigateShowYear: ['z y'],
toggleSort: ['z s'],
toggleVerbatimSearch: ['z v'],
showImagesLarge: ['z l'],
showImagesMedium: ['z e'],
showImagesIcon: ['z i'],
copyUrlKey: [],
};
const DEFAULT_OPTIONS = {
...DEFAULT_KEYBINDINGS,
wrapNavigation: false,
autoSelectFirst: true,
hideOutline: false,
delay: 0,
googleIncludeCards: true,
googleIncludeMemex: false,
googleIncludePlaces: true,
customCSS: DEFAULT_CSS,
simulateMiddleClick: false,
customGitlabUrl: '^https://(www.)?\\.*git.*\\.',
};
const keybindingStringToArray = (kb) => {
// Alternative: kb.split(/, */);
return kb.split(',').map((t) => t.trim());
};
// eslint-disable-next-line no-unused-vars
const keybindingArrayToString = (kb) => {
return kb.join(', ');
};
/**
* @param {StorageArea} storage The storage area to which this section will
* write.
* @param {Object} defaultValues The default options.
* @constructor
*/
class BrowserStorage {
constructor(storage, defaultValues) {
this.storage = storage;
this.values = {};
this.defaultValues = defaultValues;
}
load() {
// this.storage.get(null) returns all the data stored:
// https://developer.chrome.com/extensions/storage#method-StorageArea-get
return this.storage.get(null).then((values) => {
this.values = values;
// Prior to versions 0.4.* the keybindings were stored as strings, so we
// migrate them to arrays if needed.
let migrated = false;
for (const [key, value] of Object.entries(this.values)) {
if (!(key in DEFAULT_KEYBINDINGS) || Array.isArray(value)) {
continue;
}
migrated = true;
this.values[key] = keybindingStringToArray(value);
}
if (migrated) {
return this.save();
}
});
}
save() {
return this.storage.set(this.values);
}
get(key) {
const value = this.values[key];
if (value != null) {
return value;
}
return this.defaultValues[key];
}
set(key, value) {
this.values[key] = value;
}
clear() {
return this.storage.clear().then(() => {
this.values = {};
});
}
getAll() {
// Merge options from storage with defaults.
return { ...this.defaultValues, ...this.values };
}
}
const STORAGE_KEY = 'webSearchNavigator';
class LocalStorage {
constructor(defaultValues) {
this.values = {};
this.defaultValues = defaultValues;
this.load();
}
load() {
const storedData = localStorage.getItem(STORAGE_KEY);
if (storedData) {
this.values = JSON.parse(storedData);
} else {
this.values = { ...this.defaultValues };
this.save();
}
}
save() {
localStorage.setItem(STORAGE_KEY, JSON.stringify(this.values));
}
get(key) {
const value = this.values[key];
if (value != null) {
return value;
}
return this.defaultValues[key];
}
set(key, value) {
this.values[key] = value;
this.save();
}
clear() {
localStorage.removeItem(STORAGE_KEY);
this.values = { ...this.defaultValues };
}
getAll() {
// Merge options from storage with defaults.
return { ...this.defaultValues, ...this.values };
}
}
const createSyncedOptions = () => {
if (globalThis.IS_USERSCRIPT) {
console.log('Create LocalStorage options');
return new LocalStorage(DEFAULT_OPTIONS);
}
return new BrowserStorage(browser.storage.sync, DEFAULT_OPTIONS);
};
// eslint-disable-next-line no-unused-vars
class ExtensionOptions {
constructor() {
this.sync = createSyncedOptions();
if (globalThis.IS_USERSCRIPT) {
this.local = createSyncedOptions();
return;
}
this.local = new BrowserStorage(browser.storage.local, {
lastQueryUrl: null,
lastFocusedIndex: 0,
});
}
load() {
return Promise.all([this.local.load(), this.sync.load()]);
}
}
/**
* This file contains search engine specific logic via search engine objects.
*
* A search engine object must provide the following:
* - {regex} urlPattern
* - {CSS selector} searchBoxSelector
* - {SearchResult[]} getSearchResults()
*
* Optional functions/properties:
* - {Array} tabs
* Default: {}
* - {int} getTopMargin: used if top results are not entirely visible
* Default: 0
* - {int} getBottomMargin: used if bottom results are not entirely visible.
* Relevant for some search engines, since Firefox and Chrome show a tooltip
* with the URL of focused links at the bottom and can hide some of the
* search result at the bottom.
* Default: getDefaultBottomMargin()
* - {Function} onChangedResults: function for registering a callback on
* changed search results. The callback gets a single boolean parameter that
* is set to true if the only change is newly appended results.
* Default: null (meaning there's no support for such events)
* - {None} changeTools(period)
*
* Every SearchResult must provide the element and highlightClass properties and
* optionally the following:
* - {Callback} anchorSelector: callback for getting the anchor
* Default: the element itself
* - {Callback} highlightedElementSelector: callback for getting the
* highlighted element
* Default: the element itself
* - {Callback} containerSelector: callback for getting the container that
* needs to be visible when an element is selected.
* Default: the element itself
*/
class SearchResult {
// We must declare the private class fields.
#element;
#anchorSelector;
#highlightedElementSelector;
#containerSelector;
/**
* @param {Element} element
* @param {function|null} anchorSelector
* @param {string} highlightClass
* @param {function|null} highlightedElementSelector
* @param {function|null} containerSelector
*/
constructor(
element,
anchorSelector,
highlightClass,
highlightedElementSelector,
containerSelector,
) {
this.#element = element;
this.#anchorSelector = anchorSelector;
this.highlightClass = highlightClass;
this.#highlightedElementSelector = highlightedElementSelector;
this.#containerSelector = containerSelector;
}
get anchor() {
if (!this.#anchorSelector) {
return this.#element;
}
return this.#anchorSelector(this.#element);
}
get container() {
if (!this.#containerSelector) {
return this.#element;
}
return this.#containerSelector(this.#element);
}
get highlightedElement() {
if (!this.#highlightedElementSelector) {
return this.#element;
}
return this.#highlightedElementSelector(this.#element);
}
}
// eslint-disable-next-line
/**
* @param {Array} includedSearchResults An array of
* tuples. Each tuple contains collection of the search results optionally
* accompanied with their container selector.
* @constructor
*/
const getSortedSearchResults = (
includedSearchResults,
excludedNodeList = [],
) => {
const excludedResultsSet = new Set();
for (const node of excludedNodeList) {
excludedResultsSet.add(node);
}
const searchResults = [];
for (const results of includedSearchResults) {
for (const node of results.nodes) {
const searchResult = new SearchResult(
node,
results.anchorSelector,
results.highlightClass,
results.highlightedElementSelector,
results.containerSelector,
);
const anchor = searchResult.anchor;
// Use offsetParent to exclude hidden elements, see:
// https://stackoverflow.com/a/21696585/1014208
if (
anchor != null &&
!excludedResultsSet.has(anchor) &&
anchor.offsetParent !== null
) {
// Prevent adding the same node multiple times.
excludedResultsSet.add(anchor);
searchResults.push(searchResult);
}
}
}
// Sort searchResults by their document position.
searchResults.sort((a, b) => {
const position = a.anchor.compareDocumentPosition(b.anchor);
if (position & Node.DOCUMENT_POSITION_FOLLOWING) {
return -1;
} else if (position & Node.DOCUMENT_POSITION_PRECEDING) {
return 1;
} else {
return 0;
}
});
return searchResults;
};
const getFixedSearchBoxTopMargin = (searchBoxContainer, element) => {
// When scrolling down, the search box can have a fixed position and can hide
// search results, so we try to compensate for it.
if (!searchBoxContainer || searchBoxContainer.contains(element)) {
return 0;
}
return searchBoxContainer.getBoundingClientRect().height;
};
// https://stackoverflow.com/a/7000222/2870889
// eslint-disable-next-line no-unused-vars
const isFirefox = () => {
return navigator.userAgent.toLowerCase().indexOf('firefox') >= 0;
};
// eslint-disable-next-line no-unused-vars
const getDefaultBottomMargin = (element) => {
return 28;
};
const selectorElementGetter = (selector) => {
return () => {
return document.querySelector(selector);
};
};
const nParent = (element, n) => {
while (n > 0 && element) {
element = element.parentElement;
n--;
}
return element;
};
const debounce = (callback, delayMs) => {
let timeoutId;
return (...args) => {
clearTimeout(timeoutId);
timeoutId = setTimeout(() => {
return callback(...args);
}, delayMs);
};
};
class GoogleSearch {
constructor(options) {
this.options = options;
}
get urlPattern() {
return /^https:\/\/(www\.)?google\./;
}
get searchBoxSelector() {
// Must match search engine search box
// NOTE: we used '#searchform input[name=q]' before 2020-06-05 but that
// doesn't work in the images search tab. Another option is to use
// 'input[role="combobox"]' but this doesn't work when there's also a
// dictionary search box.
// return '#searchform input[name=q]',
return 'form[role=search] [name=q]';
}
getTopMargin(element) {
return getFixedSearchBoxTopMargin(
document.querySelector('#searchform.minidiv'),
element,
);
}
getBottomMargin(element) {
return isFirefox() ? 0 : getDefaultBottomMargin();
}
onChangedResults(callback) {
if (GoogleSearch.#isImagesTab()) {
return GoogleSearch.#onImageSearchResults(callback);
}
if (this.options.googleIncludeMemex) {
return GoogleSearch.#onMemexResults(callback);
}
// https://github.com/infokiller/web-search-navigator/issues/464
const container = document.querySelector('#rcnt');
if (!container) {
return;
}
const observer = new MutationObserver(
debounce((mutationsList, observer) => {
callback(true);
}, 50),
);
observer.observe(container, {
attributes: false,
childList: true,
subtree: true,
});
}
static #isImagesTab() {
const searchParams = new URLSearchParams(window.location.search);
return searchParams.get('tbm') === 'isch';
}
static #getImagesTabResults() {
const includedElements = [
// Image links
{
nodes: document.querySelectorAll('.islrc a[data-nav="1"]'),
highlightClass: 'wsn-google-focused-image',
},
// Show more results button
{
nodes: document.querySelectorAll('#islmp [type="button"]'),
highlightClass: 'wsn-google-focused-image',
},
];
return getSortedSearchResults(includedElements, []);
}
static #regularResults() {
return [
{
nodes: document.querySelectorAll('#search .r > a:first-of-type'),
highlightClass: 'wsn-google-focused-link',
containerSelector: (n) => n.parentElement.parentElement,
},
{
nodes: document.querySelectorAll('#search .r g-link > a:first-of-type'),
highlightClass: 'wsn-google-focused-link',
containerSelector: (n) => n.parentElement.parentElement,
},
// More results button in continous loading
// https://imgur.com/a/X9zyJ24
{
nodes: document.querySelectorAll(
'#botstuff a[href^="/search"][href*="start="] h3',
),
highlightClass: 'wsn-google-focused-link',
anchorSelector: (n) => n.closest('a'),
},
// Continuously loaded results are *sometimes* in the #botstuff container
// https://imgur.com/a/s6ow0La
{
nodes: document.querySelectorAll('#botstuff a h3'),
highlightClass: 'wsn-google-focused-link',
containerSelector: (n) => nParent(n, 5),
highlightedElementSelector: (n) => nParent(n, 5),
anchorSelector: (n) => n.closest('a'),
},
// Sometimes featured snippets are not contained in #search (possibly when
// there are large images?): https://imgur.com/a/VluRKIQ
{
nodes: document.querySelectorAll('.xpdopen .g a'),
highlightClass: 'wsn-google-focused-link',
highlightedElementSelector: (n) => n.querySelector('h3'),
},
// Large YouTube video as top result: https://imgur.com/a/JIe62QV
{
nodes: document.querySelectorAll('h3 a[href*="youtube.com"]'),
highlightClass: 'wsn-google-focused-link',
highlightedElementSelector: (n) => n.closest('h3'),
},
// Sub-results: https://imgur.com/a/CJePYJM
{
nodes: document.querySelectorAll('#search h3 a:first-of-type'),
highlightClass: 'wsn-google-focused-link',
highlightedElementSelector: (n) => n.closest('h3'),
containerSelector: (n) => n.closest('tr'),
},
// Shopping results: https://imgur.com/a/wccM2iq
{
nodes: document.querySelectorAll('#rso a h4'),
anchorSelector: (n) => n.closest('a'),
highlightClass: 'wsn-google-focused-card',
highlightedElementSelector: (n) => n.closest('.sh-dgr__content'),
},
// News tab: https://imgur.com/a/MR9q31f
{
nodes: document.querySelectorAll('#search g-card a'),
highlightClass: 'wsn-google-focused-link',
},
// Jobs heading for the jobs cards section. Clicking on it takes you
// to Google's job search.
// As of 2023-05-28, the Google's jobs search URLs seem to contain two
// query string params which seem relevant:
// - ibp=htl;jobs
// - htivrt=jobs
// The first one matches the jobs heading, but also buttons in the
// jobs UI such as filtering by WFH/in-office. Therefore, we use the
// second one for specific jobs, but the first one to detect the jobs
// heading (otherwise it would be matched later in vaccines).
// eslint-disable-next-line max-len
// const jobsSelector = '#search a:is([href*="ibp=htl;jobs"], [href*="htivrt=jobs"])';
// NOTE: this must be added to the included elements before:
// - vaccines
// - vertical maps
// - books and featured snippets
// TODO: add screenshot
{
nodes: document.querySelectorAll(
// eslint-disable-next-line max-len
'#search a:is([href*="ibp=htl;jobs"],[href*="htivrt=jobs"]) [role=heading][aria-level="2"]',
),
anchorSelector: (n) => n.closest('a'),
// highlightedElementSelector: (n) => n.closest('li'),
highlightClass: 'wsn-google-focused-job-card',
},
// Same as above, but for specific job results.
// TODO: add screenshot
// Jobs cards
{
// nodes: document.querySelectorAll('#search a[href*="htivrt=jobs"]'),
// eslint-disable-next-line max-len
nodes: document.querySelectorAll('#search li a[href*="htivrt=jobs"]'),
highlightedElementSelector: (n) => n.closest('li'),
highlightClass: 'wsn-google-focused-job-card',
},
// Books tab: https://imgur.com/a/QSBIOb6
// NOTE: This is required for matching "features snippets" in the general
// search tab, and also matches other results.
{
nodes: document.querySelectorAll('#search [data-hveid] a h3'),
anchorSelector: (n) => n.closest('a'),
containerSelector: (n) => n.closest('[data-hveid]'),
highlightedElementSelector: (n) => n.closest('[data-hveid]'),
highlightClass: 'wsn-google-focused-link',
},
// Next/previous results page
{
nodes: document.querySelectorAll('#pnprev, #pnnext'),
highlightClass: 'wsn-google-card-item',
},
];
}
static #cardResults() {
const nearestChildOrSiblingOrParentAnchor = (element) => {
const childAnchor = element.querySelector('a');
if (childAnchor && childAnchor.href) {
return childAnchor;
}
const siblingAnchor = element.parentElement.querySelector('a');
if (siblingAnchor && siblingAnchor.href) {
return siblingAnchor;
}
return element.closest('a');
};
const nearestCardContainer = (element) => {
return element.closest('g-inner-card');
};
return [
// Twitter: https://imgur.com/a/fdI75JG
{
nodes: document.querySelectorAll(
'#search [data-init-vis=true] [role=heading]',
),
anchorSelector: nearestChildOrSiblingOrParentAnchor,
highlightedElementSelector: nearestCardContainer,
highlightClass: 'wsn-google-focused-card',
},
// Vertical "Top stories" results
{
nodes: document.querySelectorAll('#search [role=text] [role=heading]'),
anchorSelector: nearestChildOrSiblingOrParentAnchor,
highlightClass: 'wsn-google-focused-link',
},
// Vertical video results: https://imgur.com/a/GyKhwrx
// Vertical video results: https://imgur.com/a/8fbPnvT
{
nodes: document.querySelectorAll(
'#search video-voyager a [role=heading]',
),
anchorSelector: nearestChildOrSiblingOrParentAnchor,
containerSelector: nearestChildOrSiblingOrParentAnchor,
highlightedElementSelector: nearestChildOrSiblingOrParentAnchor,
highlightClass: 'wsn-google-focused-link',
},
// Horizontal video results: https://imgur.com/a/gRGJ7l9
// People also search for: https://imgur.com/a/QpCHKt0
{
nodes: document.querySelectorAll(
'#search g-scrolling-carousel g-inner-card a [role=heading]',
),
anchorSelector: nearestChildOrSiblingOrParentAnchor,
containerSelector: nearestCardContainer,
highlightedElementSelector: nearestCardContainer,
highlightClass: 'wsn-google-card-item',
},
// Vaccines: https://imgur.com/a/325qJzE
{
nodes: document.querySelectorAll(
'#search a.a-no-hover-decoration [role=heading]',
),
anchorSelector: nearestChildOrSiblingOrParentAnchor,
containerSelector: nearestChildOrSiblingOrParentAnchor,
highlightedElementSelector: nearestChildOrSiblingOrParentAnchor,
highlightClass: 'wsn-google-focused-link',
},
// Things to do in X: https://imgur.com/a/ibXwiuT
{
nodes: document.querySelectorAll('td a [role=heading]'),
anchorSelector: nearestChildOrSiblingOrParentAnchor,
containerSelector: (n) => n.closest('td'),
highlightedElementSelector: (n) => n.closest('td'),
highlightClass: 'wsn-google-card-item',
},
// Vertical Maps/Places: https://imgur.com/a/JXrxBCj
// Vertical recipes: https://imgur.com/a/3r7klHk
// Top stories grid: https://imgur.com/a/mY93YRF
// TODO: fix the small movements in recipes item selection.
{
nodes: document.querySelectorAll('a [role=heading]'),
anchorSelector: nearestChildOrSiblingOrParentAnchor,
containerSelector: nearestChildOrSiblingOrParentAnchor,
highlightedElementSelector: nearestChildOrSiblingOrParentAnchor,
highlightClass: 'wsn-google-card-item',
},
];
}
static #placesResults() {
const nodes = document.querySelectorAll('.vk_c a');
// The first node is usually the map image which needs to be styled
// differently.
let map;
let links = nodes;
if (nodes[0] != null && nodes[0].querySelector('img')) {
map = nodes[0];
links = Array.from(nodes).slice(1);
}
const results = [];
if (map != null) {
results.push({
nodes: [map],
highlightedElementSelector: (n) => n.parentElement,
highlightClass: 'wsn-google-focused-map',
});
}
results.push({
nodes: links,
highlightClass: 'wsn-google-focused-link',
});
return results;
}
static #memexResults() {
return [
{
nodes: document.querySelectorAll(
'#memexResults ._3d3zwUrsb4CVi1Li4H6CBw a',
),
highlightClass: 'wsn-google-focused-memex-result',
},
];
}
getSearchResults() {
if (GoogleSearch.#isImagesTab()) {
return GoogleSearch.#getImagesTabResults();
}
const includedElements = GoogleSearch.#regularResults();
if (this.options.googleIncludeCards) {
includedElements.push(...GoogleSearch.#cardResults());
}
if (this.options.googleIncludePlaces) {
includedElements.push(...GoogleSearch.#placesResults());
}
if (this.options.googleIncludeMemex) {
includedElements.push(...GoogleSearch.#memexResults());
}
const excludedElements = document.querySelectorAll(
[
// People also ask. Each one of the used selectors should be
// sufficient, but we use both to be more robust to upstream DOM
// changes.
'.related-question-pair a',
'#search .kp-blk:not(.c2xzTb) .r > a:first-of-type',
// Right hand sidebar. We exclude it because it is after all the
// results in the document order (as determined by
// Node.DOCUMENT_POSITION_FOLLOWING used in getSortedSearchResults),
// and it's confusing.
'#rhs a',
].join(', '),
);
return getSortedSearchResults(includedElements, excludedElements);
}
static #onImageSearchResults(callback) {
const container = document.querySelector('.islrc');
if (!container) {
return;
}
const observer = new MutationObserver(
debounce((mutationsList, observer) => {
callback(true);
}, 50),
);
observer.observe(container, {
attributes: false,
childList: true,
subtree: false,
});
}
static #onMemexResults(callback) {
const container = document.querySelector('#rhs');
if (!container) {
return;
}
const observer = new MutationObserver(
debounce((mutationsList, observer) => {
if (document.querySelector('#memexResults') != null) {
callback(true);
}
}, 50),
);
observer.observe(container, {
attributes: false,
childList: true,
subtree: true,
});
}
static #imageSearchTabs() {
const visibleTabs = document.querySelectorAll('.T47uwc > a');
// NOTE: The order of the tabs after the first two is dependent on the
// query. For example:
// - "cats": videos, news, maps
// - "trump": news, videos, maps
// - "california": maps, news, videos
return {
navigateSearchTab: visibleTabs[0],
navigateMapsTab: selectorElementGetter(
'.T47uwc > a[href*="maps.google."]',
),
navigateVideosTab: selectorElementGetter('.T47uwc > a[href*="&tbm=vid"]'),
navigateNewsTab: selectorElementGetter('.T47uwc > a[href*="&tbm=nws"]'),
navigateShoppingTab: selectorElementGetter(
'a[role="menuitem"][href*="&tbm=shop"]',
),
navigateBooksTab: selectorElementGetter(
'a[role="menuitem"][href*="&tbm=bks"]',
),
navigateFlightsTab: selectorElementGetter(
'a[role="menuitem"][href*="&tbm=flm"]',
),
navigateFinancialTab: selectorElementGetter(
'a[role="menuitem"][href*="/finance?"]',
),
// TODO: Disable image search's default keybindings to avoid confusing the
// user, because the default keybindings can cause an indenepdent
// navigation of the image results with another outline. The code below
// doesn't work because the key event is captured by the image search
// code, since Moustrap is bound on document, instead of a more specific
// container. The following does work, but the code needs some changes to
// support binding on a specific container per search engine:
//
// Mousetrap(document.querySelector('.islrc')).bind ...
// Mousetrap(document.querySelector('#Sva75c')).bind ...
//
// navigatePreviousResultPage: null,
// navigateNextResultPage: null,
};
}
// Array storing tuples of tabs navigation keybindings and their corresponding
// CSS selector
get previousPageButton() {
if (GoogleSearch.#isImagesTab()) {
return null;
}
return selectorElementGetter('#pnprev');
}
get nextPageButton() {
if (GoogleSearch.#isImagesTab()) {
return null;
}
return selectorElementGetter('#pnnext');
}
get tabs() {
if (GoogleSearch.#isImagesTab()) {
return GoogleSearch.#imageSearchTabs();
}
return {
navigateSearchTab: selectorElementGetter(
// eslint-disable-next-line max-len
'a[href*="/search?q="]:not([href*="&tbm="]):not([href*="maps.google."])',
),
navigateImagesTab: selectorElementGetter('a[href*="&tbm=isch"]'),
navigateVideosTab: selectorElementGetter('a[href*="&tbm=vid"]'),
navigateMapsTab: selectorElementGetter('a[href*="maps.google."]'),
navigateNewsTab: selectorElementGetter('a[href*="&tbm=nws"]'),
navigateShoppingTab: selectorElementGetter('a[href*="&tbm=shop"]'),
navigateBooksTab: selectorElementGetter('a[href*="&tbm=bks"]'),
navigateFlightsTab: selectorElementGetter('a[href*="&tbm=flm"]'),
navigateFinancialTab: selectorElementGetter('[href*="/finance?"]'),
};
}
/**
* Filter the results based on special properties
* @param {*} period, filter identifier. Accepted filter are :
* 'a' : all results
* 'h' : last hour
* 'd' : last day
* 'w' : last week
* 'm' : last month
* 'y' : last year
* 'v' : verbatim search
* null : toggle sort
*/
// TODO: Refactor this function to get enums after migrating to typescript.
changeTools(period) {
const searchParams = new URLSearchParams(window.location.search);
// Use the last value of the tbs param in case there are multiple ones,
// since the last one overrides the previous ones.
const allTbsValues = searchParams.getAll('tbs');
const lastTbsValue = allTbsValues[allTbsValues.length - 1] || '';
const match = /(qdr:.|li:1)(,sbd:.)?/.exec(lastTbsValue);
const currentPeriod = (match && match[1]) || '';
const currentSort = (match && match[2]) || '';
if (period === 'a') {
searchParams.delete('tbs');
} else if (period) {
let newTbs = '';
if (period === 'v') {
if (currentPeriod === 'li:1') {
newTbs = '';
} else {
newTbs = 'li:1';
}
} else {
newTbs = `qdr:${period}`;
}
searchParams.set('tbs', `${newTbs}${currentSort}`);
// Can't apply sort when not using period.
} else if (currentPeriod) {
searchParams.set(
'tbs',
`${currentPeriod}` + (currentSort ? '' : ',sbd:1'),
);
}
const newSearchString = '?' + searchParams.toString();
if (newSearchString !== window.location.search) {
window.location.search = newSearchString;
}
return false;
}
changeImageSize(size) {
const sizeOptions = {
LARGE: {value: 0, name: 'Large', code: 'l'},
MEDIUM: {value: 1, name: 'Medium', code: 'e'},
ICON: {value: 2, name: 'Icon', code: 'i'},
};
const openTool = document.querySelector(
'[class="PNyWAd ZXJQ7c"][jsname="I4bIT"]',
);
if (openTool != null) {
openTool.click();
}
const openSizeDropDown = document.querySelector('[aria-label="Size"]');
if (openSizeDropDown != null) {
openSizeDropDown.click();
}
const dropDownWithSize = document.querySelector(
'[class="xFo9P r9PaP Fmo8N"][jsname="wLFV5d"]',
);
const getButton = (selector) => {
let button;
if (document.querySelector(selector) != null) {
button = document.querySelector(selector);
} else {
button = null;
}
return button;
};
const setImageSize = (dropDownWithSize, buttonSelector) => {
let button = getButton(buttonSelector);
if (dropDownWithSize == null && button != null) {
button.click();
} else if (dropDownWithSize != null && button == null) {
dropDownWithSize.click();
button = getButton(buttonSelector);
button.click();
} else if (dropDownWithSize != null && button != null) {
button.click();
}
};
switch (size) {
case sizeOptions.LARGE.code:
if (
dropDownWithSize == null ||
dropDownWithSize.getAttribute('aria-label') != sizeOptions.LARGE.name
) {
setImageSize(
dropDownWithSize,
'[class="MfLWbb"][aria-label="Large"]',
);
}
break;
case sizeOptions.MEDIUM.code:
if (
dropDownWithSize == null ||
dropDownWithSize.getAttribute('aria-label') != sizeOptions.MEDIUM.name
) {
setImageSize(
dropDownWithSize,
'[class="MfLWbb"][aria-label="Medium"]',
);
}
break;
case sizeOptions.ICON.code:
if (
dropDownWithSize == null ||
dropDownWithSize.getAttribute('aria-label') != sizeOptions.ICON.name
) {
setImageSize(dropDownWithSize, '[class="MfLWbb"][aria-label="Icon"]');
}
break;
default:
break;
}
}
}
class BraveSearch {
constructor(options) {
this.options = options;
}
get urlPattern() {
return /^https:\/\/search\.brave\.com/;
}
get searchBoxSelector() {
return '.form-input, input[id=searchbox]';
}
getTopMargin(element) {
return getFixedSearchBoxTopMargin(
document.querySelector('header.navbar'),
element,
);
}
onChangedResults(callback) {
const containers = document.querySelectorAll('#results');
const observer = new MutationObserver(
debounce((mutationsList, observer) => {
callback(true);
}, 50),
);
for (const container of containers) {
observer.observe(container, {
attributes: false,
childList: true,
subtree: true,
});
}
}
static #getNewsTabResults() {
const includedElements = [
{
nodes: document.querySelectorAll('.snippet a'),
highlightClass: 'wsn-brave-search-focused-news',
containerSelector: (n) => n.parentElement,
},
];
return getSortedSearchResults(includedElements);
}
static #getVideosTabResults() {
const includedElements = [
{
nodes: document.querySelectorAll('.card a'),
highlightClass: 'wsn-brave-search-focused-card',
highlightedElementSelector: (n) => n.closest('.card'),
containerSelector: (n) => n.parentElement,
},
];
return getSortedSearchResults(includedElements);
}
getSearchResults() {
if (BraveSearch.#isTabActive(this.tabs.navigateNewsTab)) {
return BraveSearch.#getNewsTabResults();
} else if (BraveSearch.#isTabActive(this.tabs.navigateVideosTab)) {
return BraveSearch.#getVideosTabResults();
}
const includedElements = [
{
nodes: document.querySelectorAll('.snippet.fdb > a'),
highlightClass: 'wsn-brave-search-focused-link',
containerSelector: (n) => n.parentElement,
},
// News cards
{
nodes: document.querySelectorAll(
'.card[data-type="news"]:nth-child(-n+3)',
),
highlightClass: 'wsn-brave-search-focused-card',
},
// Video cards
{
nodes: document.querySelectorAll(
'.card[data-type="videos"]:nth-child(-n+3)',
),
highlightClass: 'wsn-brave-search-focused-card',
},
];
return getSortedSearchResults(includedElements);
}
static #isTabActive(tab) {
return tab && tab.parentElement.classList.contains('active');
}
get tabs() {
return {
navigateSearchTab: document.querySelector('a[href*="/search?q="]'),
navigateImagesTab: document.querySelector(
'#tab-images > a:first-of-type',
),
navigateNewsTab: document.querySelector('a[href*="/news?q="]'),
navigateVideosTab: document.querySelector(
'#tab-videos > a:first-of-type',
),
};
}
}
class StartPage {
constructor(options) {
this.options = options;
}
get urlPattern() {
return /^https:\/\/(www\.)?startpage\./;
}
get searchBoxSelector() {
return '#q';
}
getTopMargin(element) {
return getFixedSearchBoxTopMargin(
document.querySelector('div.layout-web__header'),
element,
);
}
getBottomMargin(element) {
// Startpage in Firefox has an issue where trying to scroll can result in
// window.scrollY being updated for a brief time although no scrolling is
// done, which confuses the scrollToElement function, which can lead to
// being stuck focused on a search result.
return isFirefox() ? 0 : getDefaultBottomMargin();
}
static #isSearchTab() {
return document.querySelector('div.layout-web') != null;
}
static #isImagesTab() {
return document.querySelector('div.layout-images') != null;
}
getSearchResults() {
// Don't initialize results navigation on image search, since it doesn't
// work there.
if (StartPage.#isImagesTab()) {
return [];
}
const containerSelector = (element) => {
if (StartPage.#isSearchTab()) {
return element.closest('.w-gl__result');
}
return element;
};
const includedElements = [
{
nodes: document.querySelectorAll('a.w-gl__result-url'),
highlightedElementSelector: containerSelector,
highlightClass: 'wsn-startpage-focused-link',
containerSelector: containerSelector,
},
{
nodes: document.querySelectorAll('.pagination--desktop button'),
highlightClass: 'wsn-startpage-focused-link',
},
// As of 2020-06-20, this doesn't seem to match anything.
{
nodes: document.querySelectorAll(
'.vo-sp.vo-sp--default > a.vo-sp__link',
),
highlightedElementSelector: containerSelector,
highlightClass: 'wsn-startpage-focused-link',
},
];
const excludedElements = document.querySelectorAll('button[disabled]');
return getSortedSearchResults(includedElements, excludedElements);
}
get previousPageButton() {
const menuLinks = document.querySelectorAll('.inline-nav-menu__link');
if (!menuLinks || menuLinks.length < 4) {
return null;
}
return document.querySelector(
'form.pagination__form.next-prev-form--desktop:first-of-type',
);
}
get nextPageButton() {
const menuLinks = document.querySelectorAll('.inline-nav-menu__link');
if (!menuLinks || menuLinks.length < 4) {
return null;
}
return document.querySelector(
'form.pagination__form.next-prev-form--desktop:last-of-type',
);
}
get tabs() {
const menuLinks = document.querySelectorAll('.inline-nav-menu__link');
if (!menuLinks || menuLinks.length < 4) {
return {};
}
return {
navigateSearchTab: menuLinks[0],
navigateImagesTab: menuLinks[1],
navigateVideosTab: menuLinks[2],
navigateNewsTab: menuLinks[3],
};
}
changeTools(period) {
const forms = document.forms;
let timeForm;
for (let i = 0; i < forms.length; i++) {
if (forms[i].className === 'search-filter-time__form') {
timeForm = forms[i];
}
}
switch (period) {
case 'd':
timeForm.elements['with_date'][1].click();
break;
case 'w':
timeForm.elements['with_date'][2].click();
break;
case 'm':
timeForm.elements['with_date'][3].click();
break;
case 'y':
timeForm.elements['with_date'][4].click();
break;
default:
break;
}
}
}
class YouTube {
constructor(options) {
this.options = options;
this.gridNavigation = false;
}
get urlPattern() {
return /^https:\/\/(www)\.youtube\./;
}
get searchBoxSelector() {
return 'input#search';
}
getTopMargin(element) {
return getFixedSearchBoxTopMargin(
document.querySelector('#masthead-container'),
element,
);
}
onChangedResults(callback) {
// The ytd-section-list-renderer element may not exist yet when this code
// runs, so we look for changes in the higher level elements until we find
// ytd-section-list-renderer.
const YT_CONTAINER_SELECTOR = [
'ytd-section-list-renderer',
'.ytd-section-list-renderer',
'ytd-rich-grid-renderer',
'ytd-shelf-renderer',
].join(',');
const resultsObserver = new MutationObserver(
debounce((mutationsList, observer) => {
callback(true);
}, 50),
);
let lastLoadedURL = null;
const pageObserverCallback = (mutationsList, observer) => {
const url = window.location.pathname + window.location.search;
if (url === lastLoadedURL) {
return;
} else {
resultsObserver.disconnect();
}
const containers = document.querySelectorAll(YT_CONTAINER_SELECTOR);
if (containers.length == 0) {
return;
}
lastLoadedURL = url;
callback(false);
for (const container of containers) {
resultsObserver.observe(container, {
attributes: false,
childList: true,
subtree: true,
});
}
};
// TODO: the observer callback is triggered many times because of the broad
// changes that the observer tracks. I tried to use other observation specs
// to limit it, but then it failed to detect URL changes without page load
// (which is what happened in issue #337 [1]).
// [1] https://github.com/infokiller/web-search-navigator/issues/337
const pageObserver = new MutationObserver(
debounce(pageObserverCallback, 50),
);
pageObserver.observe(document.querySelector('#page-manager'), {
attributes: false,
childList: true,
subtree: true,
});
}
getSearchResults() {
const includedElements = [
// Videos in vertical search results: https://imgur.com/a/Z8KV5Oe
{
nodes: document.querySelectorAll('a#video-title.ytd-video-renderer'),
highlightClass: 'wsn-youtube-focused-video',
highlightedElementSelector: (n) => n.closest('ytd-video-renderer'),
containerSelector: (n) => n.closest('ytd-video-renderer'),
},
// Playlist results in vertical search results: https://imgur.com/a/nPjGd9H
{
nodes: document.querySelectorAll(
'ytd-playlist-renderer a[href*="/playlist"]',
),
highlightClass: 'wsn-youtube-focused-video',
highlightedElementSelector: (n) => n.closest('ytd-playlist-renderer'),
containerSelector: (n) => n.closest('ytd-playlist-renderer'),
},
// Playlists
{
nodes: document.querySelectorAll('a.ytd-playlist-video-renderer'),
highlightClass: 'wsn-youtube-focused-video',
highlightedElementSelector: (n) =>
n.closest('ytd-playlist-video-renderer'),
containerSelector: (n) => n.closest('ytd-playlist-video-renderer'),
},
// Mixes
{
nodes: document.querySelectorAll('div#content a.ytd-radio-renderer'),
highlightClass: 'wsn-youtube-focused-video',
},
// Channels
{
nodes: document.querySelectorAll(
'ytd-grid-video-renderer a#video-title:not([aria-hidden="true"])',
),
highlightClass: 'wsn-youtube-focused-grid-video',
highlightedElementSelector: (n) => n.closest('ytd-grid-video-renderer'),
containerSelector: (n) => n.closest('ytd-grid-video-renderer'),
},
];
// checking if homepage results are present
const homePageElements = {
nodes: document.querySelectorAll(
'ytd-rich-item-renderer a#video-title-link',
),
highlightClass: 'wsn-youtube-focused-video',
highlightedElementSelector: (n) => n.closest('ytd-rich-item-renderer'),
containerSelector: (n) => n.closest('ytd-rich-item-renderer'),
};
const results = getSortedSearchResults(
[...includedElements, homePageElements],
[],
);
// When navigating away from the home page, the home page elements are still
// in the DOM but they are not visible, so we must check if they are
// visible (using offsetParent), not just if they are present.
const isHomePage = Array.from(homePageElements.nodes).some(
(n) => n.offsetParent != null,
);
const gridRow = document.querySelector('ytd-rich-grid-row');
if (isHomePage && gridRow != null) {
results.itemsPerRow = gridRow.getElementsByTagName(
'ytd-rich-item-renderer',
).length;
results.gridNavigation = results.itemsPerRow > 0;
}
return results;
}
changeTools(period) {
if (!document.querySelector('div#collapse-content')) {
const toggleButton = document.querySelectorAll(
'a.ytd-toggle-button-renderer',
)[0];
// Toggling the buttons ensures that div#collapse-content is loaded
toggleButton.click();
toggleButton.click();
}
const forms = document.querySelectorAll(
'div#collapse-content > *:first-of-type ytd-search-filter-renderer',
);
let neededForm = null;
switch (period) {
case 'h':
neededForm = forms[0];
break;
case 'd':
neededForm = forms[1];
break;
case 'w':
neededForm = forms[2];
break;
case 'm':
neededForm = forms[3];
break;
case 'y':
neededForm = forms[4];
break;
}
if (neededForm) {
neededForm.childNodes[1].click();
}
}
}
class GoogleScholar {
constructor(options) {
this.options = options;
}
get urlPattern() {
return /^https:\/\/scholar\.google\./;
}
get searchBoxSelector() {
return '#gs_hdr_tsi';
}
getSearchResults() {
const includedElements = [
{
nodes: document.querySelectorAll('.gs_rt a'),
highlightClass: 'wsn-google-focused-link',
highlightedElementSelector: (n) => n.closest('.gs_rt'),
containerSelector: (n) => n.parentElement.parentElement,
},
{
nodes: document.querySelectorAll(
'.gs_ico_nav_previous, .gs_ico_nav_next',
),
anchorSelector: (n) => n.parentElement,
highlightClass: 'wsn-google-scholar-next-page',
highlightedElementSelector: (n) => n.parentElement.children[1],
containerSelector: (n) => n.parentElement.children[1],
},
];
return getSortedSearchResults(includedElements, []);
}
get previousPageButton() {
const previousPageElement = document.querySelector('.gs_ico_nav_previous');
if (previousPageElement !== null) {
return previousPageElement.parentElement;
}
return null;
}
get nextPageButton() {
const nextPageElement = document.querySelector('.gs_ico_nav_next');
if (nextPageElement !== null) {
return nextPageElement.parentElement;
}
return null;
}
}
class Amazon {
constructor(options) {
this.options = options;
}
get urlPattern() {
return /^https:\/\/(www\.)?amazon\./;
}
get searchBoxSelector() {
return '#twotabsearchtextbox';
}
onChangedResults(callback) {
const container = document.querySelector('.s-main-slot');
if (!container) {
return;
}
const observer = new MutationObserver(
debounce((mutationsList, observer) => {
callback(false);
}, 50),
);
observer.observe(container, {
attributes: false,
childList: true,
subtree: false,
});
}
getSearchResults() {
const includedElements = [
// Carousel items
{
nodes: document.querySelectorAll(
'.s-main-slot .a-carousel-card h2 .a-link-normal.a-text-normal',
),
highlightedElementSelector: (n) => n.closest('.a-carousel-card'),
highlightClass: 'wsn-amazon-focused-carousel-item',
containerSelector: (n) => n.closest('.a-carousel-card'),
},
// Regular items.
// NOTE: Must appear after the carousel items because this selector is
// more general.
{
nodes: document.querySelectorAll(
'.s-main-slot h2 .a-link-normal.a-text-normal',
),
// highlightedElementSelector: (n) => n.parentElement.children[1],
highlightedElementSelector: (n) =>
n.closest('.a-section').parentElement.closest('.a-section'),
highlightClass: 'wsn-amazon-focused-item',
containerSelector: (n) =>
n.closest('.a-section').parentElement.closest('.a-section'),
},
// Next/previous and page numbers.
{
nodes: document.querySelectorAll('a.s-pagination-item'),
highlightClass: 'wsn-amazon-focused-item',
},
// Shopping card items
{
nodes: document.querySelectorAll(
'.sc-list-item-content .a-list-item .a-link-normal',
),
highlightClass: 'wsn-amazon-focused-cart-item',
highlightedElementSelector: (n) => n.closest('.sc-list-item-content'),
containerSelector: (n) => n.closest('.sc-list-item-content'),
},
];
// Exclude active page number and hidden carousel elements.
// TODO: The hidden carousel elements do not match at page load because
// they don't yet have the aria-hidden property set.
const excludedElements = document.querySelectorAll(
'.a-pagination .a-selected a, .a-carousel-card[aria-hidden="true"] a',
);
return getSortedSearchResults(includedElements, excludedElements);
}
get previousPageButton() {
return document.querySelector('a.s-pagination-previous');
}
get nextPageButton() {
return document.querySelector('a.s-pagination-next');
}
}
class Github {
constructor(options) {
this.options = options;
}
get urlPattern() {
return /^https:\/\/(www\.)?github\.com/;
}
get searchBoxSelector() {
// TODO: With the escape key, this only works the first time the keybinding
// is used, Since Github seem to capture this as well, which causes it to
// leave the search box.
return 'input[name="q"]';
}
static #getCommitSearchLinks() {
const commitsContainers = document.querySelectorAll(
'#commit_search_results .text-normal',
);
const commits = [];
for (const con of commitsContainers) {
const links = con.querySelectorAll('a');
if (links.length === 0) {
continue;
}
if (links.length === 1) {
commits.push(links[0]);
} else {
const prLink = con.querySelector(
'a[data-hovercard-type="pull_request"]',
);
if (prLink != null) {
commits.push(prLink);
}
}
}
return commits;
}
getSearchResults() {
const includedElements = [
// Repos
{
nodes: document.querySelectorAll('.repo-list a'),
highlightClass: 'wsn-github-focused-item',
containerSelector: (n) => n.closest('.mt-n1'),
},
// Code
{
nodes: document.querySelectorAll('#code_search_results .text-normal a'),
highlightClass: 'wsn-github-focused-item',
},
// Commits/PRs
{
nodes: Github.#getCommitSearchLinks(),
highlightClass: 'wsn-github-focused-item',
},
// Issues
{
nodes: document.querySelectorAll(
'#issue_search_results .text-normal a',
),
highlightClass: 'wsn-github-focused-item',
},
// Marketplace
{
nodes: document.querySelectorAll(
'#marketplace_search_results .text-normal a',
),
highlightClass: 'wsn-github-focused-item',
},
// Topics
{
nodes: document.querySelectorAll(
'#topic_search_results .text-normal a',
),
highlightClass: 'wsn-github-focused-item',
},
// Wikis
{
nodes: document.querySelectorAll('#wiki_search_results .text-normal a'),
highlightClass: 'wsn-github-focused-item',
},
// Users
{
nodes: document.querySelectorAll('#user_search_results .text-normal a'),
highlightClass: 'wsn-github-focused-item',
},
// Pinned repos in user profile
{
nodes: document.querySelectorAll(
'.pinned-item-list-item-content span.repo',
),
highlightClass: 'wsn-github-focused-item',
highlightedElementSelector: (n) => n.closest('a'),
containerSelector: (n) => n.closest('a'),
anchorSelector: (n) => n.closest('a'),
},
// Personal repos list in user profile
{
nodes: document.querySelectorAll(
'#user-repositories-list a[itemprop*="codeRepository"]',
),
highlightClass: 'wsn-github-focused-item',
containerSelector: (n) => n.closest('li') || n,
},
// Next/previous and page numbers.
{
nodes: document.querySelectorAll('.paginate-container a'),
highlightClass: 'wsn-github-focused-pagination',
},
];
const searchParams = new URLSearchParams(window.location.search);
// Starred repos of user
if (searchParams.get('tab') === 'stars') {
includedElements.push({
nodes: document.querySelectorAll('h3 a'),
highlightClass: 'wsn-github-focused-item',
});
}
const excludedElements = [
// Exclude small links
...document.querySelectorAll('.muted-link, .Link--muted'),
// Exclude topic tags
...document.querySelectorAll('.topic-tag'),
// Exclude small links in commits
// ...document.querySelectorAll(
// '#commit_search_results .text-normal a.message'),
];
return getSortedSearchResults(includedElements, excludedElements);
}
onChangedResults(callback) {
const container = document.querySelector('body');
if (!container) {
return;
}
// Store the last URL to detect page navigations (for example going to the
// next page of results).
let lastURL = window.location.href;
const observer = new MutationObserver(
debounce((mutationsList, observer) => {
let appendOnly = true;
if (window.location.href !== lastURL) {
lastURL = window.location.href;
appendOnly = false;
}
callback(appendOnly);
}, 50),
);
observer.observe(container, {
attributes: false,
childList: true,
subtree: false,
});
}
// Github already has built-in support for tabs:
// https://docs.github.com/en/github/getting-started-with-github/keyboard-shortcuts
get tabs() {
return {};
}
}
class Gitlab {
constructor(options) {
this.options = options;
}
get urlPattern() {
return /^https:\/\/(www\.)?gitlab\.com/;
}
get searchBoxSelector() {
return '.form-input, input[id=search]';
}
getTopMargin(element) {
return getFixedSearchBoxTopMargin(
document.querySelector('header.navbar'),
element,
);
}
onChangedResults(callback) {
const containers = document.querySelectorAll(
'.projects-list, .groups-list, #content-body',
);
const observer = new MutationObserver(async (mutationsList, observer) => {
callback(true);
});
for (const container of containers) {
observer.observe(container, {
attributes: false,
childList: true,
subtree: true,
});
}
}
getSearchResults() {
const includedElements = [
{
nodes: document.querySelectorAll('li.project-row h2 a'),
containerSelector: (n) => n.closest('li.project-row'),
highlightedElementSelector: (n) => n.closest('li.project-row'),
highlightClass: 'wsn-gitlab-focused-group-row',
},
// Org subgroups, for example:
// https://gitlab.archlinux.org/archlinux
{
nodes: document.querySelectorAll(
'ul.groups-list li.group-row a[aria-label]',
),
containerSelector: (n) => n.closest('li.group-row'),
highlightedElementSelector: (n) => n.closest('li.group-row'),
highlightClass: 'wsn-gitlab-focused-group-row',
},
// Prev/next page
{
nodes: document.querySelectorAll('li.page-item a.page-link'),
containerSelector: (n) => n.closest('li.page-item'),
highlightedElementSelector: (n) => n.closest('li.group-row'),
highlightClass: 'wsn-gitlab-focused-group-row',
},
];
return getSortedSearchResults(includedElements);
}
}
class CustomGitlab extends Gitlab {
get urlPattern() {
return new RegExp(this.options.customGitlabUrl);
}
}
// Get search engine object matching the current url
/* eslint-disable-next-line no-unused-vars */
const getSearchEngine = (options) => {
const searchEngines = [
new GoogleSearch(options),
new BraveSearch(options),
new StartPage(options),
new YouTube(options),
new GoogleScholar(options),
new Amazon(options),
new Github(options),
new Gitlab(options),
new CustomGitlab(options),
];
// Switch over all compatible search engines
const href = window.location.href;
for (let i = 0; i < searchEngines.length; i++) {
if (href.match(searchEngines[i].urlPattern)) {
return searchEngines[i];
}
}
return null;
};
/* global ExtensionOptions, getSearchEngine, Mousetrap */
/* global getDefaultBottomMargin */
// TODO: Replace with enums when switching to typescript.
const FOCUS_SCROLL_OFF = 0;
const FOCUS_SCROLL_ON = 1;
const FOCUS_SCROLL_ONLY = 2;
// Returns true if scrolling was done.
const scrollToElement = (searchEngine, element) => {
if (element == null) {
console.error('Cannot scroll to null element');
return;
}
let topMargin = 0;
if (searchEngine.getTopMargin) {
topMargin = searchEngine.getTopMargin(element);
}
let bottomMargin = getDefaultBottomMargin();
if (searchEngine.getBottomMargin) {
bottomMargin = searchEngine.getBottomMargin(element);
}
const elementBounds = element.getBoundingClientRect();
const scrollY = window.scrollY;
if (elementBounds.top < topMargin) {
// scroll element to top
element.scrollIntoView(true);
window.scrollBy(0, -topMargin);
} else if (elementBounds.bottom + bottomMargin > window.innerHeight) {
// scroll element to bottom
element.scrollIntoView(false);
window.scrollBy(0, bottomMargin);
}
return Math.abs(window.scrollY - scrollY) > 0.01;
};
const bindKeys = (bindings, toggle) => {
// NOTE: Mousetrap calls the handler even if there's a larger sequence that
// ends with the same key. For example, if the user binds both 'a b' and
// 'b', when pressing the sequence 'a b' both handlers will be called, which
// is not the desired behavior for this extension. Therefore, we first sort
// all keybindings by their sequence length, so that handlers of larger
// sequences will be called before the shorter ones. Then, we only call
// other handlers if the previous handler returned true.
bindings.sort((a, b) => {
return b[0].split(' ').length - a[0].split(' ').length;
});
let lastEvent;
let lastHandlerResult;
for (const [shortcut, element, global, callback] of bindings) {
const wrappedCallback = (event) => {
if (!toggle['active']) {
return true;
}
if (event === lastEvent && !lastHandlerResult) {
return;
}
lastEvent = event;
lastHandlerResult = callback(event);
return lastHandlerResult;
};
if (global) {
/* eslint-disable-next-line new-cap */
Mousetrap(element).bindGlobal(shortcut, wrappedCallback);
} else {
/* eslint-disable-next-line new-cap */
Mousetrap(element).bind(shortcut, wrappedCallback);
}
}
};
class SearchResultsManager {
constructor(searchEngine, options) {
this.searchEngine = searchEngine;
this.options = options;
this.focusedIndex = -1;
this.isInitialFocusSet = false;
}
reloadSearchResults() {
this.searchResults = this.searchEngine.getSearchResults();
if (!this.isInitialFocusSet) {
this.setInitialFocus();
}
}
setInitialFocus() {
if (this.searchResults.length === 0) {
return;
}
const lastNavigation = this.options.local.values;
if (
location.href === lastNavigation.lastQueryUrl &&
lastNavigation.lastFocusedIndex >= 0 &&
lastNavigation.lastFocusedIndex < this.searchResults.length
) {
this.focus(lastNavigation.lastFocusedIndex, FOCUS_SCROLL_ON);
} else if (this.options.sync.get('autoSelectFirst')) {
// Highlight the first result when the page is loaded, but don't scroll to
// it because there may be KP cards such as stock graphs.
this.focus(0, FOCUS_SCROLL_OFF);
}
}
/**
* Returns the element to click on upon navigation. The focused element in the
* document is preferred (if there is one) over the highlighted result. Note
* that the focused element does not have to be an anchor <a> element.
*
* @param {boolean} linkOnly If true the focused element is preferred only
* when it is a link with "href" attribute.
* @return {Element}
*/
getElementToNavigate(linkOnly = false) {
const focusedElement = document.activeElement;
// StartPage seems to still focus and change it to body when the page loads.
if (focusedElement == null || focusedElement.localName === 'body') {
if (
this.focusedIndex < 0 ||
this.focusedIndex >= this.searchResults.length
) {
return null;
}
return this.searchResults[this.focusedIndex].anchor;
}
const isLink =
focusedElement.localName === 'a' && focusedElement.hasAttribute('href');
if (!linkOnly || isLink) {
return focusedElement;
}
}
highlight(searchResult) {
const highlighted = searchResult.highlightedElement;
if (highlighted == null) {
console.error('No element to highlight: %o', highlighted);
return;
}
highlighted.classList.add(searchResult.highlightClass);
if (
this.options.sync.get('hideOutline') ||
searchResult.anchor !== highlighted
) {
searchResult.anchor.classList.add('wsn-no-outline');
}
}
unhighlight(searchResult) {
const highlighted = searchResult.highlightedElement;
if (highlighted == null) {
console.error('No element to unhighlight: %o', highlighted);
return;
}
highlighted.classList.remove(searchResult.highlightClass);
highlighted.classList.remove('wsn-no-outline');
}
focus(index, scroll = FOCUS_SCROLL_ONLY) {
if (this.focusedIndex >= 0) {
const searchResult = this.searchResults[this.focusedIndex];
// If the current result is outside the viewport and FOCUS_SCROLL_ONLY was
// requested, scroll to the current hidden result, but don't focus on the
// new result.
// This behavior is intended to handle cases where the user scrolls away
// from the currently focused result and then presses the keybindings to
// focus on the previous/next result. In this case, since the user
// doesn't see the current result, it's more intuitive to only scroll to
// the current result, and then on the next keypress they can focus on the
// previous/next result and actually see on what result they want to focus
// on.
if (
scroll === FOCUS_SCROLL_ONLY &&
scrollToElement(this.searchEngine, searchResult.container)
) {
return;
}
// Remove highlighting from previous item.
this.unhighlight(searchResult);
}
const searchResult = this.searchResults[index];
if (!searchResult) {
this.focusedIndex = -1;
return;
}
this.highlight(searchResult);
// We already scroll below, so no need for focus to scroll. The scrolling
// behavior of `focus` also seems less predictable and caused an issue, see:
// https://github.com/infokiller/web-search-navigator/issues/35
searchResult.anchor.focus({preventScroll: true});
// Ensure whole search result container is visible in the viewport, not only
// the search result link.
if (scroll !== FOCUS_SCROLL_OFF) {
scrollToElement(this.searchEngine, searchResult.container);
}
this.focusedIndex = index;
this.isInitialFocusSet = true;
}
focusNext(shouldWrap) {
if (this.focusedIndex < this.searchResults.length - 1) {
this.focus(this.focusedIndex + 1);
} else if (shouldWrap) {
this.focus(0);
}
}
focusPrevious(shouldWrap) {
if (this.focusedIndex > 0) {
this.focus(this.focusedIndex - 1);
} else if (shouldWrap) {
this.focus(this.searchResults.length - 1);
} else {
window.scrollTo(window.scrollX, 0);
}
}
focusDown(shouldWrap) {
if (
this.focusedIndex + this.searchResults.itemsPerRow <
this.searchResults.length
) {
this.focus(this.focusedIndex + this.searchResults.itemsPerRow);
} else if (shouldWrap) {
const focusedRowIndex =
this.focusedIndex % this.searchResults.itemsPerRow;
this.focus(focusedRowIndex);
}
}
focusUp(shouldWrap) {
if (this.focusedIndex - this.searchResults.itemsPerRow >= 0) {
this.focus(this.focusedIndex - this.searchResults.itemsPerRow);
} else if (shouldWrap) {
const focusedRowIndex =
this.focusedIndex % this.searchResults.itemsPerRow;
this.focus(
this.searchResults -
1 -
this.searchResults.itemsPerRow +
focusedRowIndex,
);
} else {
window.scrollTo(window.scrollY, 0);
}
}
}
class WebSearchNavigator {
constructor() {
this.bindings = [];
this.bindingsToggle = {active: true};
}
async init() {
this.options = new ExtensionOptions();
await this.options.load();
this.searchEngine = await getSearchEngine(this.options.sync.getAll());
if (this.searchEngine == null) {
return;
}
const sleep = (milliseconds) => {
return new Promise((resolve) => setTimeout(resolve, milliseconds));
};
await sleep(this.options.sync.get('delay'));
this.injectCSS();
this.initKeybindings();
}
injectCSS() {
const style = document.createElement('style');
style.textContent = this.options.sync.get('customCSS');
document.head.append(style);
}
initKeybindings() {
this.bindingsToggle['active'] = false;
for (const [shortcut, element, ,] of this.bindings) {
/* eslint-disable-next-line new-cap */
const ms = Mousetrap(element);
ms.unbind(shortcut);
ms.reset();
}
const isFirstCall = this.bindings.length === 0;
this.bindings = [];
// UGLY WORKAROUND: Results navigation breaks YouTube space keybinding for
// pausing/resuming a video. A workaround is to click on an element on the
// page (except the video), but for now I'm disabling results navigation
// when watching a video.
// TODO: Find a proper fix.
if (!window.location.href.match(/^https:\/\/(www)\.youtube\.com\/watch/)) {
this.initResultsNavigation(isFirstCall);
}
this.initTabsNavigation();
this.initChangeToolsNavigation();
this.initSearchInputNavigation();
this.bindingsToggle = {active: true};
bindKeys(this.bindings, this.bindingsToggle);
}
initSearchInputNavigation() {
let searchInput = document.querySelector(
this.searchEngine.searchBoxSelector,
);
if (searchInput == null) {
return;
}
// Only apply the extension logic if the key is not something the user may
// have wanted to type into the searchbox, so that we don't interfere with
// regular typing.
const shouldHandleSearchInputKey = (event) => {
return event.ctrlKey || event.metaKey || event.key === 'Escape';
};
// In Github, the search input element changes while in the page, so we
// redetect it if it's not visible.
const detectSearchInput = () => {
if (searchInput != null && searchInput.offsetParent != null) {
return true;
}
searchInput = document.querySelector(this.searchEngine.searchBoxSelector);
return searchInput != null && searchInput.offsetParent != null;
};
// If insideSearchboxHandler returns true, outsideSearchboxHandler will also
// be called (because it's defined on document, hence has lower priority),
// in which case we don't want to handle the event. Therefore, we store the
// last event handled in insideSearchboxHandler, and only handle the event
// in outsideSearchboxHandler if it's not the same one.
let lastEvent;
const outsideSearchboxHandler = (event) => {
if (!detectSearchInput()) {
return;
}
if (event === lastEvent) {
return !shouldHandleSearchInputKey(event);
}
const element = document.activeElement;
if (
element.isContentEditable ||
['textarea', 'input'].includes(element.tagName.toLowerCase())
) {
return true;
}
// Scroll to the search box in case it's outside the viewport so that it's
// clear to the user that it has focus.
scrollToElement(this.searchEngine, searchInput);
searchInput.select();
// searchInput.click();
return false;
};
const insideSearchboxHandler = (event) => {
if (!detectSearchInput()) {
return;
}
lastEvent = event;
if (!shouldHandleSearchInputKey(event)) {
return true;
}
// Everything is selected; deselect all.
if (
searchInput.selectionStart === 0 &&
searchInput.selectionEnd === searchInput.value.length
) {
// Scroll to the search box in case it's outside the viewport so that
// it's clear to the user that it has focus.
scrollToElement(this.searchEngine, searchInput);
searchInput.setSelectionRange(
searchInput.value.length,
searchInput.value.length,
);
return false;
}
// Closing search suggestions via document.body.click() or
// searchInput.blur() breaks the state of google's controller.
// The suggestion box is closed, yet it won't re-appear on the next
// search box focus event.
// Input can be blurred only when the suggestion box is already
// closed, hence the blur event is queued.
window.setTimeout(() => searchInput.blur());
// Invoke the default handler which will close-up search suggestions
// properly (google's controller won't break), but it won't remove the
// focus.
return true;
};
this.register(
this.options.sync.get('focusSearchInput'),
outsideSearchboxHandler,
);
// Bind globally, otherwise Mousetrap ignores keypresses inside inputs.
// We must bind it separately to the search box element, or otherwise the
// key event won't always be captured (for example this is the case on
// Google Search as of 2020-06-22), presumably because the javascript in the
// page will disable further processing.
this.register(
this.options.sync.get('focusSearchInput'),
insideSearchboxHandler,
searchInput,
true,
);
}
registerObject(obj) {
for (const [optionName, elementOrGetter] of Object.entries(obj)) {
this.register(this.options.sync.get(optionName), () => {
if (elementOrGetter == null) {
return true;
}
let element;
if (elementOrGetter instanceof HTMLElement) {
element = elementOrGetter;
} else {
element = elementOrGetter();
}
if (element == null) {
return true;
}
// Some search engines use forms instead of links for navigation
if (element.tagName == 'FORM') {
element.submit();
} else {
element.click();
}
return false;
});
}
}
initTabsNavigation() {
const tabs = this.searchEngine.tabs || {};
this.registerObject(tabs);
}
initResultsNavigation(isFirstCall) {
this.registerObject({
navigatePreviousResultPage: this.searchEngine.previousPageButton,
navigateNextResultPage: this.searchEngine.nextPageButton,
});
this.resetResultsManager();
let gridNavigation = this.resultsManager.searchResults.gridNavigation;
this.registerResultsNavigationKeybindings(gridNavigation);
// NOTE: we must not call onChangedResults multiple times, otherwise the
// URL change detection logic (which exists in YouTube) will break.
if (!isFirstCall || !this.searchEngine.onChangedResults) {
return;
}
this.searchEngine.onChangedResults((appendedOnly) => {
if (appendedOnly) {
this.resultsManager.reloadSearchResults();
} else {
this.resetResultsManager();
}
// In YouTube, the initial load does not always detect the grid navigation
// (because it can happen before results are actually loaded to the page).
// In this case, we must rebind the navigation keys after the results are
// loaded.
if (gridNavigation != this.resultsManager.searchResults.gridNavigation) {
gridNavigation = this.resultsManager.searchResults.gridNavigation;
this.initKeybindings();
}
});
}
resetResultsManager() {
if (this.resultsManager != null && this.resultsManager.focusedIndex >= 0) {
const searchResult =
this.resultsManager.searchResults[this.resultsManager.focusedIndex];
// NOTE: it seems that search results can become undefined when the DOM
// elements are removed (for example when the results change).
if (searchResult != null) {
this.resultsManager.unhighlight(searchResult);
}
}
this.resultsManager = new SearchResultsManager(
this.searchEngine,
this.options,
);
this.resultsManager.reloadSearchResults();
}
registerResultsNavigationKeybindings(gridNavigation) {
const getOpt = (key) => {
return this.options.sync.get(key);
};
const onFocusChange = (callback) => {
return () => {
if (!this.resultsManager.isInitialFocusSet) {
this.resultsManager.focus(0);
} else {
const _callback = callback.bind(this.resultsManager);
_callback(getOpt('wrapNavigation'));
}
return false;
};
};
if (!gridNavigation) {
this.register(
getOpt('nextKey'),
onFocusChange(this.resultsManager.focusNext),
);
this.register(
getOpt('previousKey'),
onFocusChange(this.resultsManager.focusPrevious),
);
} else {
this.register(
getOpt('nextKey'),
onFocusChange(this.resultsManager.focusDown),
);
this.register(
getOpt('previousKey'),
onFocusChange(this.resultsManager.focusUp),
);
// Left
this.register(
getOpt('navigatePreviousResultPage'),
onFocusChange(this.resultsManager.focusPrevious),
);
// Right
this.register(
getOpt('navigateNextResultPage'),
onFocusChange(this.resultsManager.focusNext),
);
}
this.register(getOpt('navigateKey'), () => {
const link = this.resultsManager.getElementToNavigate();
if (link == null) {
return true;
}
const lastNavigation = this.options.local.values;
lastNavigation.lastQueryUrl = location.href;
lastNavigation.lastFocusedIndex = this.resultsManager.focusedIndex;
this.options.local.save();
// If the element is a link, use the href to directly navigate, since some
// websites will open it in a new tab.
if (link.localName === 'a' && link.href) {
window.location.href = link.href;
} else {
link.click();
}
return false;
});
this.register(getOpt('navigateNewTabKey'), () => {
const link = this.resultsManager.getElementToNavigate(true);
if (link == null) {
return true;
}
browser.runtime.sendMessage({
type: 'tabsCreate',
options: {
url: link.href,
active: true,
},
});
return false;
});
this.register(getOpt('navigateNewTabBackgroundKey'), () => {
const link = this.resultsManager.getElementToNavigate(true);
if (link == null) {
return true;
}
if (getOpt('simulateMiddleClick')) {
const mouseEventParams = {
bubbles: true,
cancelable: false,
view: window,
button: 1,
which: 2,
buttons: 0,
clientX: link.getBoundingClientRect().x,
clientY: link.getBoundingClientRect().y,
};
const middleClickMousedown = new MouseEvent(
'mousedown',
mouseEventParams,
);
link.dispatchEvent(middleClickMousedown);
const middleClickMouseup = new MouseEvent('mouseup', mouseEventParams);
link.dispatchEvent(middleClickMouseup);
}
browser.runtime.sendMessage({
type: 'tabsCreate',
options: {
url: link.href,
active: false,
},
});
return false;
});
this.register(getOpt('copyUrlKey'), () => {
const link = this.resultsManager.getElementToNavigate();
if (
link == null || link.localName !== 'a' || !link.href ||
!navigator.clipboard
) {
return true;
}
navigator.clipboard.writeText(link.href).then(
() => false,
(err) => true,
);
});
}
initChangeToolsNavigation() {
if (this.searchEngine.changeTools == null) {
return;
}
const getOpt = (key) => {
return this.options.sync.get(key);
};
this.register(getOpt('navigateShowAll'), () =>
this.searchEngine.changeTools('a'),
);
this.register(getOpt('navigateShowHour'), () =>
this.searchEngine.changeTools('h'),
);
this.register(getOpt('navigateShowDay'), () =>
this.searchEngine.changeTools('d'),
);
this.register(getOpt('navigateShowWeek'), () =>
this.searchEngine.changeTools('w'),
);
this.register(getOpt('navigateShowMonth'), () =>
this.searchEngine.changeTools('m'),
);
this.register(getOpt('navigateShowYear'), () =>
this.searchEngine.changeTools('y'),
);
this.register(getOpt('toggleVerbatimSearch'), () =>
this.searchEngine.changeTools('v'),
);
this.register(getOpt('toggleSort'), () =>
this.searchEngine.changeTools(null),
);
this.register(getOpt('showImagesLarge'), () =>
this.searchEngine.changeImageSize('l'),
);
this.register(getOpt('showImagesMedium'), () =>
this.searchEngine.changeImageSize('e'),
);
this.register(getOpt('showImagesIcon'), () =>
this.searchEngine.changeImageSize('i'),
);
}
register(shortcuts, callback, element = document, global = false) {
for (const shortcut of shortcuts) {
this.bindings.push([shortcut, element, global, callback]);
}
}
}
const extension = new WebSearchNavigator();
extension.init();
// Some weird escaping things going on
const NEWLINE = String.fromCharCode(10);
const OPTIONS_HTML = atob(
`
<!DOCTYPE html>
<html>

<head>
    <title>Options for Web Search Navigator</title>
    <link rel="stylesheet" href="options_page.css">
</head>

<body>
    <section id="general-settings-container">
        <h2>General settings</h2>
        <div class="option">
            <label for="wrap-navigation">
                <input type="checkbox" id="wrap-navigation"> Wrap around when navigating before/after the first/last search result
            </label>
        </div>
        <div class="option">
            <label for="auto-select-first">
                <input type="checkbox" id="auto-select-first"> Focus on first search result automatically after the page loads
            </label>
        </div>
    </section>
    <section id="google-settings-container">
        <h2>Google specific settings</h2>
        <div class="option">
            <label for="google-include-cards">
                <input type="checkbox" id="google-include-cards"> Include cards (top stories, twitter, videos) in regular Google search page
            </label>
        </div>
        <div class="option">
            <label for="google-include-places">
                <input type="checkbox" id="google-include-places"> Include Places in regular Google search page
            </label>
        </div>
        <div class="option">
            <label for="google-include-memex">
                <input type="checkbox" id="google-include-memex"> Include WorldBrain's Memex extension results in Google search page
            </label>
        </div>
    </section>
    <section id="keybindings-container">
        <h2>Keybindings</h2>
        <details>
            <summary><h3>Help</h3></summary>
            <div class="help">
                All keybindings should be specified in
                <a href="https://github.com/ccampbell/mousetrap" target="_blank">Mousetrap</a> format. Examples:
                <ul>
                    <li>
                        <kbd class="keybinding">a</kbd>
                    </li>
                    <li>
                        <kbd class="keybinding">z y</kbd>
                    </li>
                    <li>
                        <kbd class="keybinding">ctrl+a</kbd>
                    </li>
                    <li>
                        <kbd class="keybinding">command+a</kbd>
                    </li>
                    <li>
                        <kbd class="keybinding">a, ctrl+b, z y, command+c</kbd> - multiple shortcuts that will be treated equivalently</li>
                </ul>
                Special keys names: backspace, tab, clear, enter, return, esc, escape, space, up, down, left, right, home, end, pageup, pagedown,
                del, delete, and f1 through f19. In order to disable a keybinding, delete its keybinding in the textbox.

                Note that not all search engines support all the keybindings.
            </div>
        </details>
        <details>
            <summary><h3>Common actions</h3></summary>
            <div class="option">
                <label for="next-key" class="option-desc">Next search result</label>
                <input id="next-key" class="input-keybinding" type="text" value="down, j">
            </div>
            <div class="option">
                <label for="previous-key" class="option-desc">Previous search result</label>
                <input id="previous-key" class="input-keybinding" type="text" value="up, k">
            </div>
            <div class="option">
                <label for="navigate-key" class="option-desc">Open</label>
                <input id="navigate-key" class="input-keybinding" type="text" value="return, space">
            </div>
            <div class="option">
                <label for="navigate-new-tab-background-key" class="option-desc">Open in a new background tab</label>
                <input id="navigate-new-tab-background-key" class="input-keybinding" type="text" value="ctrl+shift+return, command+shift+return, ctrl+shift+space">
            </div>
            <div class="option">
                <label for="navigate-new-tab-key" class="option-desc">Open in a new window/tab</label>
                <input id="navigate-new-tab-key" class="input-keybinding" type="text" value="ctrl+return, command+return, ctrl+space">
            </div>
            <div class="option">
                <label for="focus-search-input" class="option-desc">Focus search box</label>
                <input id="focus-search-input" class="input-keybinding" type="text" value="/, escape">
            </div>
            <div class="option">
                <label for="navigate-next-result-page" class="option-desc">Next page</label>
                <input id="navigate-next-result-page" class="input-keybinding" type="text" value="right">
            </div>
            <div class="option">
                <label for="navigate-previous-result-page" class="option-desc">Previous page</label>
                <input id="navigate-previous-result-page" class="input-keybinding" type="text" value="left">
            </div>
            <div class="option">
              <label for="copy-url-key" class="option-desc">Copy URL of focus</label>
              <input id="copy-url-key" class="input-keybinding" type="text" value="">
            </div>
        </details>
        <details>
            <summary><h3>Results filtering</h3></summary>
            <div class="option">
                <label for="navigate-show-all" class="option-desc">Turn off filter (show all results)</label>
                <input id="navigate-show-all" class="input-keybinding" type="text" value="z z, ctrl-shift-a">
            </div>
            <div class="option">
                <label for="navigate-show-hour" class="option-desc">Filter results by past hour</label>
                <input id="navigate-show-hour" class="input-keybinding" type="text" value="z h, ctrl-shift-h">
            </div>
            <div class="option">
                <label for="navigate-show-day" class="option-desc">Filter results by past 24 hours</label>
                <input id="navigate-show-day" class="input-keybinding" type="text" value="z d, ctrl-shift-d">
            </div>
            <div class="option">
                <label for="navigate-show-week" class="option-desc">Filter results by past week</label>
                <input id="navigate-show-week" class="input-keybinding" type="text" value="z w, ctrl-shift-w">
            </div>
            <div class="option">
                <label for="navigate-show-month" class="option-desc">Filter results by past month</label>
                <input id="navigate-show-month" class="input-keybinding" type="text" value="z m, ctrl-shift-m">
            </div>
            <div class="option">
                <label for="navigate-show-year" class="option-desc">Filter results by past year</label>
                <input id="navigate-show-year" class="input-keybinding" type="text" value="z y, ctrl-shift-y">
            </div>
            <div class="option">
                <label for="toggle-sort" class="option-desc">Toggle sort by date/relevance</label>
                <input id="toggle-sort" class="input-keybinding" type="text" value="z s, ctrl-shift-s">
            </div>
            <div class="option">
                <label for="toggle-verbatim-search" class="option-desc">Toggle verbatim search</label>
                <input id="toggle-verbatim-search" class="input-keybinding" type="text" value="z v, ctrl-shift-v">
            </div>
            <div class="option">
                <label for="show-images-large" class="option-desc">Filter image results by large size</label>
                <input id="show-images-large" class="input-keybinding" type="text" value="z l">
            </div>
            <div class="option">
                <label for="show-images-medium" class="option-desc">Filter image results by medium size</label>
                <input id="show-images-medium" class="input-keybinding" type="text" value="z e">
            </div>
            <div class="option">
                <label for="show-images-icon" class="option-desc">Filter image results by icon size</label>
                <input id="show-images-icon" class="input-keybinding" type="text" value="z i">
            </div>
        </details>
        <details>
            <summary><h3>Google and Startpage</h3></summary>
            <div class="option">
                <label for="navigate-search-tab" class="option-desc">Go to All (= default search tab)</label>
                <input id="navigate-search-tab" class="input-keybinding" type="text" value="a, s">
            </div>
            <div class="option">
                <label for="navigate-images-tab" class="option-desc">Go to Images</label>
                <input id="navigate-images-tab" class="input-keybinding" type="text" value="i">
            </div>
            <div class="option">
                <label for="navigate-videos-tab" class="option-desc">Go to Videos</label>
                <input id="navigate-videos-tab" class="input-keybinding" type="text" value="v">
            </div>
            <div class="option">
                <label for="navigate-maps-tab" class="option-desc">Go to Maps</label>
                <input id="navigate-maps-tab" class="input-keybinding" type="text" value="m">
            </div>
            <div class="option">
                <label for="navigate-news-tab" class="option-desc">Go to News</label>
                <input id="navigate-news-tab" class="input-keybinding" type="text" value="n">
            </div>
            <div class="option">
                <label for="navigate-shopping-tab" class="option-desc">Go to Shopping</label>
                <input id="navigate-shopping-tab" class="input-keybinding" type="text" value="alt+n">
            </div>
            <div class="option">
                <label for="navigate-books-tab" class="option-desc">Go to Books</label>
                <input id="navigate-books-tab" class="input-keybinding" type="text" value="b">
            </div>
            <div class="option">
                <label for="navigate-flights-tab" class="option-desc">Go to Flights</label>
                <input id="navigate-flights-tab" class="input-keybinding" type="text" value="alt+l">
            </div>
            <div class="option">
                <label for="navigate-financial-tab" class="option-desc">Go to Financial</label>
                <input id="navigate-financial-tab" class="input-keybinding" type="text" value="f">
            </div>
        </details>
    </section>
    <section id="search-engines-container">
        <h2>EXPERIMENTAL: Alternative search engines</h2>
        <details class="help">
            <summary><h3>Help</h3></summary>
            There is experimental support for using this extension in the websites below.
            Note that some features are still buggy in certain websites.
            You can enable or disable the extension of these websites at any time by clicking on the checkboxes.
            When you enable a website, the browser will prompt you for additional permissions which are needed to be able to run this extension on that website.
        </details>
        <div class="option">
            <label for="brave-search">
                <input type="checkbox" id="brave-search"> Enable on Brave Search
            </label>
        </div>
        <div class="option">
            <label for="startpage">
                <input type="checkbox" id="startpage"> Enable on Startpage
            </label>
        </div>
        <div class="option">
            <label for="youtube">
                <input type="checkbox" id="youtube"> Enable on YouTube
            </label>
        </div>
        <div class="option">
            <label for="google-scholar">
                <input type="checkbox" id="google-scholar"> Enable on Google Scholar
            </label>
        </div>
        <div class="option">
            <label for="amazon">
                <input type="checkbox" id="amazon"> Enable on Amazon
            </label>
        </div>
        <div class="option">
            <label for="github">
                <input type="checkbox" id="github"> Enable on Github
            </label>
        </div>
        <div class="option">
            <label for="gitlab">
                <input type="checkbox" id="gitlab"> Enable on Gitlab
            </label>
        </div>
        <div class="option">
            <label for="custom-gitlab">
                <input type="checkbox" id="custom-gitlab"> Enable on custom Gitlab
            </label>
        </div>
    </section>
    <section id="appearance-container">
        <h2>Appearance</h2>
        <div class="option">
            <label for="hide-outline">
                <input type="checkbox" id="hide-outline"> Hide outline on selected search result
            </label>
        </div>
        <div class="option">
            <h3>EXPERIMENTAL: Custom CSS</h3>
            You can set custom CSS rules to change how the focused search results are highlighted. The textarea below contains the default CSS rules.
            If you want to reset the CSS to the defaults, set the textarea content to an empty string and save.
            <details>
                <summary><h3>Edit CSS rules</h3></summary>
                <textarea name="custom-css-textarea" id="custom-css-textarea"></textarea>
            </details>
        </div>
    </section>
    <section id="advanced-settings-container">
        <details>
            <summary><h2>Advanced</h2></summary>
            <div class="option">
                <div class="help">
                    This option can be used as a workaround for some websites.
                </div>
                <div id="delay-container">
                    <label for="delay">
                        <input type="number" id="delay">Delay extension initialization in milliseconds
                    </label>
                </div>
            </div>
            <div class="option">
                <label for="simulate-middle-click">
                    <input type="checkbox" id="simulate-middle-click"> Simulate middle click when opening in a new background tab
                </label>
            </div>
            <div class="option">
                <h3>Custom Gitlab URL regex</h3>
                <div class="help">
                    Define private Gitlab URL regex. Default is ^https://(www\.)?.*git.*\. 
                </div>
                <div id="custom-gitlab-url-container">
                    <label for="custom-gitlab-url">
                        <input type="text" id="custom-gitlab-url">
                    </label>
                </div>
            </div>
        </details>
    </section>

    <div id="status"></div>
    <div id="buttons-container">
        <button id="save">Save</button>
        <button id="reset">Reset to defaults</button>
    </div>

    <script src="browser-polyfill.js"></script>
    <script src="options.js"></script>
    <script src="options_page.js"></script>
</body>

</html>

`.replaceAll(NEWLINE, ''),
);
const OPTIONS_CSS = atob(
`
Ym9keSB7CiAgd2lkdGg6IDQwMHB4Owp9CgpzZWN0aW9uIHsKICBtYXJnaW4tYm90dG9tOiAxMHB4Owp9CgpoMiB7CiAgZm9udC1zaXplOiAxLjRlbTsKICBmb250LXdlaWdodDogNTUwOwogIG1hcmdpbi10b3A6IDEwcHg7CiAgbWFyZ2luLWJvdHRvbTogMTBweDsKfQoKaDMgewogIGZvbnQtc2l6ZTogMS4yZW07CiAgZm9udC13ZWlnaHQ6IDQ1MDsKICBtYXJnaW4tdG9wOiAxMHB4OwogIG1hcmdpbi1ib3R0b206IDVweDsKfQoKc3VtbWFyeSB7CiAgbWFyZ2luLXRvcDogNXB4OwogIG1hcmdpbi1ib3R0b206IDVweDsKfQoKZGV0YWlscyB7CiAgbWFyZ2luLWJvdHRvbTogNXB4Owp9CgpzdW1tYXJ5IGgzLApzdW1tYXJ5IGgyIHsKICBkaXNwbGF5OiBpbmxpbmU7Cn0KCi5vcHRpb24gewogIG1hcmdpbi1ib3R0b206IDVweDsKfQoKLm9wdGlvbi1kZXNjIHsKICBkaXNwbGF5OiBpbmxpbmUtYmxvY2s7CiAgd2lkdGg6IDQ4JTsKfQoKLmlucHV0LWtleWJpbmRpbmcgewogIG1hcmdpbi1sZWZ0OiBhdXRvOwogIG1hcmdpbi1yaWdodDogMDsKICB3aWR0aDogNDglOwp9CgouaGVscCB7CiAgZm9udC13ZWlnaHQ6IDM1MDsKfQoKI2N1c3RvbS1jc3MtdGV4dGFyZWEgewogIHdpZHRoOiAxMDAlOwogIGhlaWdodDogNDAwcHg7Cn0KCi5zZWFyY2gtZW5naW5lLWNoZWNrYm94IHsKICBkaXNwbGF5OiBibG9jazsKfQoKI2RlbGF5LWNvbnRhaW5lciB7CiAgbWFyZ2luLXRvcDogNXB4Owp9CgojZGVsYXkgewogIHdpZHRoOiA3NXB4OwogIG1hcmdpbi1yaWdodDogNXB4Owp9Cgojc3RhdHVzIHsKICBmb250LXdlaWdodDogYm9sZDsKfQoKI2J1dHRvbnMtY29udGFpbmVyIGJ1dHRvbiB7CiAgbWFyZ2luLXRvcDogNXB4OwogIG1hcmdpbi1ib3R0b206IDVweDsKfQoKLyogRmlyZWZveCBzcGVjaWZpYyBvdmVycmlkZXMgKi8KQC1tb3otZG9jdW1lbnQgdXJsLXByZWZpeCgiIikgewogIGJvZHkgewogICAgd2lkdGg6IDYwMHB4OwogICAgLyogV2l0aG91dCB0aGlzLCB0aGUgRmlyZWZveCBvcHRpb25zIHBhZ2UgYm9keSBpcyBjbG9zZSB0byB0aGUgYm9yZGVyICovCiAgICBtYXJnaW4tbGVmdDogMTBweDsKICB9CgogIC5vcHRpb24tZGVzYyB7CiAgICB3aWR0aDogMjgwcHg7CiAgfQoKICAjY3VzdG9tLWNzcy10ZXh0YXJlYSB7CiAgICB3aWR0aDogNjAwcHg7CiAgICBoZWlnaHQ6IDYwMHB4OwogIH0KfQo=
`.replaceAll(NEWLINE, ''),
);
const OPTIONS_JS = atob(
`
const DEFAULT_CSS = `/* NOTE:
 *
 * - Using !important is needed for some styles because otherwise they get
 *   overriden by the search engine stylesheets
 * - Using outline works better than border sometimes because creating the
 *   border can move other elements, for example the page numbers are moved in
 *   Google Scholar when highlighting the prev/next buttons.
 */

:root {
  --result-outline: 1px solid black;
}

@media (prefers-color-scheme: dark) {
  :root {
    --result-outline: 1px solid #aaaaaa;
  }
}

html[dark], [dark] {
  --result-outline: 1px solid #aaaaaa;
}

.wsn-google-focused-link {
    position: relative;
    /* This is required for the arrow to appear when navigating sub-results, see
     * also: https://github.com/infokiller/web-search-navigator/issues/357 */
    overflow: visible !important;
}

.wsn-google-focused-link::before,
.wsn-google-focused-map::before,
.wsn-gitlab-focused-link::before,
.wsn-brave-search-focused-link::before,
.wsn-startpage-focused-link::before {
    content: "\u25BA";
    margin-right: 25px;
    left: -25px;
    position: absolute;
}

.wsn-brave-search-focused-news {
  position: relative;
}

.wsn-brave-search-focused-news::before {
  content: "\u25BA";
  top: 5px;
  left: -45px;
  position: absolute;
}

.wsn-google-focused-image {
    outline: var(--result-outline) !important;
    /* Images are less visible with a thin outline */
    outline-width: 2px;
}

.wsn-google-focused-card,
.wsn-brave-search-focused-card,
.wsn-google-focused-job-card {
    border: var(--result-outline) !important;
}

.wsn-google-focused-map,
.wsn-google-card-item,
.wsn-gitlab-focused-group-row {
    outline: var(--result-outline) !important;
}

.wsn-google-focused-memex-result {
    border: var(--result-outline) !important;
    box-sizing: border-box;
    -moz-box-sizing: border-box;
    -webkit-box-sizing: border-box;
}

/* Startpage has dark themes where a black outline won't be visible */
.wsn-startpage-focused-link {
    outline: 1px solid #435a69 !important;
    outline-offset: 3px;
}

.wsn-youtube-focused-video {
    outline: var(--result-outline) !important;
    outline-offset: 1px;
}

.wsn-youtube-focused-grid-video {
    border: var(--result-outline) !important;
}

.wsn-google-scholar-next-page {
    /* Using outline works better than border for the Scholar previous/next
     * buttons because border moves the page numbers a bit. */
    outline: var(--result-outline) !important;
}

.wsn-amazon-focused-item {
    outline: var(--result-outline) !important;
    outline-offset: 3px;
}

.wsn-amazon-focused-cart-item,
.wsn-amazon-focused-carousel-item {
    border: var(--result-outline) !important;
}

.wsn-github-focused-item,
.wsn-github-focused-pagination {
    outline: var(--result-outline) !important;
    outline-offset: 2px;
}

/* This rule is only used when the "hide outline" option is enabled, and is used
 * to disable the website's default search result outlining */
.wsn-no-outline,
.wsn-no-outline:focus {
    outline: none;
}`;

const DEFAULT_KEYBINDINGS = {
  nextKey: ['down', 'j'],
  previousKey: ['up', 'k'],
  navigatePreviousResultPage: ['left', 'h'],
  navigateNextResultPage: ['right', 'l'],
  navigateKey: ['return', 'space'],
  navigateNewTabBackgroundKey: ['ctrl+return', 'command+return', 'ctrl+space'],
  navigateNewTabKey: [
    'ctrl+shift+return',
    'command+shift+return',
    'ctrl+shift+space',
  ],
  navigateSearchTab: ['a', 's'],
  navigateImagesTab: ['i'],
  navigateVideosTab: ['v'],
  navigateMapsTab: ['m'],
  navigateNewsTab: ['n'],
  navigateShoppingTab: ['alt+s'],
  navigateBooksTab: ['b'],
  navigateFlightsTab: ['alt+l'],
  navigateFinancialTab: ['f'],
  focusSearchInput: ['/', 'escape'],
  navigateShowAll: ['z z'],
  navigateShowHour: ['z h'],
  navigateShowDay: ['z d'],
  navigateShowWeek: ['z w'],
  navigateShowMonth: ['z m'],
  navigateShowYear: ['z y'],
  toggleSort: ['z s'],
  toggleVerbatimSearch: ['z v'],
  showImagesLarge: ['z l'],
  showImagesMedium: ['z e'],
  showImagesIcon: ['z i'],
  copyUrlKey: [],
};

const DEFAULT_OPTIONS = {
  ...DEFAULT_KEYBINDINGS,
  wrapNavigation: false,
  autoSelectFirst: true,
  hideOutline: false,
  delay: 0,
  googleIncludeCards: true,
  googleIncludeMemex: false,
  googleIncludePlaces: true,
  customCSS: DEFAULT_CSS,
  simulateMiddleClick: false,
  customGitlabUrl: '^https://(www.)?\\.*git.*\\.',
};

const keybindingStringToArray = (kb) => {
  // Alternative: kb.split(/, */);
  return kb.split(',').map((t) => t.trim());
};

// eslint-disable-next-line no-unused-vars
const keybindingArrayToString = (kb) => {
  return kb.join(', ');
};

/**
 * @param {StorageArea} storage The storage area to which this section will
 *  write.
 * @param {Object} defaultValues The default options.
 * @constructor
 */
class BrowserStorage {
  constructor(storage, defaultValues) {
    this.storage = storage;
    this.values = {};
    this.defaultValues = defaultValues;
  }
  load() {
    // this.storage.get(null) returns all the data stored:
    // https://developer.chrome.com/extensions/storage#method-StorageArea-get
    return this.storage.get(null).then((values) => {
      this.values = values;
      // Prior to versions 0.4.* the keybindings were stored as strings, so we
      // migrate them to arrays if needed.
      let migrated = false;
      for (const [key, value] of Object.entries(this.values)) {
        if (!(key in DEFAULT_KEYBINDINGS) || Array.isArray(value)) {
          continue;
        }
        migrated = true;
        this.values[key] = keybindingStringToArray(value);
      }
      if (migrated) {
        return this.save();
      }
    });
  }
  save() {
    return this.storage.set(this.values);
  }
  get(key) {
    const value = this.values[key];
    if (value != null) {
      return value;
    }
    return this.defaultValues[key];
  }
  set(key, value) {
    this.values[key] = value;
  }
  clear() {
    return this.storage.clear().then(() => {
      this.values = {};
    });
  }
  getAll() {
    // Merge options from storage with defaults.
    return { ...this.defaultValues, ...this.values };
  }
}

const STORAGE_KEY = 'webSearchNavigator';

class LocalStorage {
  constructor(defaultValues) {
    this.values = {};
    this.defaultValues = defaultValues;
    this.load();
  }

  load() {
    const storedData = localStorage.getItem(STORAGE_KEY);

    if (storedData) {
      this.values = JSON.parse(storedData);
    } else {
      this.values = { ...this.defaultValues };
      this.save();
    }
  }

  save() {
    localStorage.setItem(STORAGE_KEY, JSON.stringify(this.values));
  }

  get(key) {
    const value = this.values[key];
    if (value != null) {
      return value;
    }
    return this.defaultValues[key];
  }

  set(key, value) {
    this.values[key] = value;
    this.save();
  }

  clear() {
    localStorage.removeItem(STORAGE_KEY);
    this.values = { ...this.defaultValues };
  }

  getAll() {
    // Merge options from storage with defaults.
    return { ...this.defaultValues, ...this.values };
  }
}

const createSyncedOptions = () => {
  if (globalThis.IS_USERSCRIPT) {
    console.log('Create LocalStorage options');
    return new LocalStorage(DEFAULT_OPTIONS);
  }
  return new BrowserStorage(browser.storage.sync, DEFAULT_OPTIONS);
};

// eslint-disable-next-line no-unused-vars
class ExtensionOptions {
  constructor() {
    this.sync = createSyncedOptions();
    if (globalThis.IS_USERSCRIPT) {
      this.local = createSyncedOptions();
      return;
    }
    this.local = new BrowserStorage(browser.storage.local, {
      lastQueryUrl: null,
      lastFocusedIndex: 0,
    });
  }

  load() {
    return Promise.all([this.local.load(), this.sync.load()]);
  }
}

`.replaceAll(NEWLINE, ''),
);
const OPTIONS_PAGE_JS = atob(
`
// Based on https://developer.chrome.com/extensions/optionsV2

/* global keybindingStringToArray, keybindingArrayToString */
/* global createSyncedOptions, DEFAULT_CSS */

const GOOGLE_DOMAINS = [
  'ad', 'ae', 'al', 'am', 'as', 'at', 'az', 'ba', 'be', 'bf', 'bg', 'bi', 'bj',
  'bs', 'bt', 'by', 'ca', 'cat', 'cd', 'cf', 'cg', 'ch', 'ci', 'cl', 'cm', 'cn',
  'co.ao', 'co.bw', 'co.ck', 'co.cr', 'co.id', 'co.il', 'co.in', 'co.jp',
  'co.ke', 'co.kr', 'co.ls', 'co.ma', 'co.mz', 'co.nz', 'co.th', 'co.tz',
  'co.ug', 'co.uk', 'co.uz', 'co.ve', 'co.vi', 'co.za', 'co.zm', 'co.zw', 'com',
  'com.af', 'com.ag', 'com.ai', 'com.ar', 'com.au', 'com.bd', 'com.bh',
  'com.bn', 'com.bo', 'com.br', 'com.bz', 'com.co', 'com.cu', 'com.cy',
  'com.do', 'com.ec', 'com.eg', 'com.et', 'com.fj', 'com.gh', 'com.gi',
  'com.gt', 'com.hk', 'com.jm', 'com.kh', 'com.kw', 'com.lb', 'com.ly',
  'com.mm', 'com.mt', 'com.mx', 'com.my', 'com.na', 'com.nf', 'com.ng',
  'com.ni', 'com.np', 'com.om', 'com.pa', 'com.pe', 'com.pg', 'com.ph',
  'com.pk', 'com.pr', 'com.py', 'com.qa', 'com.sa', 'com.sb', 'com.sg',
  'com.sl', 'com.sv', 'com.tj', 'com.tr', 'com.tw', 'com.ua', 'com.uy',
  'com.vc', 'com.vn', 'cv', 'cz', 'de', 'dj', 'dk', 'dm', 'dz', 'ee', 'es',
  'fi', 'fm', 'fr', 'ga', 'ge', 'gg', 'gl', 'gm', 'gp', 'gr', 'gy', 'hn', 'hr',
  'ht', 'hu', 'ie', 'im', 'iq', 'is', 'it', 'je', 'jo', 'kg', 'ki', 'kz', 'la',
  'li', 'lk', 'lt', 'lu', 'lv', 'md', 'me', 'mg', 'mk', 'ml', 'mn', 'ms', 'mu',
  'mv', 'mw', 'ne', 'nl', 'no', 'nr', 'nu', 'pl', 'pn', 'ps', 'pt', 'ro', 'rs',
  'ru', 'rw', 'sc', 'se', 'sh', 'si', 'sk', 'sm', 'sn', 'so', 'sr', 'st', 'td',
  'tg', 'tk', 'tl', 'tm', 'tn', 'to', 'tt', 'vg', 'vu', 'ws',
];

const AMAZON_DOMAINS = [
  'ca',
  'cn',
  'co.jp',
  'co.uk',
  'com',
  'com.au',
  'com.br',
  'com.mx',
  'de',
  'es',
  'fr',
  'in',
  'it',
  'nl',
];

const generateURLPatterns = (prefix, domains, suffix) => {
  const urls = [];
  for (const domain of domains) {
    urls.push(`${prefix}.${domain}${suffix}`);
  }
  return urls;
};

// Authorized urls for compatible search engines
const OPTIONAL_PERMISSIONS_URLS = {
  'brave-search': ['https://search.brave.com/*'],
  'startpage': [
    // It used to be 'https://www.startpage.com/*/*search*' but when requesting
    // this URL chrome actually grants permission to the URL below. This
    // discrepancy causes the options page to think that we don't have
    // permission for startpage.
    'https://www.startpage.com/*',
    'https://startpage.com/*',
  ],
  'youtube': ['https://www.youtube.com/*'],
  'google-scholar': generateURLPatterns(
      'https://scholar.google',
      GOOGLE_DOMAINS,
      '/*',
  ),
  'github': ['https://github.com/*'],
  'amazon': generateURLPatterns('https://www.amazon', AMAZON_DOMAINS, '/*'),
  'gitlab': ['https://gitlab.com/*'],
  'custom-gitlab': ['https://*/*'],
};

globalThis._browser_userscript_polyfill.permissions.getAll = () => ({
  origins: Object.values(OPTIONAL_PERMISSIONS_URLS).flat(),
})

const KEYBINDING_TO_DIV = {
  nextKey: 'next-key',
  previousKey: 'previous-key',
  navigatePreviousResultPage: 'navigate-previous-result-page',
  navigateNextResultPage: 'navigate-next-result-page',
  navigateKey: 'navigate-key',
  navigateNewTabKey: 'navigate-new-tab-key',
  navigateNewTabBackgroundKey: 'navigate-new-tab-background-key',
  navigateSearchTab: 'navigate-search-tab',
  navigateImagesTab: 'navigate-images-tab',
  navigateVideosTab: 'navigate-videos-tab',
  navigateMapsTab: 'navigate-maps-tab',
  navigateNewsTab: 'navigate-news-tab',
  navigateShoppingTab: 'navigate-shopping-tab',
  navigateBooksTab: 'navigate-books-tab',
  navigateFlightsTab: 'navigate-flights-tab',
  navigateFinancialTab: 'navigate-financial-tab',
  focusSearchInput: 'focus-search-input',
  navigateShowAll: 'navigate-show-all',
  navigateShowHour: 'navigate-show-hour',
  navigateShowDay: 'navigate-show-day',
  navigateShowWeek: 'navigate-show-week',
  navigateShowMonth: 'navigate-show-month',
  navigateShowYear: 'navigate-show-year',
  toggleSort: 'toggle-sort',
  toggleVerbatimSearch: 'toggle-verbatim-search',
  showImagesLarge: 'show-images-large',
  showImagesMedium: 'show-images-medium',
  showImagesIcon: 'show-images-icon',
  copyUrlKey: 'copy-url-key',
};

/**
 * Add other search engines domain on user input
 * @param {Element} checkbox
 */
const setSearchEnginePermission_ = async (checkbox) => {
  const urls = OPTIONAL_PERMISSIONS_URLS[checkbox.id];
  if (checkbox.checked) {
    checkbox.checked = false;
    const granted = await browser.permissions.request({origins: urls});
    checkbox.checked = granted;
  } else {
    browser.permissions.remove({origins: urls});
  }
};

class OptionsPageManager {
  async init() {
    await this.loadOptions();
    const braveSearch = document.getElementById('brave-search');
    braveSearch.addEventListener('change', () => {
      setSearchEnginePermission_(braveSearch);
    });
    const startpage = document.getElementById('startpage');
    startpage.addEventListener('change', () => {
      setSearchEnginePermission_(startpage);
    });
    const youtube = document.getElementById('youtube');
    youtube.addEventListener('change', () => {
      setSearchEnginePermission_(youtube);
    });
    const googleScholar = document.getElementById('google-scholar');
    googleScholar.addEventListener('change', () => {
      setSearchEnginePermission_(googleScholar);
    });
    const github = document.getElementById('github');
    github.addEventListener('change', () => {
      setSearchEnginePermission_(github);
    });
    const amazon = document.getElementById('amazon');
    amazon.addEventListener('change', () => {
      setSearchEnginePermission_(amazon);
    });
    const gitlab = document.getElementById('gitlab');
    gitlab.addEventListener('change', () => {
      setSearchEnginePermission_(gitlab);
    });
    const customGitlab = document.getElementById('custom-gitlab');
    customGitlab.addEventListener('change', () => {
      setSearchEnginePermission_(customGitlab);
    });
    // NOTE: this.saveOptions and this.resetToDefaults cannot be passed directly
    // or otherwise `this` won't be bound to the object.
    document.getElementById('save').addEventListener('click', () => {
      this.saveOptions();
    });
    document.getElementById('reset').addEventListener('click', () => {
      this.resetToDefaults();
    });
  }

  // Saves options from the DOM to browser.storage.sync.
  async saveOptions() {
    const getOpt = (key) => {
      return this.options.get(key);
    };
    const setOpt = (key, value) => {
      console.log('Set', key, value, this.options.storage, this.options);
      this.options.set(key, value);
    };
    // Handle non-keybindings settings first
    setOpt(
        'wrapNavigation',
        document.getElementById('wrap-navigation').checked,
    );
    setOpt(
        'autoSelectFirst',
        document.getElementById('auto-select-first').checked,
    );
    setOpt('hideOutline', document.getElementById('hide-outline').checked);
    setOpt('delay', document.getElementById('delay').value);
    setOpt(
        'googleIncludeCards',
        document.getElementById('google-include-cards').checked,
    );
    setOpt(
        'googleIncludeMemex',
        document.getElementById('google-include-memex').checked,
    );
    setOpt(
        'googleIncludePlaces',
        document.getElementById('google-include-places').checked,
    );
    // Handle keybinding options
    for (const [key, optName] of Object.entries(KEYBINDING_TO_DIV)) {
      // Keybindings are stored internally as arrays, but edited by users as
      // comman delimited strings.
      setOpt(
          key,
          keybindingStringToArray(document.getElementById(optName).value),
      );
    }
    const customCSS = document.getElementById('custom-css-textarea').value;
    if (getOpt('customCSS') !== DEFAULT_CSS || customCSS !== DEFAULT_CSS) {
      if (customCSS.trim()) {
        setOpt('customCSS', customCSS);
      } else {
        setOpt('customCSS', DEFAULT_CSS);
      }
    }
    setOpt(
        'simulateMiddleClick',
        document.getElementById('simulate-middle-click').checked,
    );
    const gitlabURLRegex = document.getElementById('custom-gitlab-url').value;
    try {
      new RegExp(gitlabURLRegex);
      setOpt(
          'customGitlabUrl',
          document.getElementById('custom-gitlab-url').value,
      );
    } catch (e) {
      const status = document.getElementById('status');
      status.textContent = `Invalid gitlab URL regex: ${e.message}`;
      return;
    }
    try {
      await this.options.save();
      this.flashMessage('Options saved');
    } catch (e) {
      this.flashMessage('Error when saving options');
    }
  }

  loadSearchEnginePermissions_(permissions) {
    // Check what URLs we have permission for.
    const braveSearch = document.getElementById('brave-search');
    braveSearch.checked = OPTIONAL_PERMISSIONS_URLS['brave-search'].every(
        (url) => {
          return permissions.origins.includes(url);
        },
    );
    const startpage = document.getElementById('startpage');
    startpage.checked = OPTIONAL_PERMISSIONS_URLS['startpage'].every((url) => {
      return permissions.origins.includes(url);
    });
    const youtube = document.getElementById('youtube');
    youtube.checked = OPTIONAL_PERMISSIONS_URLS['youtube'].every((url) => {
      return permissions.origins.includes(url);
    });
    const googleScholar = document.getElementById('google-scholar');
    googleScholar.checked = OPTIONAL_PERMISSIONS_URLS['google-scholar'].every(
        (url) => {
          return permissions.origins.includes(url);
        },
    );
    const amazon = document.getElementById('amazon');
    amazon.checked = OPTIONAL_PERMISSIONS_URLS['amazon'].every((url) => {
      return permissions.origins.includes(url);
    });
    const github = document.getElementById('github');
    github.checked = OPTIONAL_PERMISSIONS_URLS['github'].every((url) => {
      return permissions.origins.includes(url);
    });
    const gitlab = document.getElementById('gitlab');
    gitlab.checked = OPTIONAL_PERMISSIONS_URLS['gitlab'].every((url) => {
      return permissions.origins.includes(url);
    });
    const customGitlab = document.getElementById('custom-gitlab');
    customGitlab.checked = OPTIONAL_PERMISSIONS_URLS['custom-gitlab'].every(
        (url) => {
          return permissions.origins.includes(url);
        },
    );
  }

  // Load options from browser.storage.sync to the DOM.
  async loadOptions() {
    this.options = createSyncedOptions();
    const [, permissions] = await Promise.all([
      this.options.load(),
      browser.permissions.getAll(),
    ]);
    this.loadSearchEnginePermissions_(permissions);
    const getOpt = (key) => {
      return this.options.get(key);
    };
    // Handle checks separately.
    document.getElementById('wrap-navigation').checked =
      getOpt('wrapNavigation');
    document.getElementById('auto-select-first').checked =
      getOpt('autoSelectFirst');
    document.getElementById('hide-outline').checked = getOpt('hideOutline');
    document.getElementById('delay').value = getOpt('delay');
    document.getElementById('custom-gitlab-url').value =
      getOpt('customGitlabUrl');
    document.getElementById('google-include-cards').checked =
      getOpt('googleIncludeCards');
    document.getElementById('google-include-memex').checked =
      getOpt('googleIncludeMemex');
    document.getElementById('google-include-places').checked = getOpt(
        'googleIncludePlaces',
    );
    // Restore options from divs.
    for (const [key, optName] of Object.entries(KEYBINDING_TO_DIV)) {
      // Keybindings are stored internally as arrays, but edited by users as
      // comman delimited strings.
      document.getElementById(optName).value = keybindingArrayToString(
          getOpt(key),
      );
    }
    // Load custom CSS
    document.getElementById('custom-css-textarea').value = getOpt('customCSS');
    document.getElementById('simulate-middle-click').checked = getOpt(
        'simulateMiddleClick',
    );
  }

  async resetToDefaults() {
    try {
      await this.options.clear();
      await this.loadOptions();
      this.flashMessage('Options set to defaults');
    } catch (e) {
      this.flashMessage('Error when setting options to defaults');
    }
  }

  flashMessage(message) {
    // Update status to let user know.
    const status = document.getElementById('status');
    status.textContent = message;
    setTimeout(() => {
      status.textContent = '';
    }, 3000);
  }
}

const manager = new OptionsPageManager();
// NOTE: manager.init cannot be passed directly or otherwise `this` won't be
// bound to the object.
document.addEventListener('DOMContentLoaded', () => {
  manager.init();
});

`.replaceAll(NEWLINE, ''),
);
const BROWSER_POLYFILL_JS = atob(
`
globalThis.IS_USERSCRIPT = true;

const USE_GM = false;
const PREFIX = 'userscript-polyfill';

globalThis._localStorage_browser_polyfill = {
  get: async (...args) => {
    args = args.filter(Boolean);
    debugger;
    console.log('[localStorage] Get: ', ...args);
    const out = {};
    for (const k of args) {
      out[k] = USE_GM ? GM_getValue(k) : localStorage[`${PREFIX}_${k}`];
    }
    return out;
  },
  set: async (...args) => {
    debugger;
    console.log('[localStorage] Set: ', ...args);
  },
  clear: async () => {
    debugger;
    console.log('[localStorage] Clear');
  },
};

globalThis._browser_userscript_polyfill = {
  runtime: {
    sendMessage: (msg) => {
      if (msg.type === 'tabsCreate') {
        window.open(msg.options.url, '_blank');
      }
    },
    id: '093889f3-43be-45e3-bc5a-e257e75b466d',
  },
  storage: {sync: globalThis._localStorage_browser_polyfill, local: globalThis._localStorage_browser_polyfill},
  permissions: {
    remove: () => {},
    add: () => {},
    request: () => {},
    getAll: () => ({}),
  },
};
console.log(globalThis.browser, _browser_userscript_polyfill);
Object.assign(globalThis, {browser: globalThis._browser_userscript_polyfill, chrome: globalThis._browser_userscript_polyfill});
(function(a,b){if("function"==typeof define&&define.amd)define("webextension-polyfill",["module"],b);else if("undefined"!=typeof exports)b(module);else{var c={exports:{}};b(c),a.browser=c.exports}})("undefined"==typeof globalThis?"undefined"==typeof self?this:self:globalThis,function(a){"use strict";if(!(globalThis.chrome&&globalThis.chrome.runtime&&globalThis.chrome.runtime.id))throw new Error("This script should only be loaded in a browser extension.");if(!(globalThis.browser&&globalThis.browser.runtime&&globalThis.browser.runtime.id)){a.exports=(a=>{const b={alarms:{clear:{minArgs:0,maxArgs:1},clearAll:{minArgs:0,maxArgs:0},get:{minArgs:0,maxArgs:1},getAll:{minArgs:0,maxArgs:0}},bookmarks:{create:{minArgs:1,maxArgs:1},get:{minArgs:1,maxArgs:1},getChildren:{minArgs:1,maxArgs:1},getRecent:{minArgs:1,maxArgs:1},getSubTree:{minArgs:1,maxArgs:1},getTree:{minArgs:0,maxArgs:0},move:{minArgs:2,maxArgs:2},remove:{minArgs:1,maxArgs:1},removeTree:{minArgs:1,maxArgs:1},search:{minArgs:1,maxArgs:1},update:{minArgs:2,maxArgs:2}},browserAction:{disable:{minArgs:0,maxArgs:1,fallbackToNoCallback:!0},enable:{minArgs:0,maxArgs:1,fallbackToNoCallback:!0},getBadgeBackgroundColor:{minArgs:1,maxArgs:1},getBadgeText:{minArgs:1,maxArgs:1},getPopup:{minArgs:1,maxArgs:1},getTitle:{minArgs:1,maxArgs:1},openPopup:{minArgs:0,maxArgs:0},setBadgeBackgroundColor:{minArgs:1,maxArgs:1,fallbackToNoCallback:!0},setBadgeText:{minArgs:1,maxArgs:1,fallbackToNoCallback:!0},setIcon:{minArgs:1,maxArgs:1},setPopup:{minArgs:1,maxArgs:1,fallbackToNoCallback:!0},setTitle:{minArgs:1,maxArgs:1,fallbackToNoCallback:!0}},browsingData:{remove:{minArgs:2,maxArgs:2},removeCache:{minArgs:1,maxArgs:1},removeCookies:{minArgs:1,maxArgs:1},removeDownloads:{minArgs:1,maxArgs:1},removeFormData:{minArgs:1,maxArgs:1},removeHistory:{minArgs:1,maxArgs:1},removeLocalStorage:{minArgs:1,maxArgs:1},removePasswords:{minArgs:1,maxArgs:1},removePluginData:{minArgs:1,maxArgs:1},settings:{minArgs:0,maxArgs:0}},commands:{getAll:{minArgs:0,maxArgs:0}},contextMenus:{remove:{minArgs:1,maxArgs:1},removeAll:{minArgs:0,maxArgs:0},update:{minArgs:2,maxArgs:2}},cookies:{get:{minArgs:1,maxArgs:1},getAll:{minArgs:1,maxArgs:1},getAllCookieStores:{minArgs:0,maxArgs:0},remove:{minArgs:1,maxArgs:1},set:{minArgs:1,maxArgs:1}},devtools:{inspectedWindow:{eval:{minArgs:1,maxArgs:2,singleCallbackArg:!1}},panels:{create:{minArgs:3,maxArgs:3,singleCallbackArg:!0},elements:{createSidebarPane:{minArgs:1,maxArgs:1}}}},downloads:{cancel:{minArgs:1,maxArgs:1},download:{minArgs:1,maxArgs:1},erase:{minArgs:1,maxArgs:1},getFileIcon:{minArgs:1,maxArgs:2},open:{minArgs:1,maxArgs:1,fallbackToNoCallback:!0},pause:{minArgs:1,maxArgs:1},removeFile:{minArgs:1,maxArgs:1},resume:{minArgs:1,maxArgs:1},search:{minArgs:1,maxArgs:1},show:{minArgs:1,maxArgs:1,fallbackToNoCallback:!0}},extension:{isAllowedFileSchemeAccess:{minArgs:0,maxArgs:0},isAllowedIncognitoAccess:{minArgs:0,maxArgs:0}},history:{addUrl:{minArgs:1,maxArgs:1},deleteAll:{minArgs:0,maxArgs:0},deleteRange:{minArgs:1,maxArgs:1},deleteUrl:{minArgs:1,maxArgs:1},getVisits:{minArgs:1,maxArgs:1},search:{minArgs:1,maxArgs:1}},i18n:{detectLanguage:{minArgs:1,maxArgs:1},getAcceptLanguages:{minArgs:0,maxArgs:0}},identity:{launchWebAuthFlow:{minArgs:1,maxArgs:1}},idle:{queryState:{minArgs:1,maxArgs:1}},management:{get:{minArgs:1,maxArgs:1},getAll:{minArgs:0,maxArgs:0},getSelf:{minArgs:0,maxArgs:0},setEnabled:{minArgs:2,maxArgs:2},uninstallSelf:{minArgs:0,maxArgs:1}},notifications:{clear:{minArgs:1,maxArgs:1},create:{minArgs:1,maxArgs:2},getAll:{minArgs:0,maxArgs:0},getPermissionLevel:{minArgs:0,maxArgs:0},update:{minArgs:2,maxArgs:2}},pageAction:{getPopup:{minArgs:1,maxArgs:1},getTitle:{minArgs:1,maxArgs:1},hide:{minArgs:1,maxArgs:1,fallbackToNoCallback:!0},setIcon:{minArgs:1,maxArgs:1},setPopup:{minArgs:1,maxArgs:1,fallbackToNoCallback:!0},setTitle:{minArgs:1,maxArgs:1,fallbackToNoCallback:!0},show:{minArgs:1,maxArgs:1,fallbackToNoCallback:!0}},permissions:{contains:{minArgs:1,maxArgs:1},getAll:{minArgs:0,maxArgs:0},remove:{minArgs:1,maxArgs:1},request:{minArgs:1,maxArgs:1}},runtime:{getBackgroundPage:{minArgs:0,maxArgs:0},getPlatformInfo:{minArgs:0,maxArgs:0},openOptionsPage:{minArgs:0,maxArgs:0},requestUpdateCheck:{minArgs:0,maxArgs:0},sendMessage:{minArgs:1,maxArgs:3},sendNativeMessage:{minArgs:2,maxArgs:2},setUninstallURL:{minArgs:1,maxArgs:1}},sessions:{getDevices:{minArgs:0,maxArgs:1},getRecentlyClosed:{minArgs:0,maxArgs:1},restore:{minArgs:0,maxArgs:1}},storage:{local:{clear:{minArgs:0,maxArgs:0},get:{minArgs:0,maxArgs:1},getBytesInUse:{minArgs:0,maxArgs:1},remove:{minArgs:1,maxArgs:1},set:{minArgs:1,maxArgs:1}},managed:{get:{minArgs:0,maxArgs:1},getBytesInUse:{minArgs:0,maxArgs:1}},sync:{clear:{minArgs:0,maxArgs:0},get:{minArgs:0,maxArgs:1},getBytesInUse:{minArgs:0,maxArgs:1},remove:{minArgs:1,maxArgs:1},set:{minArgs:1,maxArgs:1}}},tabs:{captureVisibleTab:{minArgs:0,maxArgs:2},create:{minArgs:1,maxArgs:1},detectLanguage:{minArgs:0,maxArgs:1},discard:{minArgs:0,maxArgs:1},duplicate:{minArgs:1,maxArgs:1},executeScript:{minArgs:1,maxArgs:2},get:{minArgs:1,maxArgs:1},getCurrent:{minArgs:0,maxArgs:0},getZoom:{minArgs:0,maxArgs:1},getZoomSettings:{minArgs:0,maxArgs:1},goBack:{minArgs:0,maxArgs:1},goForward:{minArgs:0,maxArgs:1},highlight:{minArgs:1,maxArgs:1},insertCSS:{minArgs:1,maxArgs:2},move:{minArgs:2,maxArgs:2},query:{minArgs:1,maxArgs:1},reload:{minArgs:0,maxArgs:2},remove:{minArgs:1,maxArgs:1},removeCSS:{minArgs:1,maxArgs:2},sendMessage:{minArgs:2,maxArgs:3},setZoom:{minArgs:1,maxArgs:2},setZoomSettings:{minArgs:1,maxArgs:2},update:{minArgs:1,maxArgs:2}},topSites:{get:{minArgs:0,maxArgs:0}},webNavigation:{getAllFrames:{minArgs:1,maxArgs:1},getFrame:{minArgs:1,maxArgs:1}},webRequest:{handlerBehaviorChanged:{minArgs:0,maxArgs:0}},windows:{create:{minArgs:0,maxArgs:1},get:{minArgs:1,maxArgs:2},getAll:{minArgs:0,maxArgs:1},getCurrent:{minArgs:0,maxArgs:1},getLastFocused:{minArgs:0,maxArgs:1},remove:{minArgs:1,maxArgs:1},update:{minArgs:2,maxArgs:2}}};if(0===Object.keys(b).length)throw new Error("api-metadata.json has not been included in browser-polyfill");class c extends WeakMap{constructor(a,b=void 0){super(b),this.createItem=a}get(a){return this.has(a)||this.set(a,this.createItem(a)),super.get(a)}}const d=a=>a&&"object"==typeof a&&"function"==typeof a.then,e=(b,c)=>(...d)=>{a.runtime.lastError?b.reject(new Error(a.runtime.lastError.message)):c.singleCallbackArg||1>=d.length&&!1!==c.singleCallbackArg?b.resolve(d[0]):b.resolve(d)},f=a=>1==a?"argument":"arguments",g=(a,b)=>function(c,...d){if(d.length<b.minArgs)throw new Error(`Expected at least ${b.minArgs} ${f(b.minArgs)} for ${a}(), got ${d.length}`);if(d.length>b.maxArgs)throw new Error(`Expected at most ${b.maxArgs} ${f(b.maxArgs)} for ${a}(), got ${d.length}`);return new Promise((f,g)=>{if(b.fallbackToNoCallback)try{c[a](...d,e({resolve:f,reject:g},b))}catch(e){console.warn(`${a} API method doesn't seem to support the callback parameter, `+"falling back to call it without a callback: ",e),c[a](...d),b.fallbackToNoCallback=!1,b.noCallback=!0,f()}else b.noCallback?(c[a](...d),f()):c[a](...d,e({resolve:f,reject:g},b))})},h=(a,b,c)=>new Proxy(b,{apply(b,d,e){return c.call(d,a,...e)}});let i=Function.call.bind(Object.prototype.hasOwnProperty);const j=(a,b={},c={})=>{let d=Object.create(null),e=Object.create(a);return new Proxy(e,{has(b,c){return c in a||c in d},get(e,f){if(f in d)return d[f];if(!(f in a))return;let k=a[f];if("function"==typeof k){if("function"==typeof b[f])k=h(a,a[f],b[f]);else if(i(c,f)){let b=g(f,c[f]);k=h(a,a[f],b)}else k=k.bind(a);}else if("object"==typeof k&&null!==k&&(i(b,f)||i(c,f)))k=j(k,b[f],c[f]);else if(i(c,"*"))k=j(k,b[f],c["*"]);else return Object.defineProperty(d,f,{configurable:!0,enumerable:!0,get(){return a[f]},set(b){a[f]=b}}),k;return d[f]=k,k},set(b,c,e){return c in d?d[c]=e:a[c]=e,!0},defineProperty(a,b,c){return Reflect.defineProperty(d,b,c)},deleteProperty(a,b){return Reflect.deleteProperty(d,b)}})},k=a=>({addListener(b,c,...d){b.addListener(a.get(c),...d)},hasListener(b,c){return b.hasListener(a.get(c))},removeListener(b,c){b.removeListener(a.get(c))}}),l=new c(a=>"function"==typeof a?function(b){const c=j(b,{},{getContent:{minArgs:0,maxArgs:0}});a(c)}:a),m=new c(a=>"function"==typeof a?function(b,c,e){let f,g,h=!1,i=new Promise(a=>{f=function(b){h=!0,a(b)}});try{g=a(b,c,f)}catch(a){g=Promise.reject(a)}const j=!0!==g&&d(g);if(!0!==g&&!j&&!h)return!1;const k=a=>{a.then(a=>{e(a)},a=>{let b;b=a&&(a instanceof Error||"string"==typeof a.message)?a.message:"An unexpected error occurred",e({__mozWebExtensionPolyfillReject__:!0,message:b})}).catch(a=>{console.error("Failed to send onMessage rejected reply",a)})};return j?k(g):k(i),!0}:a),n=({reject:b,resolve:c},d)=>{a.runtime.lastError?a.runtime.lastError.message==="The message port closed before a response was received."?c():b(new Error(a.runtime.lastError.message)):d&&d.__mozWebExtensionPolyfillReject__?b(new Error(d.message)):c(d)},o=(a,b,c,...d)=>{if(d.length<b.minArgs)throw new Error(`Expected at least ${b.minArgs} ${f(b.minArgs)} for ${a}(), got ${d.length}`);if(d.length>b.maxArgs)throw new Error(`Expected at most ${b.maxArgs} ${f(b.maxArgs)} for ${a}(), got ${d.length}`);return new Promise((a,b)=>{const e=n.bind(null,{resolve:a,reject:b});d.push(e),c.sendMessage(...d)})},p={devtools:{network:{onRequestFinished:k(l)}},runtime:{onMessage:k(m),onMessageExternal:k(m),sendMessage:o.bind(null,"sendMessage",{minArgs:1,maxArgs:3})},tabs:{sendMessage:o.bind(null,"sendMessage",{minArgs:2,maxArgs:3})}},q={clear:{minArgs:1,maxArgs:1},get:{minArgs:1,maxArgs:1},set:{minArgs:1,maxArgs:1}};return b.privacy={network:{"*":q},services:{"*":q},websites:{"*":q}},j(a,p,b)})(chrome)}else a.exports=globalThis.browser});
//# sourceMappingURL=browser-polyfill.min.js.map

// webextension-polyfill v.0.12.0 (https://github.com/mozilla/webextension-polyfill)

/* This Source Code Form is subject to the terms of the Mozilla Public
 * License, v. 2.0. If a copy of the MPL was not distributed with this
 * file, You can obtain one at http://mozilla.org/MPL/2.0/. */

`.replaceAll(NEWLINE, ''),
);
function showOptions() {
const CONTAINER_ID = 'webNavigatorIframe';
if (document.getElementById(CONTAINER_ID)) {
document.getElementById(CONTAINER_ID).remove();
}
const iframe = document.createElement('iframe');
const iframe_container = document.createElement('div');
iframe_container.id = CONTAINER_ID;
iframe_container.onclick = () => {
iframe_container?.remove();
};
iframe.onclick = (e) => {
e.stopPropagation();
};
const BETTER_STYLES = `
body {padding: 30px; max-width: 600px; margin: 0 auto;}
* {box-sizing: border-box; padding: 0; margin: 0; font-family: sans-serif;}
h1, h2, h3 {font-weight: 100;}
`;
const OUT_HTML = OPTIONS_HTML.replaceAll(
`<script src="options.js"></script>`,
`<script>${NEWLINE}${NEWLINE}${OPTIONS_JS}${NEWLINE}${NEWLINE}</script>`,
)
.replaceAll(
`<script src="options_page.js"></script>`,
`<script>${NEWLINE}${NEWLINE}${OPTIONS_PAGE_JS}${NEWLINE}${NEWLINE}</script>`,
)
.replaceAll(
`<script src="browser-polyfill.js"></script>`,
`<script>${NEWLINE}${NEWLINE}${BROWSER_POLYFILL_JS}${NEWLINE}${NEWLINE}</script>`,
)
.replaceAll(
`<link rel="stylesheet" href="options_page.css">`,
`<style>${NEWLINE}${NEWLINE}${BETTER_STYLES}${NEWLINE}${NEWLINE}${OPTIONS_CSS}${NEWLINE}${NEWLINE}</style>`,
);
console.log({ OUT_HTML });
iframe.srcdoc = OUT_HTML;
Object.assign(iframe_container.style, {
position: 'fixed',
display: 'grid',
cursor: 'pointer',
placeItems: 'center',
inset: 0,
backgroundColor: '#0003',
zIndex: 100000,
});
iframe_container.appendChild(iframe);
Object.assign(iframe.style, {
width: '80vw',
height: '80vh',
border: 'none',
borderRadius: '3px',
overflow: 'hidden',
background: '#fff',
});
document.body.appendChild(iframe_container);
return { el: iframe, container: iframe_container };
}
globalThis.showOptions = showOptions;
// console.log(showOptions);
// setTimeout(() => showOptions(), 4000);
// TODO: Make the options page use postMessage to parent and localStorage to utilize settings
function blobToDataURL(blob) {
return new Promise((resolve, reject) => {
const reader = new FileReader();
reader.onload = function (e) {
resolve(reader.result);
};
reader.onerror = function (e) {
reject(reader.error);
};
reader.onabort = function (e) {
reject(new Error('Read aborted'));
};
reader.readAsDataURL(blob);
});
}
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment