// ==UserScript== // @name Google Search Extra Buttons // @name:ru GoogleSearchExtraButtons // @description Add buttons (last 1/2/3 days, weeks, PDF search etc.) for Google search page // @description:ru Кнопки вариантов поиска для страницы поиска Google (1-2-3 дня, недели, PDF, ...) // @version 20.2016.4.12 // @namespace spmbt.github.com // @include http://www.google.*/search* // @include https://www.google.*/search* // @include https://www.google.*/* // @include https://encrypted.google.*/search* // @include https://encrypted.google.*/* // @include https://spmbt.github.io/googleSearchExtraButtons/saveYourLocalStorage.html // @downloadURL none // ==/UserScript== if(location.host=='spmbt.github.io'){ window.addEventListener('message', function(ev){ if(/^https?:\/\/www\.google\./.test(ev.origin)){ var d = typeof ev.data =='string' && ev.data[0] =='{' ? JSON.parse(ev.data) : ev.data; if(!d.do) return; var tok = d.tok, key = d.key; switch(d.do){ case 'set': var prev = localStorage[key]; if(d.val !==undefined) localStorage[key] = JSON.stringify(d.val); else localStorage.removeItem(key); break; case 'get': prev = localStorage[key]; prev = prev === undefined || typeof prev =='string'&& prev[0] !='{'? prev : JSON.parse(prev); break; case 'remove': prev = localStorage[key]; if(prev !==undefined) localStorage.removeItem(key); } //console.log('[io]', tok, 'prev=', prev); //ev.source.postMessage(JSON.stringify(prev !==undefined ? {tok: tok, prev: prev} : {tok: tok, undef:1}), ev.origin); }},!1); }else (function(setts){ //lang, sites, lastHoursLess var $x = function(el, h){if(h) for(var i in h) el[i] = h[i]; return el;} //===extend=== ,$pd = function(ev){ev.preventDefault();} ,$q = function(q, el){return (el||document).querySelector(q)} ,lh = location.href ,d = document ,$e = function(g){ //===create or use existing element=== //g={el|clone,cl,ht,cs,at,atRemove,on,apT} g.el = g.el || g.clone ||'DIV'; var o = g.o = g.clone && g.clone.cloneNode && g.clone.cloneNode(!0) || (typeof g.el =='string' ? d.createElement(g.el) : g.el); if(o){ //execute if exist if(g.cl) o.className = g.cl; if(g.clAdd) o.classList.add(g.clAdd); if(g.cs) $x(o.style, g.cs); if(g.ht || g.at){ var at = g.at ||{}; if(g.ht) at.innerHTML = g.ht;} if(at) for(var i in at){ if(i=='innerHTML') o[i] = at[i]; else o.setAttribute(i, at[i]);} if(g.atRemove) for(var i in g.atRemove) o.removeAttribute(g.atRemove[i]); if(g.on) for(var i in g.on) if(g.on[i]) o.addEventListener(i, g.on[i],!1); g.ap && o.appendChild(g.ap); g.apT && g.apT.appendChild(o); } return o; }, addRules = function(css){ var heads = d.getElementsByTagName('head') ,node = d.createElement('style'); heads.length && heads[0].appendChild(node); node.appendChild(d.createTextNode(css)); }; /** * check occurrence of third-party event with growing interval * @constructor * @param{Number} h.t start period of check * @param{Number} h.i number of checks * @param{Number} h.m multiplier of period increment * @param{Function} h.check event condition * @param{Function} h.occur event handler */ var Tout = function(h){ var th = this; (function(){ if((h.dat = h.check() )) //wait of positive result, then occurs h.occur(); else if(h.i-- >0) //next slower step th.ww = setTimeout(arguments.callee, (h.t *= h.m) ); })(); } //for xLocStor: ,xLocStorOrigin = d.location.protocol +'//spmbt.github.io' ,qr, qrs ={} //set of queries "key-calls" (ок, toutLitt, toutLong, noService, noStorage) ,qrI = 0 //queries counter ,qrN = 12 //max number of waiting queries ,errIMax = 120, errNMax = errIMax //max number of errors ,ns ='googXButtons_' //namespace for keys ,listenMsg /** * external localStorage for using another domain if current domain storage is erased anywhere * @param{String} h.do - action: set|get|remove * @param{String} h.key * @param{Object|undefined} h.val (any type) * @param{Number|undefined} h.toutLitt * @param{Number|undefined} h.tout * @param{Function} h.cB - callback with 2 arguments * @param{Function|undefined} h.err - callback for err with one argument */ ,xLocStor = function(h){ var h0 = h; h.toutLitt = h.toutLitt || 400; h.tout = h.tout || 4000; var ifr = d.getElementById('xLocStor') ,query = function(){ if((qrI += 1) > qrN){ xCatch('longQrs', null, h); return;} ifr.contentWindow.postMessage(JSON.stringify($x({ do: h.do ,tok: token ,key: ns + h.key }, h.val !==undefined ? {val: h.val}:{}) ) , xLocStorOrigin); qrs[token] = $x({ //for wait of response wToutLitt: (function(h, qrI, errIMax){return setTimeout(function(){ qrI -= 1; if((errIMax -= 1) >=0) ;//console.warn('toutLitt', h); chkErrMax(); }, h.toutLitt);})(h, qrI, errIMax) ,wTout: (function(h, qrI){return setTimeout(function(){ qrI -= 1; //xCatch('tout', null, h); //xLocStor(h0); }, h.tout);})(h, qrI) }, h); } ,token = +new Date() + (Math.random()+'').substr(1,8) ,el = h.el; delete h.el; if(ifr) query(); else ifr = $e({ el: 'iframe', at:{id: 'xLocStor' ,src: xLocStorOrigin +'/googleSearchExtraButtons/saveYourLocalStorage.html'}, cs: {display: 'none'}, on: {load: query}, apT: el || d.body }); if(!listenMsg) addEventListener('message', function(ev){ if(ev.origin == xLocStorOrigin){ // {"tok":""[,"err":""],"h":...} //console.log('from_io', JSON.parse(ev.data)) var resp = ev.data && ev.data[0] =='{' && JSON.parse(ev.data); if(!resp) xCatch('bad_format', resp, h); if(( qr = qrs[resp.tok] )){ qrI -= 1; qr.cB(resp.prev, resp.undef); var er = qr.err; delete qrs[resp.tok];} // else ignore unsufficient token if(resp.err && (!er || er(resp.err)) ) //individual or common error processing depends of er() xCatch(resp.err, resp, h); }},!1); listenMsg =1; }, //for tests: localStorage.googXButtons_dwmyh = JSON.stringify({h:[1,2,1,1,1]}) //$('#xLocStor').contentWindow.postMessage('{"do":"get","key":"googXButtons_dwmyh"}','https://spmbt.github.io') xCatch = function(er, resp, h){ if((errIMax -= 1) >=0) console.error('tok:', resp && resp.tok ||'--','; err:', er,'; h:', h,'; respH:', resp && resp.h); chkErrMax(); }, chkErrMax = function(){if(!errIMax) console.error('Too many err messages:', errNMax)} ,$l ={ru:{ 'search in PDF files':'поиск по документам PDF' ,'search in':'искать по' ,'from / to':'за период' ,'last':['за последний','за последние','за последнюю'] ,'day':'сутки' ,'days':['дня','дней'] ,'week':'неделю' ,'weeks':['недели','недель'] ,'month':'месяц' ,'months':['месяца','месяцев'] ,'year':'год' ,'years':['года','лет'] ,'hour':'час' ,'hours':['часа','часов'] ,'Settings':'Настройки' ,'of userscript':'юзерскрипта' ,'reload page for effect':'перезагрузить страницу' ,'Interface language':'Язык интерфейса' ,'Less positions at the end of selects':'Меньше выбора в конце селектов' ,'Sites':'Сайты' },fr:{ 'search in PDF files':'la recherche dans les fichiers PDF' ,'search in':'rechercher dans' ,'from / to':'pour la période' ,'last':['le dernier','dans les derniers','dans les derniers'] ,'day':'jour' ,'days':['jours','jours'] ,'week':'semaine' ,'weeks':['semaines','semaines'] ,'month':'mois' ,'months':['mois','mois'] ,'year':'an' ,'years':['ans','ans'] ,'hour':'heure' ,'hours':['heures','heures'] ,'Settings':'Paramètres' ,'of userscript':'de Userscript' ,'reload page for effect':'recharger la page pour effet' ,'Interface language':'Langue de l\'interface' ,'Less positions at the end of selects':'Moins de choix les longues listes' ,'Sites':'Les sites' },de:{ 'search in PDF files':'Suche in PDF-Dateien' ,'search in':'Suche in' ,'from / to':'im Zeitraum' ,'last':['letzte','letzte','letzte'] ,'day':'Tag' ,'days':['Tage','Tage'] ,'week':'Woche' ,'weeks':['Wochen','Wochen'] ,'month':'Monat' ,'months':['Monate','Monate'] ,'year':'Jahr' ,'years':['Jahre','Jahre'] ,'hour':'Stunde' ,'hours':['Stunden','Stunden'] ,'Settings':'Einstellungen' ,'of userscript':'von Userscript' ,'reload page for effect':'Seite neu laden' ,'Interface language':'Sprache' ,'Less positions at the end of selects':'Weniger Auswahl in langen Listen' ,'Sites':'Websites' },es:{ 'search in PDF files':'búsqueda en archivos PDF' ,'search in':'busca en' ,'from / to':'para el período' ,'last':['el último','en los últimos','en los últimos'] ,'day':'día' ,'days':['días','días'] ,'week':'Semana' ,'weeks':['semanas','semanas'] ,'month':'mes' ,'months':['meses','meses'] ,'year':'año' ,'years':['años','años'] ,'hour':'hora' ,'hours':['horas','horas'] ,'Settings':'Ajustes' ,'of userscript':'de userscript' ,'reload page for effect':'página para efecto de recargar' ,'Interface language':'Idioma de interfaz' ,'Less positions at the end of selects':'Menos elección en listas largas' ,'Sites':'Sitios' }}; //if !lang, then no hints addRules('.hp .sfsbc,.sfsbc{display: inline-block}.siteList:hover button{display: block}' +'.gb_Ib >.gb_e{height:47px}.gb_Fb{z-index:1087}.tsf-p{z-index:203}' +'.lsbb .xButt,.lsbb >.siteList{z-index: 2002; width:34px; height:17px; padding:0 2px; line-height:14px;' +'font-size:14px; border:1px solid transparent; background-color:#4889f1; color:#fff; opacity: 0.64}' +'.lsbb >.siteList{width:32px; height:auto; padding:1px 0 2px; text-align:center}' +'.lsbb >.siteList .lsb{font-weight: normal; color:#d4d4d4}.lsbb .lsb:hover{opacity: 1; color:#fff}' +'.siteList .sett .txt{padding: 0 2px}' +'.siteList .settIn{display: none; width: 250px; padding: 2px 4px; text-align:left; border:1px solid #48f;' +'background-color:#eef; color:#336}' +'.siteList .settIn hr{margin:2px 0}' +'.sbibtd .sfsbc .nojsb, .siteList .sett:hover .settIn, .siteList .settIn.changed,' +'.siteList .settIn.changed .reload{display: block}.siteList .settIn .reload, .siteList.hiddn{display: none}' +'div.gb_g[aria-label="promo"]{display: none}'); //hide promo xLocStor({do:'get', key:'sett', val:setts, cB: function(prev,undef){ S = prev || setts; S.dwmyh = S.dwmyh || setts.dwmyh; //temp. transitional expr. console.timeStamp = function(){}; new Tout({t:120, i:8, m: 1.6 ,check: function(){ return d && d.getElementsByName('q') && d.getElementsByName('q')[0]; }, occur: function(){ var lang = S.lang != null && S.lang || setts.lang ,sites = S.sites && (S.sites.length && S.sites[0] || S.sites.length >1) && S.sites || typeof sites =='string'&& [sites] || !S.sites && setts.sites || null; var strSites = sites && sites.join('\n').replace(/^\n/,'\n\n') ||'' ,$L = $l[lang] || $l.ru; //default template of lang if(!lang || !$l[lang] || lang =='en') for(var l in $L){ //replace 'en' lang for default or substitution if($L[l] instanceof Array) for(var l2 in $L[l]) $L[l][l2] = l; else $L[l] = l; } var $LSettings = $L['Settings']; if(sites && sites.length) sites.push($LSettings) var mainPg = /\/search\?/.test(lh) ,inputSearch = this.dat ,buttSearch = d.getElementsByName("btnG") && d.getElementsByName('btnG')[0] ,buttS ={ PDF:{url:'filetype:pdf', txt:$L['search in PDF files']} ,site:{url:'site:'+ S.sites[0], txt:$L['search in']+' '+ S.sites[0], one:'day'} //you may comment this line ,'.. : ..':{url:'', txt:$L['from / to']} ,'1D':{url:'&tbs=qdr:d', txt:$L['last'][1] +' '+ $L['day'], one:'day', up:13,lett:'D'} ,'1W':{url:'&tbs=qdr:w', txt:$L['last'][2] +' '+ $L['week'], one:'week', up:14,lett:'W'} ,'1M':{url:'&tbs=qdr:m', txt:$L['last'][0] +' '+ $L['month'], one:'month', up:20,lett:'M'} ,'1Y':{url:'&tbs=qdr:y', txt:$L['last'][0] +' '+ $L['year'], one:'year', up:10,lett:'Y'} ,'1H':{url:'&tbs=qdr:h', txt:$L['last'][0] +' '+ $L['hour'], one:'hour', up:23,lett:'H'} ,DOC:{url:'filetype:doc', txt:$L['search in PDF files'].replace(/PDF/,'DOC')} }, ii = 0, iD = -1; !sites && delete buttS.site; buttSearch.parentNode.style.position ='relative'; if(buttSearch && top == self) for(var i in buttS) if(i !='site'|| S.sites){ //buttons under search input line if(i.length ==2) iD++; else iD=-1; var bI = buttS[i] ,Gesch = ({m:'letzter',f:'letzte',n:'letztes'})['m,f,m,n,f'.split(',')[iD]] ,hint = function(j){return (j+1) +' '+ (j % 10 || j==10 ? $L[bI.one +'s'][j % 10 <4 && (j/10|0)!=1 ?0:1] : $L[bI.one]) } ,butt2 = $e({clone: i =='site'|| i.length ==2 ? $e({cl: 'siteList', cs: {cursor:'default'}, at: {site: S.sites[0], date: bI.url} }) : i !='.. : ..'|| mainPg ? buttSearch : $e({cl: 'siteList hiddn'}) ,clAdd:'xButt' ,atRemove: ['id', 'name'] ,at: {value: iD !=-1 && S.dwmyh[iD] !=1 ? S.dwmyh[iD] + bI.lett : i ,innerHTML: '' +(iD !=-1 && S.dwmyh[iD] !=1 ? S.dwmyh[iD] + bI.lett : i) +''} ,cs: {position: 'absolute', top: '33px', left: (-127 + 37 * (ii++ - (ii >2 && !mainPg))) +'px'} ,on: {click: (function(bI, i, iD){ return /PDF|DOC|site/.test(i) ? function(ev){ var doc, t = ev.target; if(t.className =='defa') saveLocStor('','','remove'); $pd(ev); if(((t.getAttribute('site') ==null && t.parentNode.getAttribute('site') ==null) || t.getAttribute('site')==$LSettings || t.parentNode.getAttribute('site')==$LSettings) && !/PDF|DOC/.test(t.getAttribute('value'))) return; //console.log('clic:',i,bI,ev, t.className, inputSearch.value, this.form) inputSearch.value = inputSearch.value.replace(/ site\:[\w.]*$/ig, '') .replace(/( |\+|&as_)filetype(:|%3A)[^\&]*/g,'') +' '+ (/PDF|DOC/.test(i) ? bI.url : 'site:'+ (t.getAttribute('site')|| t.parentNode.getAttribute('site')||'')); if(/xButt|txt/.test(t.className) && !(i=='site' && !(/list/.test(t.parentNode.className) || /list/.test(t.parentNode.parentNode.className)))) buttSearch.click(); }: !bI.url ? function(ev){ //from-to date var el = $q('#cdrlnk'), o; el && el.dispatchEvent(((o = d.createEvent('Events')).initEvent('click', !0, !1), o)); $pd(ev); }: function(ev){ //last interval location.href = '/search?q='+ encodeURIComponent(inputSearch.value) +((ev.target.getAttribute('date')||ev.target.parentNode.getAttribute('date')) + (ev.target.getAttribute('value')||ev.target.parentNode.getAttribute('value')).replace(/\D/g,'') || bI.url) +(/[&?]tbm=/.test(lh) ? '&'+/tbm=[^&]*/.exec(lh)[0]:''); //saving type of page S.dwmyh[iD] = +(ev.target.getAttribute('value')||ev.target.parentNode.getAttribute('value')).replace(/\D/g,''); $pd(ev); ev.stopPropagation(); saveLocStor(); } })(bI, i, iD), mouseover: i =='site' || i.length ==2 ? (function(bI,i){return function(ev){ clearTimeout(bI.ww); $q('.list', ev.currentTarget).style.display ='block'; }})(bI,i) :'', mouseout: i =='site' || i.length ==2 ? (function(bI,i){return function(ev){ var t = ev.currentTarget; clearTimeout(bI.ww); bI.ww = setTimeout(function(){ $q('.list',t).style.display ='none'; }, 450); }})(bI,i) :'', change: saveLocStor } ,apT: buttSearch.parentNode }); bI.el = butt2; if(i =='site' || i.length ==2){ //dropdown lists under some buttons var siteList = $e({cl:'list',cs:{display:'none'}, apT: butt2}), arr =[]; for(var j =0; j <= bI.up -1 -(i=='1W'&& S.lastHoursLess ?4:0) -(i=='1M'&& S.lastHoursLess ?9:0); j++) if(i !='1H' || !S.lastHoursLess || j < 8 || j % 2 ) arr.push(hint(j)); //console.log(S.sites,i, S.dwmyh); var list = i == 'site' ? sites||[] : i =='1D'&& !sites ? arr.concat([$LSettings]) : arr; for(var j in list) if(j !=0 || iD!=-1 && S.dwmyh[iD] !=1) var sI = list[j] ,butt3 = $e({clone: sI==$LSettings ? $e({cl: 'sett lsb'}) : buttSearch ,clAdd:'xButt' ,atRemove:['id','name'] ,at:{value: sI ,site: sI ,date: bI.url ,title: sI==$LSettings || !lang ?'':(i =='site' ? $L['search in'] +' '+ sI : j==0 ? bI.txt : $L['last'][1] +' '+ sI).replace(/letzte/,Gesch) ,innerHTML:''+ sI +''+ (sI != $LSettings &&!(!S.sites && i =='1H') ?'':'
' +$L.Settings +' '+ $L['of userscript'] +'
"Google Search Extra Buttons"
' +$L['Interface language'] +':
' +'' +'
' +'Default settings' +$L.Sites +':

