Add "Search English pages" option to the language filter on Google search Tools.
// ==UserScript== // @name Google Search English Filter // @name:ja Google Search English Filter // @name:zh-CN Google Search English Filter // @description Add "Search English pages" option to the language filter on Google search Tools. // @description:ja Google検索のツールで選べる絞り込み言語として、英語を追加します。 // @description:zh-CN 作为可以通过Google搜索工具选择的缩小语言,追加英语。 // @namespace knoa.jp // @include https://www.google.*/search?* // @version 1.1.2 // @grant none // ==/UserScript== (function(){ const SCRIPTID = 'GoogleSearchEnglishFilter'; const SCRIPTNAME = 'Google Search English Filter'; const DEBUG = false;/* [update] 1.1.2 Minor fix. [bug] [todo] 言語設定で複数選んでいると「日本語のページを検索」が「英語と日本語のページを検索」になるので、日本語だけが選べない langコードを取得して、ラベルは取得できないけどうまいことやるしかないか [possible] [memo] https://www.google.com/search?q=google&client=firefox-b&sxsrf=ACYBGNTaF1aCsCLcgnQOwwDAo3nGoELowQ:1577943675296&source=lnt&tbs=lr:lang_1ja&lr=lang_ja&sa=X&ved=2ahUKEwif0PahmuTmAhWhGaYKHRCLDE0QpwV6BAgKEBk https://www.google.com/search?q=test&hl=zh-CN&sxsrf=ACYBGNQ3oa8YIfHamy9rqBV9t5530dg6Nw:1577946432840&source=lnt&tbs=lr:lang_1zh-CN%7Clang_1zh-TW&lr=lang_zh-CN%7Clang_zh-TW&sa=X&ved=2ahUKEwiIhOrEpOTmAhVME6YKHWwjBeoQpwV6BAgLEBk fetchしてもロケール言語による選択肢は取得できない。 */ if(window === top && console.time) console.time(SCRIPTID); const MS = 1, SECOND = 1000*MS, MINUTE = 60*SECOND, HOUR = 60*MINUTE, DAY = 24*HOUR, WEEK = 7*DAY, MONTH = 30*DAY, YEAR = 365*DAY; const RESET = 'GoogleSearchEnglishFilter_RESET'; const LANGUAGES = [ /* If you edited LANGUAGES, you should search "GoogleSearchEnglishFilter_RESET" on Google to apply your update */ /* https://www.google.com/search?q=GoogleSearchEnglishFilter_RESET */ {code: 'en', label: 'Search English pages', value: 'lang_en'}, //{code: 'ja', label: 'Search Japanese pages', value: 'lang_ja'}, //{code: 'fr', label: 'Search French pages', value: 'lang_fr'}, //{code: 'ru', label: 'Search Russian pages', value: 'lang_ru'}, //{code: 'es', label: 'Search Spanish pages', value: 'lang_es'}, //{code: 'ar', label: 'Search Arabic pages', value: 'lang_ar'}, //{code: 'zh-CN_zh-TW', label: 'Search Chinese (Simplified) and Chinese (Traditional) pages', value: 'lang_zh-CN%7Clang_zh-TW'}, //{code: 'zh-CN', label: 'Search Chinese (Simplified) pages', value: 'lang_zh-CN'}, //{code: 'zh-TW', label: 'Search Chinese (Traditional) pages', value: 'lang_zh-TW'}, ]; const LANGUAGEQUERY = /(\?|&)(lr)=([^&]+)/; const RETRY = 10; let site = { targets: { as: () => $$('#hdtbMenus a[href^="/"]'),/* possible anchors for language selector links */ }, get: { languageList: () => {/* list parent including any language, Japanese,... */ if(LANGUAGEQUERY.test(location.href) === false){/* not filtered yet */ const option = Array.from(elements.as).find(a => a.href.includes('&lr='));/* such as "Japanese" link */ if(option === undefined) return error('Not found: language option'); const languageList = option.parentNode.parentNode.parentNode; const anyLanguageText = languageList.firstElementChild.textContent.trim();/* must be "any language" */ Storage.save('anyLanguageText', anyLanguageText); return languageList; } else{ const anyLanguageText = Storage.read('anyLanguageText'); if(!anyLanguageText) return error('Not saved yet: anyLanguageText'); const option = Array.from(elements.as).find(a => a.textContent.includes(anyLanguageText)); const languageList = option.parentNode.parentNode.parentNode; return languageList; } }, languageData: (li) => { let a = li.querySelector('a[href]'), url = a ? a.href : location.href; let match = url.match(LANGUAGEQUERY); if(match === null) return log('LANGUAGEQUERY doesn\'t match.', url); return { code: match[3].replace(/lang_/g, '').replace(/%7C/g, '_'), label: li.textContent, value: match[3], }; }, listItem: (languageList, languageData) => { let a = languageList.querySelector('a[href]'); if(a === null) return log('a[href] doesn\'t exist.'); let url = [a.href, location.href].find(href => LANGUAGEQUERY.test(href)); if(url === undefined) return log('URL doesn\'t match.'); let li = a.parentNode.parentNode.cloneNode(true), lia = li.querySelector('a[href]'); li.id = SCRIPTID + '-' + languageData.code; li.dataset.value = languageData.value; lia.href = url.replace(LANGUAGEQUERY, `$1$2=${languageData.value}`); lia.textContent = li.dataset.label = languageData.label; return li; } }, is: { reset: () => location.href.includes(RESET), }, }; let html, elements = {}, timers = {}, sizes = {}; let languagesData = []; let core = { initialize: function(){ html = document.documentElement; html.classList.add(SCRIPTID); core.ready(); core.addStyle(); }, ready: function(){ core.getTargets(site.targets, RETRY).then(() => { log("I'm ready."); core.readLanguages(); core.getLanguages(); core.addLanguages(); }); }, readLanguages: function(){ /* read the saved preferences */ if(site.is.reset()){ languagesData = LANGUAGES; alert(`${SCRIPTNAME} has reset.`); }else{ languagesData = Storage.read('languagesData') || LANGUAGES; } }, getLanguages: function(){ let languageList = elements.languageList = site.get.languageList(); /* add dataset for each list items */ Array.from(languageList.children).forEach((li, i) => { if(i === 0) return;/*any language*/ let languageData = site.get.languageData(li); li.dataset.code = languageData.code; li.dataset.label = languageData.label; li.dataset.value = languageData.value; /* get default languages */ if(languagesData.find(l => l.code === languageData.code)) return; languagesData.splice(i - 1, 0, languageData);/*keep the order of the languages*/ }); /* get and update localized labels */ languagesData.forEach(languageData => { let li = Array.from(languageList.children).find(li => li.dataset.code === languageData.code); if(li) languageData.label = li.dataset.label; }); Storage.save('languagesData', languagesData); }, addLanguages: function(){ let languageList = elements.languageList; languagesData.forEach((languageData, i) => { if(Array.from(languageList.children).some(li => li.dataset.code === languageData.code)) return; let li = site.get.listItem(languageList, languageData); languageList.insertBefore(li, languageList.children[i + 1]); }); }, getTarget: function(selector, retry = 10, interval = 1*SECOND){ const key = selector.name; const get = function(resolve, reject){ let selected = selector(); if(selected === null || selected.length === 0){ if(--retry) return log(`Not found: ${key}, retrying... (${retry})`), setTimeout(get, interval, resolve, reject); else return reject(new Error(`Not found: ${selector.name}, I give up.`)); }else{ if(selected.nodeType === Node.ELEMENT_NODE) selected.dataset.selector = key;/* element */ else selected.forEach((s) => s.dataset.selector = key);/* elements */ elements[key] = selected; resolve(selected); } }; return new Promise(function(resolve, reject){ get(resolve, reject); }); }, getTargets: function(selectors, retry = 10, interval = 1*SECOND){ return Promise.all(Object.values(selectors).map(selector => core.getTarget(selector, retry, interval))); }, addStyle: function(name = 'style'){ if(core.html[name] === undefined) return; let style = createElement(core.html[name]()); document.head.appendChild(style); if(elements[name] && elements[name].isConnected) document.head.removeChild(elements[name]); elements[name] = style; }, html: { style: () => ` <style type="text/css"> [id^="${SCRIPTID}"]:active, [id^="${SCRIPTID}"]:hover{ background-color: rgba(0,0,0,.1); } [id^="${SCRIPTID}"] a{ color: #0c0c0d; } g-menu-item:not(:hover){ background-color: white !important; } </style> `, }, }; const setTimeout = window.setTimeout.bind(window), clearTimeout = window.clearTimeout.bind(window), setInterval = window.setInterval.bind(window), clearInterval = window.clearInterval.bind(window), requestAnimationFrame = window.requestAnimationFrame.bind(window), requestIdleCallback = window.requestIdleCallback.bind(window); const alert = window.alert.bind(window), confirm = window.confirm.bind(window), prompt = window.prompt.bind(window), getComputedStyle = window.getComputedStyle.bind(window), fetch = window.fetch.bind(window); if(!('isConnected' in Node.prototype)) Object.defineProperty(Node.prototype, 'isConnected', {get: function(){return document.contains(this)}}); class Storage{ static key(key){ return (SCRIPTID) ? (SCRIPTID + '-' + key) : key; } static save(key, value, expire = null){ key = Storage.key(key); localStorage[key] = JSON.stringify({ value: value, saved: Date.now(), expire: expire, }); } static read(key){ key = Storage.key(key); if(localStorage[key] === undefined) return undefined; let data = JSON.parse(localStorage[key]); if(data.value === undefined) return data; if(data.expire === undefined) return data; if(data.expire === null) return data.value; if(data.expire < Date.now()) return localStorage.removeItem(key); return data.value; } static delete(key){ key = Storage.key(key); delete localStorage.removeItem(key); } static saved(key){ key = Storage.key(key); if(localStorage[key] === undefined) return undefined; let data = JSON.parse(localStorage[key]); if(data.saved) return data.saved; else return undefined; } } const $ = function(s, f){ let target = document.querySelector(s); if(target === null) return null; return f ? f(target) : target; }; const $$ = function(s){return document.querySelectorAll(s)}; const createElement = function(html = '<span></span>'){ let outer = document.createElement('div'); outer.innerHTML = html; return outer.firstElementChild; }; const observe = function(element, callback, options = {childList: true, attributes: false, characterData: false, subtree: false}){ let observer = new MutationObserver(callback.bind(element)); observer.observe(element, options); return observer; }; const log = function(){ if(typeof DEBUG === 'undefined') return; console.log(...log.build(new Error(), ...arguments)); }; log.build = function(error, ...args){ let l = log.last = log.now || new Date(), n = log.now = new Date(); let line = log.format.getLine(error), callers = log.format.getCallers(error); //console.log(error.stack); return [SCRIPTID + ':', /* 00:00:00.000 */ n.toLocaleTimeString() + '.' + n.getTime().toString().slice(-3), /* +0.000s */ '+' + ((n-l)/1000).toFixed(3) + 's', /* :00 */ ':' + line, /* caller.caller */ (callers[2] ? callers[2] + '() => ' : '') + /* caller */ (callers[1] || '') + '()', ...args ]; }; log.formats = [{ name: 'Firefox Scratchpad', detector: /MARKER@Scratchpad/, getLine: (e) => e.stack.split('\n')[1].match(/([0-9]+):[0-9]+$/)[1], getCallers: (e) => e.stack.match(/^[^@]*(?=@)/gm), }, { name: 'Firefox Console', detector: /MARKER@debugger/, getLine: (e) => e.stack.split('\n')[1].match(/([0-9]+):[0-9]+$/)[1], getCallers: (e) => e.stack.match(/^[^@]*(?=@)/gm), }, { name: 'Firefox Greasemonkey 3', detector: /\/gm_scripts\//, getLine: (e) => e.stack.split('\n')[1].match(/([0-9]+):[0-9]+$/)[1], getCallers: (e) => e.stack.match(/^[^@]*(?=@)/gm), }, { name: 'Firefox Greasemonkey 4+', detector: /MARKER@user-script:/, getLine: (e) => e.stack.split('\n')[1].match(/([0-9]+):[0-9]+$/)[1] - 500, getCallers: (e) => e.stack.match(/^[^@]*(?=@)/gm), }, { name: 'Firefox Tampermonkey', detector: /MARKER@moz-extension:/, getLine: (e) => e.stack.split('\n')[1].match(/([0-9]+):[0-9]+$/)[1] - 2, getCallers: (e) => e.stack.match(/^[^@]*(?=@)/gm), }, { name: 'Chrome Console', detector: /at MARKER \(<anonymous>/, getLine: (e) => e.stack.split('\n')[2].match(/([0-9]+):[0-9]+\)?$/)[1], getCallers: (e) => e.stack.match(/[^ ]+(?= \(<anonymous>)/gm), }, { name: 'Chrome Tampermonkey', detector: /at MARKER \(chrome-extension:.*?\/userscript.html\?name=/, getLine: (e) => e.stack.split('\n')[2].match(/([0-9]+):[0-9]+\)?$/)[1] - 1, getCallers: (e) => e.stack.match(/[^ ]+(?= \(chrome-extension:)/gm), }, { name: 'Chrome Extension', detector: /at MARKER \(chrome-extension:/, getLine: (e) => e.stack.split('\n')[2].match(/([0-9]+):[0-9]+\)?$/)[1], getCallers: (e) => e.stack.match(/[^ ]+(?= \(chrome-extension:)/gm), }, { name: 'Edge Console', detector: /at MARKER \(eval/, getLine: (e) => e.stack.split('\n')[2].match(/([0-9]+):[0-9]+\)$/)[1], getCallers: (e) => e.stack.match(/[^ ]+(?= \(eval)/gm), }, { name: 'Edge Tampermonkey', detector: /at MARKER \(Function/, getLine: (e) => e.stack.split('\n')[2].match(/([0-9]+):[0-9]+\)$/)[1] - 4, getCallers: (e) => e.stack.match(/[^ ]+(?= \(Function)/gm), }, { name: 'Safari', detector: /^MARKER$/m, getLine: (e) => 0,/*e.lineが用意されているが最終呼び出し位置のみ*/ getCallers: (e) => e.stack.split('\n'), }, { name: 'Default', detector: /./, getLine: (e) => 0, getCallers: (e) => [], }]; log.format = log.formats.find(function MARKER(f){ if(!f.detector.test(new Error().stack)) return false; //console.log('////', f.name, 'wants', 0/*the exact line number here*/, '\n' + new Error().stack); return true; }); const error = function(){ if(typeof DEBUG === 'undefined') return; let body = Array.from(arguments).join(' '); if(error.notifications[body]) return; Notification.requestPermission(); error.notifications[body] = new Notification(SCRIPTNAME, {body: body}); error.notifications[body].addEventListener('click', function(e){ Object.values(error.notifications).forEach(n => n.close()); error.notifications = {}; }); console.error(...log.build(new Error(), ...arguments)); }; error.notifications = {}; core.initialize(); if(window === top && console.timeEnd) console.timeEnd(SCRIPTID); })();