您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
增强Google搜索功能:扩展时间范围选项、自动展开工具菜单、多语言支持
// ==UserScript== // @name Google 搜索增强套件 // @name:en Google Search Enhancement Suite // @name:ja Google検索拡張ツール // @name:zh-CN Google 搜索增强套件 // @name:zh-TW Google 搜尋增強套件 // @namespace https://github.com/your-github-username // @version 3.3.1 // @description 增强Google搜索功能:扩展时间范围选项、自动展开工具菜单、多语言支持 // @description:en Enhance Google Search: Extended time ranges, auto-expand tools menu, multi-language support // @description:ja Google検索の拡張機能:期間指定オプションの追加、ツールメニューの自動展開、多言語対応 // @description:zh-CN 增强Google搜索功能:扩展时间范围选项、自动展开工具菜单、多语言支持 // @description:zh-TW 增強Google搜索功能:擴展時間範圍選項、自動展開工具菜單、多語言支持 // @author Hu3rror // @license MIT // @match https://www.google.com/search* // @icon https://www.google.com/favicon.ico // @grant none // @homepageURL https://github.com/hu3rror/my-userscript // @supportURL https://github.com/hu3rror/my-userscript/issues // ==/UserScript== (function(){ const SCRIPTID = 'GoogleSearchVariousRanges'; const SCRIPTNAME = 'Google Search Various Ranges'; const DEBUG = false; 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 LANGS = ['en', 'ja', 'fr', 'ru', 'zh-CN', 'zh-TW', 'es', 'ar']; const RANGES = { "qdr:h": { "qdr:h": ["Past hour", "1 時間以内", "Moins d'une heure", "За час", "过去 1 小时内", "過去 1 小時內", "Última hora", "آخر ساعة"], "qdr:h2": ["Past 2 hours", "2 時間以内", "Moins de 2 heures", "За 2 часа", "过去 2 小时内", "過去 2 小時內", "Últimas 2 horas", "آخر ساعتين"], "qdr:h12": ["Past 12 hours", "12 時間以内", "Moins de 12 heures", "За 12 часов", "过去 12 小时内", "過去 12 小時內", "Últimas 12 horas", "آخر ١٢ ساعة"], }, "qdr:d": { "qdr:d": ["Past day", "1 日以内", "Moins d'un jour", "За 1 дня", "过去 1 天内", "過去 1 天內", "Último 1 día", "آخر 24 ساعة"], "qdr:d2": ["Past 2 days", "2 日以内", "Moins de 2 jours", "За 2 дня", "过去 2 天内", "過去 2 天內", "Últimos 2 días", "آخر يومين"], "qdr:d3": ["Past 3 days", "3 日以内", "Moins de 3 jours", "За 3 дня", "过去 3 天内", "過去 3 天內", "Últimos 3 días", "آخر ٣ أيام"], }, "qdr:w": { "qdr:w": ["Past week", "1 週間以内", "Moins d'une semaine", "За неделю", "过去 1 周内", "過去 1 週內", "Última semana", "آخر أسبوع"], "qdr:w2": ["Past 2 weeks", "2 週間以内", "Moins de 2 semaines", "За 2 недели", "过去 2 周内", "過去 2 週內", "Últimas 2 semanas", "آخر أسبوعين"], }, "qdr:m": { "qdr:m": ["Past month", "1 か月以内", "Moins d'un mois", "За месяц", "过去 1 个月内", "過去 1 個月內", "Último mes", "آخر شهر"], "qdr:m2": ["Past 2 months", "2 か月以内", "Moins de 2 mois", "За 2 месяца", "过去 2 个月内", "過去 2 個月內", "Últimos 2 meses", "آخر شهرين"], "qdr:m3": ["Past 3 months", "3 か月以内", "Moins de 3 mois", "За 3 месяца", "过去 3 个月内", "過去 3 個月內", "Últimos 3 meses", "آخر ٣ شهور"], "qdr:m6": ["Past 6 months", "6 か月以内", "Moins de 6 mois", "За 6 месяца", "过去 6 个月内", "過去 6 個月內", "Últimos 6 meses", "آخر ٦ شهور"], }, "qdr:y": { "qdr:y": ["Past year", "1 年以内", "Moins d'une an", "За год", "过去 1 年内", "過去 1 年內", "Último año", "آخر سنة"], "qdr:y2": ["Past 2 years", "2 年以内", "Moins de 2 ans", "За 2 года", "过去 2 年内", "過去 2 年內", "Últimos 2 años", "آخر سنتين"], "qdr:y5": ["Past 5 years", "5 年以内", "Moins de 5 ans", "За 5 года", "过去 5 年内", "過去 5 年內", "Últimos 5 años", "آخر ٥ سنوات"], }, }; const LANGUAGES = { "lang_en": ["English", "英語", "Anglais", "английский", "英语", "英語", "Inglés", "الإنجليزية"], "lang_ja": ["Japanese", "日本語", "Japonais", "японский", "日语", "日語", "Japonés", "اليابانية"], "lang_zh-TW": ["Traditional Chinese", "繁体中文", "Chinois traditionnel", "традиционный китайский", "繁体中文", "繁體中文", "Chino tradicional", "الصينية التقليدية"], }; const PERIODS = []; const site = { targets: { tools: () => document.querySelector('#hdtb-tls'), // 新增工具按钮选择器 rangeAnchor: () => (location.href.includes('qdr:h')) ? $('a[href*="qdr:d"]') : $('a[href*="qdr:h"]'), langAnchor: () => $('a[href*="lr=lang_"]'), }, hiddenTargets: { dropdown: () => $('#hdtbMenus'), listParent: () => elements.rangeList ? elements.rangeList.parentNode : null, langListParent: () => elements.langList ? elements.langList.parentNode : null, }, get: { index: () => { const lang = document.documentElement.lang; if (!lang) return 0; const langCode = lang.split('-')[0]; const fullLangCode = lang; if (fullLangCode === 'zh-TW') return LANGS.indexOf('zh-TW'); if (fullLangCode === 'zh-CN' || langCode === 'zh') return LANGS.indexOf('zh-CN'); return LANGS.indexOf(langCode) !== -1 ? LANGS.indexOf(langCode) : 0; }, rangeRow: (rangeAnchor) => rangeAnchor.parentNode.parentNode, rangeList: (rangeRow) => rangeRow.parentNode, langRow: (langAnchor) => langAnchor.parentNode.parentNode, langList: (langRow) => langRow.parentNode, customRange: (rangeList) => rangeList.lastElementChild, customRangeHref: (href, from, to) => href.replace(/(qdr:)[a-z][0-9]*/, `cdr:1,cd_min:${from},cd_max:${to}`), rangeAnchors: (rangeList) => rangeList.querySelectorAll('a[href*="qdr:"]'), langAnchors: (langList) => langList.querySelectorAll('a[href*="lr=lang_"]'), cleanUrl: (url) => { let clean = url.replace(/&?lr=lang_[a-zA-Z-]+/g, ''); clean = clean.replace(/&+/g, '&').replace(/^&|&$/g, ''); return clean; }, }, }; const PADDING = 32 + 32; let elements = {}, sizes = {}, timers = {}; let core = { initialize: function(){ elements.html = document.documentElement; elements.html.classList.add(SCRIPTID); core.ready(); }, ready: function(){ if(document.hidden) return document.addEventListener('visibilitychange', core.ready, {once: true}); core.getTargets(site.targets, 40, 250).then(() => { log("I'm ready."); core.rebuildRanges(); core.addLanguages(); core.addCustomPeriods(); core.calculateWidth(); core.autoClickTools(); // 新增自动点击功能 }).catch(e => { console.error(`${SCRIPTID}:`, e); }); }, // 新增自动点击方法 autoClickTools: function(){ timers.expand = setInterval(() => { const tools = elements.tools; if(!tools) return; const activeElement = document.activeElement; if(tools.getAttribute('aria-expanded') === 'true') { return clearInterval(timers.expand); } tools.click(); activeElement.focus(); }, 250); }, rebuildRanges: function(){ const index = site.get.index(); const rangeAnchor = elements.rangeAnchor; const rangeRow = elements.rangeRow = site.get.rangeRow(rangeAnchor); const rangeList = elements.rangeList = site.get.rangeList(rangeRow); while(rangeList.children[1] !== rangeList.lastElementChild) rangeList.children[1].remove(); rangeList.children[0].dataset.selector = rangeList.children[1].dataset.selector = 'rangeRow'; Object.keys(RANGES).forEach(r => { const row = rangeRow.cloneNode(true), a = row.querySelector('a'); row.dataset.selector = 'rangeRow'; Object.keys(RANGES[r]).forEach(c => { const range = rangeAnchor.cloneNode(true); range.dataset.selector = 'rangeAnchor'; range.href = range.href.replace(/qdr:[hd]/, c); range.textContent = RANGES[r][c][index]; if(location.href.includes(c + '&')) range.dataset.selected = 'true'; a.parentNode.append(range); }); a.remove(); rangeList.lastElementChild.before(row); }); }, addLanguages: function(){ if (!elements.langAnchor) return; const index = site.get.index(); const langAnchor = elements.langAnchor; const langRow = elements.langRow = site.get.langRow(langAnchor); const langList = elements.langList = site.get.langList(langRow); const existingLangItems = langList.querySelectorAll('a[href*="lr=lang_"]'); if (existingLangItems.length === 0) return; const parentContainer = existingLangItems[0].parentNode; Object.keys(LANGUAGES).forEach(lang => { const template = existingLangItems[0].cloneNode(true); const cleanUrl = site.get.cleanUrl(langAnchor.href); template.href = cleanUrl + (cleanUrl.includes('?') ? '&' : '?') + `lr=${lang}`; template.textContent = LANGUAGES[lang][index]; template.dataset.selector = 'langAnchor'; if(location.href.includes(lang)) { template.dataset.selected = 'true'; } parentContainer.insertBefore(template, parentContainer.lastElementChild); }); }, addCustomPeriods: function(){ let customRange = site.get.customRange(elements.rangeList); for(let i = 0; PERIODS[i]; i++){ let line = document.createElement('div'); for(let key in PERIODS[i]){ let a = elements.rangeAnchor.cloneNode(true); a.href = site.get.customRangeHref(a.href, PERIODS[i][key][0], PERIODS[i][key][1]); a.textContent = key; line.appendChild(a); } customRange.parentNode.appendChild(line); } }, calculateWidth: function(){ core.getTargets(site.hiddenTargets).then(() => { if (elements.dropdown) { elements.dropdown.style.visibility = 'hidden'; elements.dropdown.style.display = 'block'; } if (elements.listParent) { elements.listParent.style.visibility = 'hidden'; elements.listParent.style.display = 'block'; } sizes.maxwidth = 0; if (elements.rangeList) { let as = site.get.rangeAnchors(elements.rangeList); for(let i = 0, a; a = as[i]; i++){ if(sizes.maxwidth < a.offsetWidth) sizes.maxwidth = a.offsetWidth; } } if (elements.langList) { let langAs = site.get.langAnchors(elements.langList); for(let i = 0, a; a = langAs[i]; i++){ if(sizes.maxwidth < a.offsetWidth) sizes.maxwidth = a.offsetWidth; } } if(sizes.maxwidth === 0) return setTimeout(core.calculateWidth, 250); if (elements.dropdown) { elements.dropdown.style.visibility = ''; elements.dropdown.style.display = ''; } if (elements.listParent) { elements.listParent.style.visibility = ''; elements.listParent.style.display = 'none'; } if (elements.langListParent) { elements.langListParent.style.visibility = ''; elements.langListParent.style.display = 'none'; } core.addStyle(); }).catch(e => { console.error(`${SCRIPTID} calculateWidth:`, e); }); }, 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 resolve(null); }else{ if(selected.nodeType === Node.ELEMENT_NODE) selected.dataset.selector = key; else selected.forEach((s) => s.dataset.selector = key); 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', d = document){ if(html[name] === undefined) return; if(d.head){ let style = createElement(html[name]()), id = SCRIPTID + '-' + name, old = d.getElementById(id); style.id = id; d.head.appendChild(style); if(old) old.remove(); } else{ let observer = observe(d.documentElement, function(){ if(!d.head) return; observer.disconnect(); core.addStyle(name); }); } }, }; const html = { style: () => ` <style type="text/css"> [data-selector="rangeRow"]:not(:first-child):not(:last-child):hover, [data-selector="rangeRow"]:not(:first-child):not(:last-child):active{ background-color: transparent; } [data-selector="rangeAnchor"], [data-selector="langAnchor"]{ display: inline-block !important; width: ${sizes.maxwidth - PADDING}px !important; padding-right: 20px !important; } [data-selector="rangeAnchor"]:hover, [data-selector="rangeAnchor"]:active, [data-selector="langAnchor"]:hover, [data-selector="langAnchor"]:active{ background-color: rgba(0,0,0,.1); } [data-selector="rangeAnchor"][data-selected="true"], [data-selector="langAnchor"][data-selected="true"]{ background-image: url(//ssl.gstatic.com/ui/v1/menu/checkmark.png); background-position: left center; background-repeat: no-repeat; } [data-selector="langAnchor"] { font-family: inherit; font-size: inherit; color: inherit; white-space: nowrap; } [data-selector="langAnchor"][data-selected="true"] { font-weight: bold; } :root[data-theme="dark"] [data-selector="rangeAnchor"]:hover, :root[data-theme="dark"] [data-selector="rangeAnchor"]:active, :root[data-theme="dark"] [data-selector="langAnchor"]:hover, :root[data-theme="dark"] [data-selector="langAnchor"]:active, .dark [data-selector="rangeAnchor"]:hover, .dark [data-selector="rangeAnchor"]:active, .dark [data-selector="langAnchor"]:hover, .dark [data-selector="langAnchor"]:active{ background-color: rgba(255,255,255,.1) !important; } :root[data-theme="dark"] g-menu-item:not(:hover), .dark g-menu-item:not(:hover){ background-color: #202124 !important; } :root[data-theme="dark"] [data-selector="rangeAnchor"][data-selected="true"], :root[data-theme="dark"] [data-selector="langAnchor"][data-selected="true"], .dark [data-selector="rangeAnchor"][data-selected="true"], .dark [data-selector="langAnchor"][data-selected="true"]{ background-image: url(//ssl.gstatic.com/ui/v1/menu/checkmark_white.png); } @media (prefers-color-scheme: dark) { .dark-mode [data-selector="rangeAnchor"]:hover, .dark-mode [data-selector="rangeAnchor"]:active, .dark-mode [data-selector="langAnchor"]:hover, .dark-mode [data-selector="langAnchor"]:active{ background-color: rgba(255,255,255,.1) !important; } .dark-mode g-menu-item:not(:hover){ background-color: #202124 !important; } .dark-mode [data-selector="rangeAnchor"][data-selected="true"], .dark-mode [data-selector="langAnchor"][data-selected="true"]{ background-image: url(//ssl.gstatic.com/ui/v1/menu/checkmark_white.png); } } </style> `, }; const $ = function(s, f = undefined){ let target = document.querySelector(s); if(target === null) return null; return f ? f(target) : target; }; const $$ = function(s, f = undefined){ let targets = document.querySelectorAll(s); return f ? f(targets) : targets; }; const createElement = function(html = '<div></div>'){ let outer = document.createElement('div'); outer.insertAdjacentHTML('afterbegin', html); return outer.firstElementChild; }; const observe = function(target, callback, options = {childList: true, subtree: true}){ const observer = new MutationObserver(callback); observer.observe(target, options); return observer; }; const log = function(){ if(typeof DEBUG === 'undefined' || !DEBUG) return; let l = log.last = log.now || new Date(), n = log.now = new Date(); let error = new Error(), line = log.format.getLine(error), callers = log.format.getCallers(error); console.log( SCRIPTID + ':', n.toLocaleTimeString() + '.' + n.getTime().toString().slice(-3), '+' + ((n-l)/1000).toFixed(3) + 's', ':' + line, (callers[2] ? callers[2] + '() => ' : '') + (callers[1] || '') + '()', ...arguments ); }; 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, 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; return true; }); core.initialize(); if(window === top) console.timeEnd(SCRIPTNAME); })();