' +'' + $L['reload page for effect'] +'' +'
')} ,cs: {position: sI != $LSettings ?'static':'absolute',display:'block', width:'auto', height: sI != $LSettings ?'18px':'16px' ,margin:'2px 0 -1px -13px', padding:0, textAlign:'left', fontWeight:'normal', opacity:1} ,on:{click: function(ev){ //console.log('c3',ev.target.outerHTML); var chk = $q('#hoursLess'); if(chk && /hoursLess/.test(ev.target.id)){ chk.outerHTML = ''; saveLocStor(); } $pd(ev);}} ,apT: siteList }); siteList.style.height ='auto'; siteList.style.textAlign ='center'; } } } }); }, el: d.body}); var saveLocStor = function(ev, val, do2){ var aaa,aab,aac, t = ev && ev.target.form || document.documentElement || document.body; xLocStor({do: do2 ||'set', key:'sett' , val:{lang: (aaa=d.querySelectorAll('.lang', t))[aaa.length-1].value ,sites: (aab=d.querySelectorAll('.sites', t))[aab.length-1].value.replace(/^[ \t]*|[ \n\t]*$/g,'') .split('\n') ,lastHoursLess: (aac=d.querySelectorAll('.less', t))[aac.length-1].checked ,dwmyh: S.dwmyh || setts.dwmyh } ,cB: function(prev){ console.info('Settings are saved. prev=', prev);} }); $q('.siteList .settIn').classList.add('changed'); }; })({ //write "lang:''," to remove hints; 'en' for English hints (fr - Français, es - espagnol), 'ru' for Russian lang:''|| (navigator.languages && navigator.languages[1] || navigator.language.substr(0,2)) //='' if hide hints, or 2 letters from $l{} ,sites: [ //=array or one site in string '','slashdot.org','reddit.com','techcrunch.com','habrahabr.ru','geektimes.ru' ,'smashingmagazine.com','engadget.com'] //write your favorite sites ,lastHoursLess: 1 //=boolean - not show odd some values of hours after 8 h ,dwmyh: [1,1,1,1,1] //=array of numbers - current vals of days, weeks, months, years, hours });