translate.google tooltip

Translates selected text into a `tooltip' via Google translate

当前为 2015-05-10 提交的版本,查看 最新版本

  1. // ==UserScript==
  2. // @name translate.google tooltip
  3. // @namespace trespassersW
  4. // @author trespassersW
  5. // @copyright trespassersW
  6. // @license MIT
  7. // @description Translates selected text into a `tooltip' via Google translate
  8. // @include http://*
  9. // @include https://*
  10. // @include file://*
  11. // about:config -> greasemonkey.fileIsGreaseable <- true
  12. // @homepageURL https://openuserjs.org/scripts/trespassersW/translate.google_tooltip
  13. // @version 3.7.97
  14. //* This is a descendant of lazyttrick's http://userscripts.org/scripts/show/36898.
  15. // 3.7.96 2015-05-10 * TTS in ff37; * DOMparser instead of IFRAME; * bugfixes
  16. // 3.7.8.2 2015-04-26 + new country flags host
  17. // 3.7.2 2015-04-20 * TTS: alt-select text inside tooltip and [ctrl/shift]-click language icon below
  18. // * [shift] tts window in IFRAME (: only works on google.* and file://* :(
  19. // * [ctrl] tts window in new tab
  20. // 3.6.2.2 2015-04-19 * gray gradient background
  21. // 3.6.1 2015-04-17 + selectable background color
  22. // 3.5.1 2015-04-15
  23. // + TTS: alt-select text inside tooltip and shift-click language icon below
  24. // * From<->To buttons fix; * err handler
  25. // 3.4.1 2015-04-09 * GT changes: GET prohibited - use POST
  26. // 3.0.0
  27. // - national flags icons -- from www.senojflags.com
  28. // 2.3
  29. // - new editable 'source text' field
  30. // 2.2.2
  31. // - backward translation - select text inside tooltip and click the icon under your selection.
  32. // 2.2.1
  33. // - Ctrl-Alt-click removes item from the history of translations
  34. // - Ability to change translation in the history -
  35. // select desired translation in the tooltip window using ctrl or alt -
  36. // which one is checked in your settings - then click on the icon below the selection.
  37. // 2.2
  38. // - history of translations
  39. // 2.1.2
  40. // - Selected text is fetched in the moment when you hover over the icon.
  41. // So, you can select a few letters, then adjust your selection using shift + arrows.
  42. // 2.0.0d
  43. // - native GT languages list
  44. // 2.0.0c
  45. // Alt key option added
  46. // If something goes wrong:
  47. // Tools->SQLite manager-> Database-> Connect_database->
  48. // %YourBrowserProfile%\gm_scripts\translate.google_tooltip.db ->
  49. // scriptvals-> alt/ctrl <- false
  50. // 2.0.0b
  51. // - exit by ESC
  52. // - 1k letters limit -- don't strain your Google
  53. //
  54. //*/
  55. // /grant GM_addStyle
  56. // @grant GM_getValue
  57. // #grant GM_log
  58. // @grant GM_openInTab
  59. // @grant GM_setValue
  60. // @grant GM_xmlhttpRequest
  61. // @icon data:image/jpg;base64, R0lGODlhIAARALP/AAAAAP///xMYfAqf////Zv/qDuCeH8VmB8DAwAAAAAAAAAAAAAAAAAAAAAAAAAAAACH5BAEAAAgALAAAAAAgABEAQASdEMlJgb00awkMKQZYjB8RBsE4AtLQvtt0sXHcEcT1pbxK0hsXRmYIGUUgA3DiQjQtssInRAjglMsa4ibtlqSGgECQymmaAwDYUhSFfKoQDQ3LdA6Hoh6qbW4sJHpFWTUAOEA3Vj1WZjF+HQU9X18oPxl0Wx4kXoFiZF1zMEJgbW8qJnAHoU4takocpW5IIISYGh1HRlh9hRZ4eIRaEQA7
  62. //
  63. // ==/UserScript==
  64.  
  65. if(document.body){
  66.  
  67. main = function (){ "use strict";
  68.  
  69. var GTsuffix=".com"; // ".fr" ".de" ".ru" ".com"
  70. //{[ hacks
  71. var UA = navigator.userAgent;
  72. 0 && (UA="Mozilla/5.0 (Windows NT 5.1; rv:37.0) Gecko/20100101 Firefox/37.0");
  73. //]}
  74.  
  75. var GTurl= "https://translate.google"+GTsuffix+"/?";
  76. var dictURL= "https://translate.google"+GTsuffix+"/translate_a/t?client=t";
  77. var ttsURL= "https://translate.google.com/translate_tts?client=t";
  78. var version= 3790;
  79.  
  80. var HREF_NO = 'javascript:void(0)';
  81.  
  82. var llii=0, _log = function(){ /* * /
  83. for (var s=++llii +':', li=arguments.length, i = 0; i<li; i++)
  84. s+=' ' + arguments[i];
  85. console.log(s)
  86. /* */
  87. }
  88. _log("tgtt..");
  89. var URL='*'; var tURL;
  90. var GT_tl='auto';
  91. var body;
  92.  
  93. // http://www.senojflags.com/
  94. var senop="https://cdn.rawgit.com",
  95. seno="/trespassersW/UserScripts/master/Flags/",
  96. senox="index.html?gtrantoltip#",
  97. senoj=senop+seno+senox,
  98. senojflags = [seno, "http://lh/Flags/" ],
  99. senoext=".png";
  100. ;
  101. //
  102.  
  103. var res_dict='gt-res-dict'; //'gt_res_dict';
  104. var languagesGoogle, isInited=false;
  105. var rtl_langs="ar fa iw ur";
  106. var inTextArea= null;
  107. var maxHT=20, maxWC=3;
  108. var sourceBH = 3, sourceDP =10;
  109. var ht=null; // history table,
  110.  
  111. var imgForw,imgBack,imgSwap,imgUse,imgSave,imgFlags,imgForwSrc,imgBackSrc;
  112. var txtSel; // text selected
  113. var currentURL, Qtxt='***';
  114. var gt_sl_gms, gt_tl_gms, gt_sl, gt_tl;
  115.  
  116. var sT;
  117. var noMup=0;
  118. var _G = "-moz-linear-gradient",_T='transparent';
  119. var G_ ='rgba(0,0,0,.1)',W_='rgba(255,255,255,.1)';
  120. var FG={
  121. t: ['#000' ,'#000' ,'#000' ,'#000' ,'#eec' ,'#000' ,'#000' ], // text
  122. l: ['#047' ,'#047' ,'#047' ,'#052' ,'#7CF' ,'#047' ,'#670000'], // links
  123. g: ['#404040','#404040' ,'#404040','#404040','#ccb' ,'#404040','#404040'] // greyed txt
  124. };
  125. var BG={
  126. t: ["yellow" ,"grey" ,"blue" ,"green" ,"dark" ,"striped", "pink" ], // titles
  127. C: ['#FFFFE1','#D1D1D1','#D3ECEC','#C4FFC4' ,'#333' , // bg Color
  128. _G+'(-45deg, #DDD, #AAA )' ,'#FFE6E6'],
  129. T: [_G+"(to right,#FFFFE1,#DDDDAA)", _G+"(to right,#D1D1D1,#A0A097)", // buttons
  130. _G+"(to right,#D3ECED,#8CCCCE)", _G+"(to right,#C4FFC4,#6BEF69)",
  131. _G+"(to right,#777,#373737)" , _G+"(to right,#CCC, #888)",
  132. _G+"(to right,#FFE6E6,#FFC6C8)" ],
  133. H: [_T,_T,_T,_T,_T, // dictionary items
  134. _G+'(to bottom ,rgba(127,127,127,.0),rgba(127,127,127,.15))', _T],
  135. F: [G_,G_,G_,G_,W_,G_,G_], // historic phrases
  136. E: ['#F4F4E8','#EEEEEE','#E8E8F4','#E8F4E8','#777777','#DDDDDD','#FFF2F2'], // Edit box
  137. f: [function(){css(0)}, function(){css(1)}, function(){css(2)},
  138. function(){css(3)}, function(){css(4)}, function(){css(5)},
  139. function(){css(6)}]
  140. }
  141.  
  142. function mousedownCleaning(evt){
  143. var divDic = getId('divDic');
  144. var divLookup = getId('divLookup');
  145. var dU = getId('divUse');
  146. var t=evt.target;
  147. noMup=0; // patch :/
  148. if(divDic) {
  149. if(!clickedInsideID(evt.target,'divDic')){
  150. evt.preventDefault(),evt.stopPropagation();
  151. if(dU && clickedInsideID(t,'divUse')){
  152. if(clickedInsideID(t,'divGetback')) forwLookup(evt); else
  153. if(clickedInsideID(t,'divGetforw')) backLookup(evt); else
  154. if(clickedInsideID(t,'imgUse')) useClick(evt); else
  155. _log('x3 click');
  156. return;
  157. }
  158. else
  159. cleanUp('MC');
  160. } else killId(dU);
  161. }
  162. killId(divLookup);
  163. }
  164.  
  165. var documentcontentEditable=false;
  166. var documentdesignMode = '';
  167. var divExtract;
  168.  
  169. var escHnd;
  170. function setEscHnd(){
  171. if(!escHnd) escHnd=
  172. document.addEventListener('keydown', escCleanup, false);
  173. }
  174.  
  175. function cleanUp(s){
  176. _log(s);
  177. var d=getId('divSourcetext');
  178. if(d) sT= d.value;
  179. killId('divDic');
  180. killId('divExtract');
  181. killId('divLookup');
  182. killId('divUse');
  183. killId('divBack');
  184. killId('divSelflag');
  185. killId('divTtsIfr');
  186.  
  187. //divExtract='';
  188. if(documentcontentEditable)
  189. documentcontent.Editable=documentcontentEditable,
  190. documentcontentEditable = false;
  191. if(documentdesignMode == 'on')
  192. document.designMode='on',
  193. documentdesignMode=null;
  194. }
  195.  
  196. function useClick(e){
  197. killId('divUse');
  198. if(e.shiftKey) ht[0][1] += ' '+txtSel;
  199. else ht[0][1] = txtSel;
  200. GM_setValue('hist',JSON.stringify(ht));
  201. if(getId('divHist')){
  202. killId('divHist');
  203. history();
  204. }
  205. }
  206. var last_tl, last_sl;
  207. function backLookup(e){
  208. if(e.shiftKey || e.ctrlKey) {
  209. noMup=1;
  210. ttsRequest(txtSel,gt_tl, e.ctrlKey!=0);
  211. return;
  212. }
  213. killId('divUse');
  214. gtRequest(txtSel,gt_sl,gt_tl);
  215. currentURL = GTurl + "langpair=" + gt_sl + "|" + gt_tl + "&text=" + escAp(txtSel);
  216. }
  217. //GET https://translate.google.com/?langpair=en|ru&text=Varnish
  218. //POST https://translate.google.com/translate_a/t?client=t&hl=ru&sl=en&tl=ru&text=Varnish
  219. function forwLookup(e){
  220. if(e.shiftKey || e.ctrlKey) {
  221. noMup=1;
  222. ttsRequest(txtSel,gt_sl, e.ctrlKey!=0);
  223. return;
  224. }
  225. killId('divUse');
  226. var t=gt_tl; gt_tl=gt_sl; gt_sl=t;
  227. gtRequest(txtSel,gt_sl,gt_tl);
  228. currentURL = GTurl + "langpair=" + gt_sl + "|" + gt_tl + "&text=" + escAp(txtSel);
  229. }
  230. var Gctrl, Galt;
  231. var sayTip="\n[shift / ctrl] listen (";
  232. function showLookupIcon(evt){
  233. Gctrl=GM_getValue('ctrl',false), Galt=GM_getValue('alt',true);
  234. if((!evt.ctrlKey && Gctrl)
  235. ||(!evt.altKey && Galt)
  236. // to avoid collision
  237. ||(evt.ctrlKey && !Gctrl)
  238. ||(evt.altKey && !Galt)
  239. ) return;
  240. evt.preventDefault(),evt.stopPropagation();
  241.  
  242. var divDic = getId('divDic');
  243. var divLookup = getId('divLookup');
  244. txtSel = getSelection(evt.target)+'';
  245. if(txtSel.length>1024){
  246. return;
  247. }
  248. //exit if no text is selected
  249. if(!txtSel || txtSel===""){
  250. _log('S:notext ')
  251. if(divDic) {
  252. if(!clickedInsideID(evt.target,'divDic'))
  253. cleanUp('no sel');
  254. }
  255. if(divLookup){
  256. killId(divLookup);
  257. }
  258. return;
  259. }
  260. //possible cleanup
  261. if(divDic){
  262. if(!clickedInsideID(evt.target,'divDic'))
  263. {cleanUp('!divdic'); return; }
  264. // inside divDic:
  265. var dU=getId('divUse');
  266. if(dU){
  267. if(!clickedInsideId('divUse')){
  268. killId(dU);
  269. } return;
  270. }
  271. try{
  272. var p= belowCursor(evt,10,10,'r');
  273. var divUse= buildEl('div', {id:'divUse',
  274. style:'z-index:11000; border: none'+
  275. ';top:' + p.t +';left:' + p.l +';right:' + p.r +';bottom: auto;'
  276. }, null, null );
  277.  
  278. var iTo = getFlagSrc(gt_tl,'to');
  279. var divForw=buildEl('span', {id:'divGetforw', // 'class': 'gootranslink', href: HREF_NO,
  280. //border: 0, src: iTo,
  281. title: gt_sl_gms + '\u2192 '+gt_tl_gms +sayTip+gt_tl+')'},
  282. null, imgH+iTo+imgT);
  283. // ['mousedown', forwLookup], imgH+iTo+imgT);
  284. divUse.appendChild(divForw);
  285. var iFrom = getFlagSrc(gt_sl,'from');
  286. var divBack=buildEl('span', {id:'divGetback', //'class': 'gootranslink', href: HREF_NO,
  287. //border: 0, src: iFrom,
  288. title: gt_tl_gms + '\u2192 '+gt_sl_gms +sayTip+gt_sl+')'},
  289. null, imgH+iFrom+imgT);
  290. // ['mousedown', backLookup], imgH+iFrom+imgT);
  291. gt_sl !='auto' && divUse.appendChild(divBack);
  292.  
  293. addEl(divUse,'img',{id: 'imgUse', border: 0,
  294. title: 'use in history\n[shift] add to history', src: imgUse},
  295. null,null);
  296.  
  297. // tp=(evt.clientY+window.pageYOffset+30)+'px';
  298. // lf=(evt.clientX+window.pageXOffset+30)+'px';
  299. body.appendChild(divUse);
  300. }catch(e){console.log('use hist\n'+e)}
  301. return;
  302. }
  303. // inside page
  304. if(!isInited) {css(-1); isInited=true; }
  305. //remove div if exists
  306. if(divLookup)
  307. killId(divLookup);
  308. //div container
  309. p = belowCursor(evt,10,10);
  310. divLookup = buildEl('div', {id:'divLookup', style: 'z-index:10000'+
  311. ';border: none;' +
  312. ';top:' + p.t +';left:' + p.l +';right:' + p.r +';bottom: auto'
  313. }, null, null);
  314.  
  315. iTo = getFlagSrc(GM_getValue('to'),'to');
  316. var iForw=buildEl('img', {'border':0, id:"imgLookForw", style: 'padding-left: 5px',
  317. src: iTo}, ['mouseover', lookup],null);
  318. var sl=GM_getValue('from','auto');
  319. iFrom = getFlagSrc(sl,'from');
  320. var iBack=buildEl('img', {'border':0, id:"imgLookBack", style: 'padding-left: 5px',
  321. src: iFrom},
  322. ['mouseover', lookup], null);
  323. if(p.r == 'auto' ){ // left half
  324. divLookup.appendChild(iForw);
  325. sl != 'auto' && divLookup.appendChild(iBack);
  326. }else{ // right half
  327. sl != 'auto' && divLookup.appendChild(iBack);
  328. divLookup.appendChild(iForw);
  329. }
  330. body.appendChild(divLookup);
  331. }
  332. function escCleanup(e){
  333. if(!e.shiftKey && !e.ctrlKey && !e.altKey && e.keyCode==27 ){
  334. cleanUp('esc');
  335. document.removeEventListener('keydown', escCleanup,false);
  336. escHnd=null;
  337. }
  338. }
  339.  
  340. function lookup(evt){
  341. var divResult = null;
  342. var divDic = getId('divDic');
  343. var divLookup = getId('divLookup');
  344. var top = divLookup.style.top;
  345. var left = divLookup.style.left;
  346. var rite = divLookup.style.right;
  347. var txtS = txtSel; // 2012-08-20
  348. txtSel = getSelection(inTextArea? inTextArea: evt.target)+'';
  349. if(!txtSel) txtSel = txtS;
  350. if(txtSel.length>1024){
  351. return;
  352. }
  353. //exit if no text is selected
  354. if(!txtSel || txtSel==""){
  355. _log('L:notext')
  356. if(divDic) {
  357. if(!clickedInsideID(evt.target,'divDic'))
  358. killId(divDic);
  359. }
  360. killId('divLookup');
  361. killId('divDic');
  362. return;
  363. }
  364. //cleanup divs
  365. killId('divDic');
  366. killId('divLookup');
  367. //div container document.body.clientHeight/Width
  368. divDic = buildEl('div',
  369. {id:'divDic', style: 'top:'+top+';left:'+left+';right:'+rite
  370. });
  371. divDic.addEventListener('mousedown', dragHandler, false);
  372. setEscHnd();
  373. body.appendChild(divDic);
  374. // patch gmail
  375.  
  376. if(document.contentEditable)
  377. documentcontentEditable = document.contentEditable,
  378. document.contentEditable = false;
  379. if(document.designMode == 'on')
  380. documentdesignMode='on',
  381. document.designMode='off';
  382.  
  383. //div result
  384. divResult = buildEl('div',
  385. {id:'divResult'}, null, 'Loading...');
  386. divDic.appendChild(divResult);
  387. /**/
  388. // history
  389. var divBottom = buildEl('div',{id:'divBottom', align: 'bottom'},null,null);
  390. addEl(divBottom,'a',
  391. {'class':"gootransbutt gootranslink gtlPassive", id:'historyLink', title: 'Translation history',
  392. align: 'left', href:HREF_NO},
  393. ['click', history], 'History');
  394. addEl(divBottom,'a',
  395. {'class':"gootransbutt gootranslink gtlPassive", id:'sourceLink', title: 'Source', href:HREF_NO},
  396. ['click', source],'Source');
  397. //options link
  398. addEl(divBottom,'a',
  399. {'class':"gootransbutt gootranslink gtlPassive", id:'optionsLink', title: 'Settings', href:HREF_NO},
  400. ['click', options], 'Options');
  401. divDic.appendChild(divBottom);
  402. /**/
  403. //lookup
  404. gt_sl = GM_getValue('from', 'auto');
  405. GT_tl = (gt_tl = GM_getValue('to',GT_tl));
  406. if( evt.target.id== 'imgLookBack' ){
  407. var t=gt_tl; gt_tl=gt_sl; gt_sl=t;
  408. }
  409. //"http://www.google.com/translate_t?text=" + txtSel + "&langpair=" + lang;
  410. gtRequest(txtSel,gt_sl,gt_tl);
  411. }
  412.  
  413. var IFR;
  414. function eStop(e){e.preventDefault(),e.stopPropagation()}
  415. function openInFrame(url){
  416. killId('divTtsIfr');
  417. var dD=getId('divDic');
  418. var IFR=buildEl('div',{id:'divTtsIfr'},null,null);
  419. var IFH=addEl(IFR,'div',{id:'divTtsIfh'},null,null);
  420. addEl(IFH, 'span',{'class':"gootransbutt gootranslink",style: 'color:red!important;'},
  421. ['click', function(e){killId('divTtsIfr')}],'&#x2716;');
  422. addEl(IFH, 'a',{'class':"gootransbutt gootranslink", id: 'divTtsLnk',
  423. href: url, target:"_blank", title: 'play in tab'},
  424. ['click',
  425. function(e)
  426. {eStop(e);GM_openInTab(e.target.href)}
  427. ],
  428. '');
  429. addEl(IFH, 'span', {},[],deURI(url));
  430. // addEl(IFR, 'br');
  431. var BFR=
  432. addEl(IFR, 'iframe',{
  433. width: "100%", height: "48", frameborder: "0",scrolling:"auto", marginheight:"0", marginwidth:"0",
  434. style:'padding-top:3px;overflow-x:hidden;',
  435. src: 'about:blank'
  436. },
  437. null,null);
  438. insAfter(IFR,getId('divBottom'));
  439. BFR.contentWindow.location.href=url;
  440. /*
  441. */
  442. }
  443.  
  444. function ttsRequest(txt,t,e){
  445. var etxt = escAp(txt);
  446. etxt=ttsURL + "&tl=" + t + "&ie=utf-8&q=" + etxt.split(' ').slice(0,19).join(' ');
  447. _log('tts> '+etxt);
  448. if(e)
  449. GM_openInTab(etxt);
  450. else
  451. openInFrame(etxt);
  452. //GM_openInTab(etxt);
  453. // sorry, firefox' decodeAudioData() does NOT support mp3
  454. }
  455.  
  456. function gtRequest(txt,s,t){
  457. var etxt = escAp(txt);
  458. currentURL = GTurl + "langpair=" + s + "|" + t + "&text="+etxt ;
  459. etxt=GTurl + "langpair=" + s + "|" + t + "&text=" + etxt.split(' ').slice(0,9).join(' ');
  460. if( !((s==last_sl && t==last_tl) || (s==last_tl && t==last_sl)) || (divExtract=='')){
  461. var c=':';
  462. //_log(s+c+last_sl+ ' '+t+c+last_tl + ' '+ divExtract );
  463. divExtract = '';
  464. Request(etxt);
  465. }else{
  466. extractResult(null);
  467. }
  468. last_sl = s; last_tl = t;
  469. }
  470. function Request(url,cb){
  471. URL=url; _log('R: '+URL);
  472. var meth=cb? 'POST': 'GET';
  473. GM_xmlhttpRequest({
  474. method: meth,
  475. url: url,
  476. headers: {
  477. "User-Agent": UA
  478. ,"Accept": "text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8"
  479. ,"Accept-Encoding": "gzip, deflate"
  480. //,"Host": "www.google.com"
  481. },
  482. onload: function(resp) {
  483. try{
  484. if(cb)
  485. cb(resp.responseText)
  486. else
  487. extractResult(resp.responseText);
  488. }catch(e){
  489. if(getId('divResult'))
  490. getId('divResult').innerHTML =
  491. '<a id="gttpErrRef" href="#">error processing response text:</a><br>'+e;
  492. getId('gttpErrRef').href=URL.subsr(0,99);
  493. }
  494. }
  495. });
  496. }
  497.  
  498. function quickLookup(){
  499. gt_sl=getId('optSelLangFrom').value;
  500. gt_tl=getId('optSelLangTo').value;
  501. GT_tl=gt_tl;
  502. saveIt();
  503. gtRequest(txtSel,gt_sl,gt_tl);
  504. }
  505. function histLookup(e){
  506. try{
  507. var txt=e.target.textContent, ix=-1;
  508. //.innerHTML??
  509. e.preventDefault();
  510. var ha = getTag('a',getId('divHist'));
  511. // ix=ha.indexOf(e.target);
  512. /**/
  513. for(var i=0,l=ha.length; i<l; i++)
  514. if(e.target == ha[i]){ ix=i; break; }
  515. /**/
  516. if(ix<0) return;
  517. if(e.ctrlKey && e.altKey){ //remove itemm
  518. if(ix==0) return;
  519. killId('divHist');
  520. ht.splice(ix,1);
  521. GM_setValue('hist',JSON.stringify(ht));
  522. history();
  523. return;
  524. }
  525. var lang = ht[ix][2].match(/([a-zA-Z-]+)\|([a-zA-Z-]+)/);
  526. gt_sl=lang[1]; gt_tl=lang[2];
  527. txtSel = txt;
  528. getId('divResult').innerHTML = 'Loading...'
  529. gtRequest(txtSel,gt_sl,gt_tl);
  530. } catch(e){console.log('broken history\n'+e)}
  531. }
  532.  
  533. function fastSwap(){
  534. if(gt_sl != 'auto'){
  535. var t= gt_sl; gt_sl=gt_tl; gt_tl=t;
  536. gtRequest(txtSel,gt_sl,gt_tl);
  537. }
  538. }
  539.  
  540. function badResponce(html,e){
  541. var dr=getId('divResult')
  542. dr.innerHTML = '';
  543. var br=addEl(dr,'a',{'class':'gootranslink'},null,'Bad Google response');
  544. br.href=URL.substr(0,100);
  545. var m=html.match(/\<title\>(.*?)\<\/title\>/);
  546. if(m && m[1])
  547. addEl(dr,'p',{},null,m[1]);
  548. //id="captcha"
  549. m=html.match(/(<img\s.*?\>)/);
  550. if(m && m[1])
  551. addEl(dr,'p',{},null,m[1]);
  552. //_log(html);
  553. return;
  554. }
  555. var ex_sl , ex_tl;
  556. function extractResult(html){
  557. if(html){
  558. var html2 = html.match(/\<body[^\>]*\>([\s\S]+)\<\/body\>/);//[1];//select body content
  559. if(!html2){ // too many lettters!!!11
  560. badResponce(html); return;
  561. }
  562. html2 = html2[1].replace(/\<script[^\<]+\<\/script\>/ig, '');//remove script tags...
  563. killId('divExtract');
  564. divExtract = (new DOMParser()).parseFromString(html2, "text/html");
  565.  
  566. ex_sl= gt_sl, ex_tl=gt_tl;
  567. }
  568. try{ //gather info
  569. // 2013-10-20
  570. var _sl = detectedLang(gt_sl);
  571. var _tl = detectedLang(gt_tl);
  572. /* ?!11 150415 */ _log('**',_sl+'>'+_tl)
  573. if( 1 || ex_sl !== gt_sl )
  574. gt_sl_gms = _sl, gt_tl_gms =_tl;
  575. else
  576. gt_sl_gms = _tl, gt_tl_gms = _sl; /* ?!11 */
  577. getId('divBottom').removeChild(getId('optionsLink'));
  578. var oL= buildEl('div', {id:'optionsLink', title: 'Settings', 'class':''},
  579. null, null);
  580. addEl(oL,'a',{id:'optionsFrom','class':'gootransbutt gootranslink'},
  581. ['click', options], gt_sl_gms +' ');
  582. addEl(oL,'a',{id:'optionsFast','class':'gootransbutt gootranslink',
  583. title: 'swap languages'}, ['click', fastSwap], imgSwap);
  584. addEl(oL,'a',{id:'optionsTo','class':'gootransbutt gootranslink ' + (getId('divOpt') ? 'gtlActive':'gtlPassive')},
  585. ['click', options], gt_tl_gms );
  586. getId('divBottom').appendChild(oL);
  587. }catch(e){ console.log('gather\n'+e); }
  588. // var translation = getXId("result_box").textContent;
  589. // first run: resolve tl = auto
  590. if(GT_tl == 'auto')try{
  591. GT_tl=getXId("gt-tl").value;
  592. if(GT_tl) GM_setValue('to', GT_tl);
  593. else GT_tl='en';
  594. gt_tl=GT_tl;
  595. }catch(e){console.log('auto?\n'+e)}
  596.  
  597. //parse info
  598. var dR=getId('divResult');
  599. dR.innerHTML = '<a class="gootranslink gootransgoo" href="#'+
  600. '" target="_blank">' + 'translating..' /*translation*/ + '</a>'; // +'<br>&nbsp;';
  601. dR.childNodes[0].setAttribute('href',currentURL); //<a href
  602. dR.childNodes[0].setAttribute('title',deURI(currentURL,"&text="));
  603. dR.style.textAlign = rtl_langs.indexOf(GT_tl) < 0? 'left':'right';
  604. dR.style.direction = rtl_langs.indexOf(GT_tl) < 0? 'ltr' : 'rtl';
  605. dR.lang=GT_tl;
  606. dict();
  607.  
  608. }
  609.  
  610. function getSelection(t){
  611. var txt = '';
  612. //get selected text
  613. if (window.getSelection){
  614. txt = window.getSelection();
  615. }
  616. else if (document.getSelection) {
  617. txt = document.getSelection();
  618. }
  619. else if (document.selection) {
  620. txt = document.selection.createRange().text;
  621. }
  622. inTextArea= t.type=='textarea' ? t : null
  623. if(inTextArea){
  624. txt=t.value.substr(t.selectionStart,t.selectionEnd-t.selectionStart);
  625. }
  626. return trim(txt);
  627. }
  628. function swapLang(){
  629. var to=getId('optSelLangTo').value,from=getId('optSelLangFrom').value;
  630. // if(from!='auto'){
  631. getId('optSelLangTo').value = from;
  632. getId('optSelLangFrom').value = to;
  633. quickLookup();
  634. // }
  635. }
  636.  
  637. function saveIt(){
  638. var bs =getId('gtp-save');
  639. bs && (bs.className ='gootranslink goounsaved');
  640. }
  641.  
  642. function options(evt){
  643. var dO = getId('divOpt');
  644. if(!dO){//show options
  645. dO = buildEl('div', {id:'divOpt' });
  646. var oL=getId('optionsLink');
  647. oL.title='Hide settings';
  648. //
  649. var dA=getId('divHist');
  650. if(dA){
  651. insAfter(dO,dA);
  652. }else if(( dA=getId('divSourceshow')) ){
  653. insAfter(dO,dA);
  654. }else{
  655. insAfter(dO,getId('divResult'));
  656. }
  657. //from
  658. addEl(dO,'a',{'class':'gootransbutt gootranslink',
  659. target:'_blank', href:senoj, title: 'choose country flag icon'},
  660. ['click',function(e){
  661. e.preventDefault(); GM_openInTab(senoj); cleanUp(); return false;}],
  662. imgH+imgFlags['AN']+imgT);
  663. addEl(dO,'span', null, null,' From: ');
  664. var gt_slist = getXId("gt-sl");
  665. gt_slist= gt_slist ? gt_slist.innerHTML+'' : languagesGoogle;
  666. /* console.log(gt_slist) /* !!! */
  667.  
  668. var oF =dO.appendChild(buildEl('select', {id:'optSelLangFrom'}, null, gt_slist));
  669. oF.value = GM_getValue('from', "auto");
  670. oF.addEventListener('change', quickLookup, false);
  671. // swap
  672. addEl(dO,'span', null, null,'&nbsp');
  673. addEl(dO,'a', {id:'opSelectLangSwap',href:HREF_NO, 'class':"gootranslink",
  674. title:'Swap languages',}, ['click', swapLang], imgSwap);
  675. //to
  676. addEl(dO,'span', null, null,' To:');
  677. var gt_tlist = getXId("gt-tl");
  678. gt_tlist= gt_tlist ? gt_tlist.innerHTML+'' : languagesGoogle;
  679. var oT =dO.appendChild(buildEl('select', {id:'optSelLangTo'}, null, gt_tlist));
  680. oT.value = GM_getValue('to', "auto");
  681. oT.addEventListener('change', quickLookup, false);
  682. //use ctrl
  683. addEl(dO,'br');
  684. addEl(dO,'span', null, null,'Use with: ');
  685. var d=addEl(dO,'input', {id:'checkCtrl', type:'checkbox'},
  686. ['change', saveIt], null );
  687. addEl(dO,'span', null, null,' Ctrl &nbsp;&nbsp; ');
  688. d.checked = GM_getValue('ctrl',false);
  689. // use alt
  690. d=addEl(dO,'input', {id:'checkAlt', type:'checkbox',
  691. title:'using Alt is highly recommended'},
  692. ['change', saveIt], null);
  693. addEl(dO,'span', null,
  694. null,' Alt &nbsp;&nbsp;&nbsp; History:&nbsp;');
  695. d.checked = GM_getValue('alt',true);
  696. // history depth
  697. d=addEl(dO,'input', {id:'histSize', type:'textbox', maxlength: 2,
  698. style: "width:2em; ", title: "set to 0 to clear history"});
  699. addEl(dO,'span', null, null,' items &nbsp; of ');
  700. d.value = maxHT;
  701. // max # words in phrase
  702. d=addEl(dO,'input', {id:'histWc', type:'textbox', maxlength: 1,
  703. style: "width:1em; ", title: "max # of words in phrase"});
  704. addEl(dO,'span', null, null,' words');
  705. d.value = maxWC;
  706. // source box params
  707. addEl(dO,'br');
  708. addEl(dO,'span', null, null,'Source box: &nbsp;');
  709. d=addEl(dO,'input', {id:'sourceBH', type:'textbox', maxlength: 1,
  710. style: "width:1em; ", title: "box height"});
  711. d.value = sourceBH;
  712.  
  713. addEl(dO,'span', null, null,' height &nbsp; ');
  714.  
  715. d=addEl(dO,'input', {id:'sourceDP', type:'textbox', maxlength: 2,
  716. style: "width:2em; ", title: "# of lines to keep"});
  717. addEl(dO,'span', null, null,' depth &nbsp; ');
  718. d.value = sourceDP;
  719.  
  720. addEl(dO,'span', null, null," No flags:");
  721. d=addEl(dO,'input', {id:'checkNoflags', type:'checkbox',
  722. title: "don't show country flag icons"});
  723. d.checked = GM_getValue('noFlags');
  724. // colours
  725. for(var b,li=BG.C.length,ii=0;ii<li;ii++){
  726. b=addEl(dO,'span',{'class':'gtBGColor', title:BG.t[ii],
  727. style: 'background:'+ BG.C[ii]+'!important;' +
  728. (ii==0?'margin-left:6px' :'')
  729. }, null,'&nbsp;');
  730. b.addEventListener('click',BG.f[ii]);
  731. }
  732. //save
  733. var oS=
  734. addEl(dO,'div', {id:'gtp-save', 'class':'gootranslink gootransbutt',
  735. title: "save changes"},
  736. ['click', saveOptions], 'save');
  737. if(!GM_getValue('from'))
  738. saveIt();
  739. getId('optionsTo').className='gootransbutt gootranslink gtlActive';
  740. //cancel
  741. }
  742. else{//hide options
  743. killId(dO);
  744. getId('optionsLink').title = 'Settings';
  745. var oTL = getId('optionsTo');
  746. oTL.className='gootransbutt gootranslink gtlPassive';
  747. }
  748. }
  749. function showTrans(){
  750. try{
  751. var hOs = GM_getValue('showTrans',false) !== true;
  752. var shhi = hOs?'gtp-trans gtp-block':'gtp-trans gtp-hide';
  753. var tds= document.getElementsByClassName('gtp-trans');
  754. for(var i=0, il=tds.length; i<il; i++){
  755. tds[i].className=shhi;
  756. }
  757. getId('gtp_transOnOff').innerHTML = hOs?"&laquo;&laquo;":"&raquo;&raquo;";
  758. GM_setValue('showTrans',hOs)
  759. } catch(e){console.log('showTrans\n'+e)}
  760. }
  761.  
  762. function detectedLang(da){
  763. if(!da) return '';
  764. var gt_slist = getXId("gt-sl");
  765. //console.log(gt_slist.innerHTML)
  766. gt_slist= gt_slist ? gt_slist.innerHTML+'' : languagesGoogle;
  767. var re= new RegExp('ion value="'+da+'">(.*?)<\/opt');
  768. var ma= gt_slist.match(re);
  769. if(ma && ma[1]) return ma[1]; return da;
  770. }
  771.  
  772. function extractDict(txt){
  773. _log('!dict')
  774.  
  775. try{
  776. if(!txt) return;
  777. if(txt.substr(0,1) !== '[')
  778. throw 'Bad Google responce';
  779. txt=txt.replace(/,(?=,)/g,',""');
  780. txt=txt.replace(/\[(?=,)/g,'["asshole"');
  781. var dA=JSON.parse(txt);
  782. var dL='';
  783. //translation
  784. if( dA && dA[0] && dA[0][0] ){
  785. var dR=getId('divResult');
  786. var tr = dA[0][0][0];
  787. dR.childNodes[0].textContent=tr;
  788. dR.style.textAlign = rtl_langs.indexOf(GT_tl) < 0? 'left':'right';
  789. dR.style.direction = rtl_langs.indexOf(GT_tl) < 0? 'ltr' : 'rtl';
  790. dR.lang=GT_tl;
  791. addHistory(txtSel,tr);
  792. } else {
  793. getId('divResult').innerHTML='Google returns nothing!'; return;
  794. }
  795. // detected lang
  796. if(gt_sl=='auto' && dA[2]){
  797. var oF = getId("optionsFrom");
  798. oF.textContent= oF.textContent+' - '+detectedLang(dA[2]) +' ';
  799. }
  800. if(dA && dA[1] && dA[1][0] ){
  801. var da=dA[1];
  802. dL=buildEl('div',{id: 'gtp_dict'});
  803. var dT=addEl(dL,'table');
  804. var dB=addEl(dT,'tbody');
  805. var showT = 'gtp-trans gtp-hide',showI = "&raquo;&raquo;"
  806. if(GM_getValue('showTrans',false) === true)
  807. showT = 'gtp-trans gtp-block', showI = "&laquo;&laquo;"
  808. for(var i=0,il=da.length; i<il; i++){
  809. var tr=addEl(dB,'tr');
  810. addEl(tr,'td',{'class': 'gtp-pos'}, null, da[i][0]);
  811. for(var j=0,jl=da[i][2].length; j<jl; j++){
  812. tr=addEl(dB,'tr');
  813. addEl(tr,'td',{'class': 'gtp-word'}, null, da[i][2][j][0]);
  814. // console.log(JSON.stringify(da[i][2][j]))
  815. da[i][2][j][1]&&
  816. addEl(tr,'td',{'class': showT}, null, da[i][2][j][1].join(', '));
  817. }
  818. }
  819. var gtdir = (getId('divResult').style.direction=='rtl') ? 'left' : 'right';
  820. addEl(dL,'a',{'class': 'gootransbutt gootranslink', id: 'gtp_transOnOff',
  821. style: 'position: absolute; top: -.5em; '+gtdir+': 1px;'
  822. },['click', showTrans],showI);
  823. killId('gtp_dict');
  824. if(dL) getId('divResult').appendChild(dL);
  825. }
  826. killId('divSourceshow');
  827. killId('divHist');
  828.  
  829. if(GM_getValue('sourceShow',true))
  830. source();
  831. if(GM_getValue('histShow',false))
  832. history();
  833. if(!GM_getValue('histWc') && !getId('divOpt')) // no settings?
  834. options(); // show options
  835. } catch(e){
  836. _log('errexDict: '+e+'\n'+txt.substr(0,100));
  837. badResponce(txt,e);
  838. }
  839. }
  840.  
  841. function onTimerDict(){
  842. var q = dictURL +
  843. "&hl="+ GM_getValue('to','auto') +
  844. "&sl=" + gt_sl + "&tl=" + gt_tl + //+'&multires=1&ssel=0&tsel=0&sc=1';
  845. "&text="+ escAp(txtSel);
  846. //console.log('dict:'+ dictURL);
  847. _log('?dict')
  848. Request(q, extractDict);
  849. }
  850.  
  851. function dict(){
  852. var dR=getId('divResult');
  853. killId('gtp_dict');
  854. // var dD=buildEl('div',{id:"gtp_dict"},null,dict)
  855. // dR.appendChild(dD);
  856. window.setZeroTimeout(onTimerDict);
  857. }
  858.  
  859. function saveSource(){
  860. try{
  861. sT = getId('divSourcetext').value;
  862. GM_setValue('sourceText',JSON.stringify(sT));
  863. }catch(e){console.log('saveSource\n'+e)}
  864. }
  865.  
  866. function source(){
  867. var divSource = getId('divSourceshow');
  868. try{
  869. if(divSource){
  870. killId(divSource);
  871. var sL = getId('sourceLink');
  872. sL.innerHTML = 'Source';
  873. sL.className = 'gootransbutt gootranslink gtlPassive'
  874. sL.title = 'Show source';
  875. killId('imgSourcesave');
  876. GM_setValue('sourceShow',false);
  877. return;
  878. }
  879. GM_setValue('sourceShow',true);
  880. divSource= buildEl('form', {id:'divSourceshow'}, null, null);
  881.  
  882. if(sT){
  883. var sTa= sT.split('\n');
  884. var tS= txtSel + ' \u2192 ' + trim(getId('divResult').childNodes[0].textContent);
  885. if(tS != sTa[0]){
  886. while(sTa.length >= sourceDP) sTa.pop();
  887. sT= tS + '\n' + sTa.join('\n');
  888. }
  889. }else sT=txtSel;
  890. if(!getId('imgSourcesave'))
  891. insAfter(
  892. buildEl('img',{id: 'imgSourcesave', title: 'save source', src: imgSave,
  893. style: 'margin-bottom: -3px;'},
  894. ['click', saveSource], null)
  895. ,getId('sourceLink'));
  896. addEl(divSource,'textarea',
  897. { id:'divSourcetext', rows: sourceDP,
  898. style: "font-family: Tahoma,sans-serif !important; height:"
  899. +(sourceBH+1)+"em;"
  900. }, null, sT),
  901. getId('divBottom');
  902. var sL=getId('sourceLink');
  903. sL.innerHTML = 'Source';
  904. sL.className= 'gootransbutt gootranslink gtlActive';
  905. sL.title = 'Hide source';
  906. }catch(e){console.log('Sourceshow\n'+e)};
  907. insAfter(divSource,getId('divResult'));
  908. }
  909. // ht: [from, to, langpair, hitCount]
  910.  
  911. function history(){
  912. var divHist = getId('divHist');
  913. try{
  914. if(divHist){
  915. killId(divHist);
  916. var hL = getId('historyLink');
  917. hL.innerHTML = 'History'; hL.className= 'gootransbutt gootranslink gtlPassive';
  918. hL.title = 'Translation history';
  919. GM_setValue('histShow',false);
  920. return;
  921. }
  922. if(!maxHT) return;
  923. GM_setValue('histShow',true);
  924. divHist = buildEl('div', {id:'divHist'},['click', histLookup], null );
  925. //
  926. for(var i=0, l=ht.length; i<l; i++){
  927. var bkg = ht[i][0].indexOf(' ')>0 ? ' goohistlink' : '';
  928. addEl(divHist,'a', {href:HREF_NO, 'class': 'gootranslink'+bkg, 'titel': ht[i][1]+
  929. ((ht[i][3]>1) ? '\u00A0'+ '['+ht[i][3]+']' : '')},
  930. null, ht[i][0]);
  931. if(i < l-1)
  932. divHist.appendChild(document.createTextNode(' '));
  933. }
  934. //addEl(divHist,'span',null,null,'<br>&nbsp;');
  935. if(getId('divSourceshow'))
  936. insAfter(divHist,getId('divSourceshow'));
  937. else
  938. insAfter(divHist,getId('divResult'));
  939. var hL=getId('historyLink')
  940. // hl.textContent = 'X';
  941. hL.title= 'Hide history';
  942. hL.innerHTML = 'History'; hL.className = 'gootransbutt gootranslink gtlActive';
  943. }catch(e){console.log('hist problem\n'+e)}
  944. }
  945.  
  946. function saveOptions(evt){
  947. try{
  948. var from = getId('optSelLangFrom').value;
  949. var to = getId('optSelLangTo').value;
  950. var ctrl = getId('checkCtrl').checked;
  951. var alt = getId('checkAlt').checked;
  952. var mh = parseInt(getId('histSize').value);
  953. var wc = parseInt(getId('histWc').value);
  954. var nf = getId('checkNoflags').checked;
  955. if(0<= mh && mh <=99 && mh<maxHT){
  956. while(ht && ht.length>mh) ht.pop();
  957. GM_setValue('hist',ht? JSON.stringify(ht):'');
  958. if(getId('divHist')){
  959. killId('divHist');
  960. history();
  961. }
  962. }
  963. maxHT=mh;
  964. var bh = parseInt(getId('sourceBH').value);
  965. if( 0< bh && bh <10) sourceBH = bh;
  966. var dp = parseInt(getId('sourceDP').value);
  967. if( 0< dp && dp <100) sourceDP = dp;
  968. if(1<= wc && wc <=9) maxWC=wc;
  969. GM_setValue('histSize',maxHT)
  970. GM_setValue('histWc',maxWC);
  971. GM_setValue('from', from);
  972. GM_setValue('to', to);
  973. GM_setValue('ctrl', ctrl);
  974. GM_setValue('alt', alt);
  975. GM_setValue('sourceBH', sourceBH);
  976. GM_setValue('sourceDP', sourceDP);
  977. GM_setValue('noFlags',nf)
  978. getId('divDic').removeChild(getId('divOpt'));
  979. getId('optionsLink').title='Settings';
  980. return;
  981. }catch(e){console.log('saveOpnions\n'+e);}
  982. }
  983.  
  984. function addHistory(src,trt){
  985. if (!maxHT) return;
  986. try{
  987. var hts=GM_getValue("hist");
  988. if( !hts ){
  989. ht=[["google translator","Der Ubersetzer","en|de",0]];
  990. hts=JSON.stringify(ht);
  991. }
  992. ht=JSON.parse(hts);
  993. var st=trim(src+''); var tt = trim(trt+'');
  994. var wc = (st.split(' ')).length;
  995. if(wc>maxWC) return;
  996. var lang=currentURL.match(/langpair=([a-zA-Z-\|]+)/)[1];
  997. var ix=-1; // find word in hist
  998. for(var i=0, l=ht.length; i<l; i++)
  999. if(st==ht[i][0]){ ix=i; break; }
  1000. //if(ix==0) return; // nothing to do
  1001. var hits=0;
  1002. if(ix>=0){
  1003. hits=ht[ix][3];
  1004. if( (gt_sl+'|'+gt_tl) == ht[ix][2] && tt != ht[ix][1]) {
  1005. tt = ht[ix][1]; // don't touch my translasion
  1006. }
  1007. ht.splice(ix,1);
  1008. }
  1009. // if(hits<4)
  1010. hits++; // delete it by your own hands
  1011. if (ht.length>maxHT){
  1012. var minHit=99999; // which item shoud i remove?
  1013. for(i=ht.length-1;i>0;i--)
  1014. if(minHit>ht[i][3]) minHit=ht[i][3]
  1015. ix=ht.length-1;
  1016. for(var i=ix; i>0; i--)
  1017. if(minHit==ht[i][3]){ ix=i; break; }
  1018. ht.splice(ix,1);
  1019. }
  1020. ht.unshift([st,tt,lang,hits]);
  1021. GM_setValue('hist',JSON.stringify(ht));
  1022. } catch(e){console.log('addHist\n'+e);}
  1023. }
  1024. var senFlag = '';
  1025. function selFlag(e){
  1026. if(!isInited) {css(-1); isInited=true; }
  1027. killId('divSelflag');
  1028. setEscHnd();
  1029. var p = belowCursor(e,10,10);
  1030. var dsf = buildEl('div',{id:'divSelflag', style:
  1031. ';top:'+p.t+';left:'+p.l+';right:'+p.r +';bottom: auto'});
  1032. var sel=addEl(dsf,'select',{id: 'optSelFlag'},
  1033. null,languagesGoogle);
  1034. sel.value = GM_getValue('to',' en');
  1035. addEl(dsf,'span',null,null,'<br><br>');
  1036. addEl(dsf,'a', {href:HREF_NO, style:'padding: 3px 12px; margin-right: 2em;',
  1037. 'class':'gootransbutt gootranslink', title: "use icon"},
  1038. ['click', function(){saveFlag(true)}], '<b>OK</b>');
  1039. addEl(dsf,'a', {href:HREF_NO, style:'padding: 3px 4px;',
  1040. 'class':'gootransbutt gootranslink'},
  1041. ['click', function(){saveFlag(false)}],
  1042. '<b>Cancel</b>');
  1043. //
  1044. senFlag = e.target.src+'';
  1045. var sm = senFlag.match(/.+\/(.+)\.png/);
  1046. if(sm && sm[1]) senFlag= sm[1];
  1047. _log(senFlag);
  1048. if(senFlag) body.appendChild(dsf);
  1049. }
  1050. function saveFlag(tf){
  1051. if(tf && senFlag){
  1052. var s= 'l-'+getId('optSelFlag').value;
  1053. GM_setValue(s,senFlag);
  1054. _log(s+': '+senFlag );
  1055. }else _log('cant save flags' );
  1056. killId('divSelflag'); return;
  1057. }
  1058. var fCSS;
  1059. function flagClick(e){
  1060. e.preventDefault();
  1061. if(e.target.nodeName == 'IMG'){
  1062. _log('hit on: ' + e.target.alt);
  1063. selFlag(e);
  1064. }else (killId('divSelflag'));
  1065. }
  1066. function belowCursor(evt,ho,vo,lr){
  1067. var p={t:'', l:'auto', r:'auto'};
  1068. p.t=(evt.clientY+window.pageYOffset+vo)+'px';
  1069. var l=(evt.clientX+window.pageXOffset+ho)+'px';
  1070. if(lr && lr=='r')
  1071. { p.l=l; return p; }
  1072. var w = window.innerWidth;
  1073. var r=(w-(evt.clientX+window.pageXOffset)+ho);
  1074. // Q: How to detect visibility & thickess of vertical scrollbar?
  1075. r-=8; //
  1076. if(r<0) r= w/4;
  1077. r+='px';
  1078. if(lr && lr=='l')
  1079. { p.r=r; return p; }
  1080. if(evt.clientX < w/2)
  1081. p.l=l;
  1082. else
  1083. p.r=r;
  1084. return p;
  1085. }
  1086. var flagLang;
  1087. function getFlagSrc(lng, where){
  1088. if(!where) where = 'to';
  1089. if(GM_getValue('noFlags')) return imgFlags[where];
  1090. var fl='l-'+lng;
  1091. var flag = GM_getValue(fl,'');
  1092. if(!flag){
  1093. flag=imgFlags[lng];
  1094. if(!flag) return flag = imgFlags[where];
  1095. }
  1096. if(flag.indexOf('http')==0
  1097. //flag.indexOf('file') ==0
  1098. ||flag.indexOf('data:') ==0)
  1099. return flag;
  1100. flagLang=fl;
  1101. // flag= 'http .. /Flags/Panama.png;'
  1102. flag= senop+seno+flag+senoext;
  1103. flagRequest(flag);
  1104. return flag;
  1105. }
  1106. function flagRequest(f){
  1107. _log('load '+f);
  1108. GM_xmlhttpRequest({
  1109. method: 'GET',
  1110. url: f,
  1111. binary: true,
  1112. overrideMimeType: "text/plain; charset=x-user-defined",
  1113. headers: {
  1114. "User-Agent": UA
  1115. ,"Accept": "image/png,image/*;q=0.8,*/*;q=0.5"
  1116. ,"Accept-Encoding": "gzip, deflate"
  1117. },
  1118. onload: function(resp) {
  1119. try{
  1120. flagStore(resp.responseText,f);
  1121. }catch(e){console.log('FlagRqst\n'+e);}
  1122. }
  1123. });
  1124. }
  1125. function flagStore(r,url){
  1126. if(r.indexOf("<head")>=0)
  1127. {
  1128. console.log("Banned!\n"+url);
  1129. GM_setValue(flagLang,url);
  1130. }
  1131. else
  1132. GM_setValue(flagLang,"data:image/png;base64," + b2b64(r));
  1133. }
  1134.  
  1135. function trim(s){
  1136. return (s+'').replace(/\s+/g,' ').replace(/^\s/,'').replace(/\s$/,'');
  1137. }
  1138.  
  1139. function killId(nod){
  1140. if(!nod) return;
  1141. var n = nod;
  1142. if(typeof n == 'string'){
  1143. n= getId(nod);
  1144. }
  1145. if(!n) return;
  1146. if(n.parentNode) n.parentNode.removeChild(n);
  1147. else _log('cant kill: '+nod)
  1148. }
  1149.  
  1150. function addEl(to,type, attrArray, eL, html){
  1151. return to.appendChild(buildEl(type, attrArray, eL, html));
  1152. }
  1153.  
  1154. function buildEl(type, attrArray, eL, html)
  1155. {
  1156. var node = document.createElement(type);
  1157. for (var attr in attrArray) if (attrArray.hasOwnProperty(attr)){
  1158. node.setAttribute(attr, attrArray[attr]);
  1159. }
  1160. if(eL){
  1161. //console.log('buildEl\n'+type+'\n'+JSON.stringify(attrArray)+'\n'+eL[0])
  1162. node.addEventListener(eL[0], eL[1], eL[2]?true:false);
  1163. }
  1164. if(html)
  1165. node.innerHTML = html;
  1166. return node;
  1167. }
  1168.  
  1169. function getId(id, parent){
  1170. if(!parent)
  1171. return document.getElementById(id);
  1172. return parent.getElementById(id);
  1173. }
  1174.  
  1175. /* */
  1176. function getXId(id){
  1177. var r=divExtract.getElementById(id);
  1178. if(r) return r;
  1179. throw "Xel bug " + id;
  1180. }
  1181.  
  1182. function getTag(name, parent){
  1183. if(!parent)
  1184. return window.document.getElementsByTagName(name);
  1185. return parent.getElementsByTagName(name);
  1186. }
  1187. /*
  1188. * Drag and drop support adapted fom http://www.hunlock.com/blogs/Javascript_Drag_and_Drop
  1189. */
  1190. var savedTarget=null; // The target layer (effectively vidPane)
  1191. var orgCursor=null; // The original mouse style so we can restore it
  1192. var dragOK=false; // True if we're allowed to move the element under mouse
  1193. var dragXoffset=0; // How much we've moved the element on the horozontal
  1194. var dragYoffset=0; // How much we've moved the element on the verticle
  1195. var didDrag=false; //set to true when we do a drag
  1196. function moveHandler(e){
  1197. if (e == null) return;// { e = window.event }
  1198. if ( e.button<=1 && dragOK ){
  1199. savedTarget.style.left = e.clientX - dragXoffset + 'px';
  1200. savedTarget.style.top = e.clientY - dragYoffset + 'px';
  1201. return false;
  1202. }
  1203. }
  1204. function dragCleanup(e) {
  1205. document.removeEventListener('mousemove',moveHandler,false);
  1206. document.removeEventListener('mouseup',dragCleanup,false);
  1207. savedTarget.style.cursor=orgCursor;
  1208. dragOK=false; //its been dragged now
  1209. didDrag=true;
  1210. }
  1211. function dragHandler(e){
  1212. var htype='-moz-grabbing';
  1213. if (e == null) return;//
  1214. var target = e.target;// != null ? e.target : e.srcElement;
  1215. orgCursor=target.style.cursor;
  1216. if(target.nodeName!='DIV' )
  1217. return;
  1218. if( e.ctrlKey || e.altKey || e.shiftKey)
  1219. return; // enable selection inside
  1220. else if(clickedInsideID(target, res_dict))
  1221. return;
  1222. if (target = clickedInsideID(target, 'divDic')) {
  1223. savedTarget=target;
  1224. target.style.cursor=htype;
  1225. dragOK=true;
  1226. dragXoffset = e.clientX-target.offsetLeft;
  1227. dragYoffset = e.clientY-target.offsetTop;
  1228. //set the left before removing the right
  1229. target.style.left = e.clientX - dragXoffset + 'px';
  1230. target.style.right = null;
  1231. document.addEventListener('mousemove',moveHandler,false);
  1232. document.addEventListener('mouseup',dragCleanup,false);
  1233. return false;
  1234. }
  1235. }
  1236. function clickedInsideID(target, id) {
  1237. if (target.getAttribute('id')==id)
  1238. return target;
  1239. if (target.parentNode) {
  1240. while (target = target.parentNode) {
  1241. try{
  1242. if (target.getAttribute('id')==id)
  1243. return target;
  1244. }catch(e){}
  1245. }
  1246. }
  1247. return null;
  1248. }
  1249. //end drag code
  1250. function b2b64(inp) { // binary data --> base64
  1251. var output = [], c1, c2, c3, e1, e2, e3, e4, i = 0;
  1252. var k="ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/=";
  1253. var len = inp.length;
  1254. while( i < len ){
  1255. c1 = inp.charCodeAt(i++); c2 = inp.charCodeAt(i++); c3 = inp.charCodeAt(i++);
  1256. e1 = (c1&255) >> 2;
  1257. e2 = ((c1 & 3) << 4) | ((c2&255) >> 4);
  1258. e3 = ((c2 & 15) << 2) | ((c3&255) >> 6);
  1259. e4 = c3 & 63;
  1260. if( isNaN(c3)) e4 = 64;
  1261. if( isNaN(c2)) e3 = 64;
  1262. output.push( k.charAt(e1) + k.charAt(e2) + k.charAt(e3) + k.charAt(e4));
  1263. } return output.join("");
  1264. }
  1265.  
  1266.  
  1267. function ttrans( s, ttab ){
  1268. for( var c, cc, t='', il=s.length, i=0; i<il; i++ )
  1269. t += (c=ttab[cc=s.charAt(i)]) ? c : cc;
  1270. return t;
  1271. }
  1272. var tabUrlEsc = {
  1273. '#':'%23', '%':'%25', '&':'%26', '.':'%2e', '/':'%2f', '?':'%3f'
  1274. };
  1275. function escAp(s){
  1276. return ttrans( s, tabUrlEsc );
  1277. }
  1278.  
  1279. function stickStyle(css){
  1280. var s=document.createElement("style"); s.type="text/css";
  1281. s.appendChild(document.createTextNode(css));
  1282. return (document.head||document.documentElement).appendChild(s);
  1283. }
  1284.  
  1285. function css(n){
  1286. var k,i=0;
  1287. _log('cssN:',n);
  1288. if(-1 === n){ try{
  1289. k= +(GM_getValue('backG',0));
  1290. if(0<=k && k< BG.C.length) i=k;
  1291. } catch(e){};}
  1292. else GM_setValue('backG', +(i=+n) );
  1293. try{
  1294. if(window.gttpCSS) window.gttpCSS.parentNode.removeChild(window.gttpCSS);
  1295. }catch(e){console.log('cssX:\n'+e)};
  1296. _log('cssS:'+i,BG.C[i]);
  1297. window.gttpCSS=
  1298. stickStyle((
  1299. '#divResult {overflow: auto !important; padding:3px !important; margin: 0 0 3px 0 !important; max-height: 480px !important;}'+
  1300. '#divResult table *{ line-height: 0.9 !important}'+
  1301. '#divDic, #divDic div,#divLookup, #divUse {padding: 0; margin:0; \
  1302. width: auto;height: auto; border: none; border-radius: 0; background: none;}'+
  1303. '#divDic, #divDic *, #divSelflag, #divSelflag *{\
  1304. font-family: Tahoma, sans-serif!important;\
  1305. font-size: small!important;\
  1306. font-style: normal!important;\
  1307. font-weight: normal!important;\
  1308. font-stretch: normal!important;\
  1309. letter-spacing: normal!important;\
  1310. line-height: 1.1;\
  1311. }'+
  1312. '#divDic,#divSelflag {position: absolute; background:'+BG.C[i]+'!important; color:'+FG.t[i]+
  1313. '!important; opacity: 1'+
  1314. ';padding:5px !important; margin:0; z-index:10000; border-radius:5px; border: thin solid gray'+
  1315. ';text-align: left !important;}'+
  1316. '#divDic{ max-width:50%; padding: 3px; margin: 0;}'+
  1317. '#divSelflag{ max-width: 180px; }'+
  1318. '.gootranslink, #divDic .gootranslink ,#divSelflag .gootranslink\
  1319. {color:'+FG.l[i]+'!important; text-decoration: none !important;\
  1320. font: small normal Tahoma,sans-serif !important;'+
  1321. 'cursor:pointer !important; }' +
  1322. '#divDic a.gootranslink:visited,\
  1323. #divDic a.gootranslink:hover,\
  1324. #divDic a.gootranslink:active\
  1325. {color:'+FG.l[i]+'!important; text-decoration: none !important;}' +
  1326. '#gtp_dict table{background:'+BG.C[i]+'!important;}'+
  1327. '#gtp_dict tr {background:'+BG.H[i]+'!important;line-height:1;}'+
  1328. '#gtp_dict tr>td{font-size:1em !important; line-height:1!important;\
  1329. background:transparent!important;'+
  1330. '}'+
  1331. 'a.goohistlink {background:'+BG.F[i] +';}'+
  1332. '#gtp_dict {margin: 0; position: relative;}'+
  1333. '#gtp_dict ol {padding: 0 .5em 0 0; margin-left: 0.2em;}'+
  1334. '#gtp_dict li {list-style: square inside; display: list-item;}'+
  1335. 'div#gtp_dict tr>td {padding-left: .25em; vertical-align:top; border: none; color:'+FG.t[i]+'; }'+
  1336. '#optSelLangFrom,#optSelLangTo {max-width: 150px; text-align: left !important; height: 1.5em;\
  1337. }'+
  1338. '#divDic input{vertical-align: baseline !important;}'+
  1339. '#divDic input[type="checkbox"]{vertical-align: text-bottom !important;}'+
  1340. '#divResult a.gootranslink.gootransgoo{font-size: 1em !important; line-height: 1;}'+
  1341. '#divOpt span {color:'+FG.t[i]+'!important;}'+
  1342. '#optSelLangFrom,#optSelLangTo,#divDic input[type="textbox"]{background:'+BG.E[i]+'!important;\
  1343. color:'+FG.t[i]+'!important;\
  1344. padding-bottom: 3px !important; margin-bottom: 4px!important;}'+
  1345. '#divExtract{word-spacing: normal !important;}'+
  1346. '#divBottom {position: relative; width: 100%; font-size: smaller; text-decoration:none; }'+
  1347. '#historyLink {display: inline; position: relative; font-size:smaller; text-decoration:none;}'+
  1348. '#sourceLink {display: inline; position: relative; margin-left: 1em; font-size:smaller; text-decoration:none;}'+
  1349. '#imgSourcesave {display: inline; position: relative; margin-left:2px;\
  1350. cursor:pointer;}'+
  1351. 'div#optionsLink {display: inline; position: relative; margin-left: 1.5em; font-size:smaller !important; text-decoration:none !important;}'+
  1352. '#divDic #optionsLink [id^="options"] {margin-right: 2px; padding-left: 2px;}'+
  1353. '#divOpt {position: relative; padding: 5px;'+
  1354. 'border-top: thin solid grey!important;}'+
  1355. '#divLookup, #divOpt, #divBottom,#divSourcetext,#divHist,#divuse {direction: ltr !important;}'+
  1356. 'div#divDic #divHist {background:'+BG.C[i]+'; position:relative; padding:5px; text-align:left !important;'+
  1357. 'border-top: thin solid grey!important; color:'+FG.t[i]+';}'+
  1358. 'div#divResult #gtp_dict {background:'+BG.C[i]+'; color:'+FG.t[i]+';\
  1359. padding:3px!important; border-radius:3px;'+
  1360. 'margin-bottom: .1em!important; overflow-y:auto !important; overflow-x:hidden; font-size:small;}'+
  1361. '#divOpt {background:'+BG.C[i]+'; position:relative; padding:5px; text-align:left !important;}'+
  1362. '#divLookup, #divUse {background-color:transparent !important; position:absolute;\
  1363. padding: 3px; margin: 0;}'+
  1364. 'div#divDic>#divSourceshow {\
  1365. border: none; padding: 0 0 4px 0; margin: 0;}'+
  1366. '#divSourceshow>#divSourcetext{ width:97%; height: 3em; line-height: 1.2; overflow: auto !important;\
  1367. padding: 0 0 0 4px; margin: 0; border: none; border-top: 1px solid #AAA}' +
  1368. '.gtlPassive:before{ content:"\u2193";}'+
  1369. '.gtlActive:before{ content:"\u2191" !important;}'+
  1370. '#imgUse, #divGetback, #divGetforw {margin-left: 5px !important; cursor: pointer;}'+
  1371. '#divSourcetext {background:'+BG.E[i]+'!important; color:'+FG.t[i]+'!important;}'+
  1372. '#divSelflag .gootransbutt, #divDic .gootransbutt {background:'+BG.T[i]+'!important;'+
  1373. 'border-radius: 3px; margin-top: 5px; }'+
  1374. '#divDic .goounsaved {background-color: #EF9024!important;'+
  1375. 'border-radius: 3px; margin-top: 5px; }'+
  1376. 'td.gtp-pos { color:'+FG.t[i]+'!important; font-weight: bold !important; text-align: left; }'+
  1377. 'td.gtp-pos:before{ content:"\u2666 "; color:'+FG.t[i]+'!important;}'+
  1378. 'td.gtp-word {color:'+FG.t[i]+'!important; padding-left: 5px; padding-right: 10px;'+
  1379. 'vertical-align: top; white-space: normal;}'+
  1380. 'td.gtp-trans {/*overflow-x: hidden;*/ vertical-align: top; white-space: normal;'+
  1381. ' width: 100%; color:'+FG.g[i]+'!important}'+
  1382. 'td.gtp-pos, td.gtp-word, td.gtp-trans {padding-bottom: 0px !important; padding-bottom: 1px !important;}'+
  1383. '.gtp-hide {display: none}'+
  1384. '.gtp-block {display: block}'+
  1385. '#divTtsIfr{position: relative;padding: 0!important;margin:3px 0 0 0!important;\
  1386. background:'+ BG.C[i] +'!important;}'+
  1387. '')
  1388. );
  1389.  
  1390. if(-1 !== n) return;
  1391. stickStyle(
  1392. '#divDic, #divDic textarea, #divDic iframe {resize: both !important; }'+
  1393. '#divDic *::-moz-selection {background: #047 !important; color: #FC8 !important; }'+
  1394. '#divUse img, #divDic img, #divLookup img {width: auto; height: auto; }'+ // rt.com :/
  1395. '#divTtsLnk:after{ content:url('+imgPlay+') }'+
  1396. '#divTtsLnk {padding: 0 2px; margin: 0 3px 0 5px;}'+
  1397. '#divTtsIfh {width: 100%;overflow-x:hidden;\
  1398. background-color: rgba(127,127,127,.25); padding: 3px 0;\
  1399. }'+
  1400. '#divResult, #divResult div, #divResult table, #divResult tr, #divResult tr td,\
  1401. #divResult a, #divBottom, \
  1402. #divOpt select, #divOpt input, .gootranslink\
  1403. { padding:0 0 0 0; margin: 0 0 0 0; background: none repeat scroll 0 0 transparent;\
  1404. border: none; line-height: 0.95; float: none}'+
  1405. '#divOpt {line-height: 2.3 !important;}\
  1406. div#divBottom{padding-top: 3px;}\
  1407. .gootransbutt#optionsLink{margin-top:0; padding-top: 3px; padding-bottom: 1px;}\
  1408. #divOpt .gtBGColor{ border:thin solid blue !important; cursor: pointer;\
  1409. padding-right:6px; margin-right: 2px;}\
  1410. .gootranslink[titel]{position: relative;}\
  1411. .gootranslink[titel]:after {\
  1412. color: #050;\
  1413. content: attr(titel);\
  1414. position: absolute;\
  1415. visibility: hidden;\
  1416. opacity: 0.25;\
  1417. left: 16px;\
  1418. top: 32px;\
  1419. min-width: 0px;\
  1420. width: auto;\
  1421. white-space: normal !important;\
  1422. -moz-hyphens: none !important;\
  1423. text-decoration: none !important;\
  1424. border: 1px #aaa solid;\
  1425. border-radius: 6px;\
  1426. background-color: #dfd;\
  1427. padding: 1px 4px;\
  1428. font-size: small; line-height:0.9;\
  1429. -webkit-transition: all .2s linear .2s;\
  1430. transition: all .2s linear .2s;\
  1431. z-index: 2147483647;\
  1432. }\
  1433. .gootranslink[titel]:hover:after {\
  1434. visibility: visible;\
  1435. opacity: 1;\
  1436. -webkit-transition: all .5s linear .7s;\
  1437. transition: all .5s linear .7s;\
  1438. }\
  1439. #divOpt #gtp-save{\
  1440. position: absolute; right: 2px;; bottom: -1.3em;\
  1441. font-weight: bold; cursor: pointer; z-index: 999999;\
  1442. }\
  1443. ');
  1444. }
  1445. function insAfter(n,e){
  1446. if(e.nextElementSibling){
  1447. e.parentNode.insertBefore(n,e.nextElementSibling);
  1448. }else{
  1449. e.parentNode.appendChild(n);
  1450. }
  1451. }
  1452. function insBefore(n,e){
  1453. e.parentNode.insertBefore(n,e);
  1454. }
  1455. var imgH='<img border=0 src="' , imgD='data:image/png;base64,',imgT='">';
  1456. var imgPlay=imgD+
  1457. 'iVBORw0KGgoAAAANSUhEUgAAAAwAAAAMCAYAAABWdVznAAAACXBIWXMAAC4jAAAuIwF4pT92AAAAGXRFWHRTb2Z0d2FyZQBBZG9iZSBJbWFnZVJlYWR5ccllPAAAAI1JREFUeNpj+H9h4v//D7YRh4FqGf6vsvuPF4PAnnQ4n+H/At3/WDEIgOjTvQg2EDP8ny7/HwWDAIy+sxnBvr0RzGb43y/0H45hAMR+fAQrm+F/B/t/DAASe3gQK5vhfwsDpgaQ2IP9WNkQDTBMsgaYJhgNcgqMfWUZDg3oGveUINh4NSBrnGdGggY0DAAAy70TuBaoTgAAAABJRU5ErkJggg=='
  1458. ;
  1459.  
  1460. //http://www.senojflags.com/images/national-flag-icons/Portugal-Flag.png
  1461. imgForwSrc= imgD+
  1462. 'R0lGODlhEAAQAIQfAAFYsQJkw15iWgRz3AeN0Xl8c2mBiQma5B6W1h+b3yqh4BKp+kWk0pudlDC2/0y79X+4z1fC/7q6r3DK/4rN7a3Kz8fKwc/Uyr3c5N7dzuTi1Ofl1enm2O3r3/b06////yH+EUNyZWF0ZWQgd2l0aCBHSU1QACH5BAEKAB8ALAAAAAAQABAAAAWf4OeNXmdyX6qKpNSc25p6TVEIgl1osljntxxPNrIILhrDZBl5OBaLj6mTa1CWk+YTOpXgsBEM5gFdmDLBiJrC2VCgB85lYJFcmg+MPAPnDP4aGg4ODxsVFBkPBwd/jRuDFBoQDxkVix+NHwBQSBmeFwoEHwEDAQEAAJQVDAwQFxCipqeoABkMiwcWFqKjpqgEBLjAwDIAHwoJyQkIzB8hADs=';
  1463. imgForw = imgH+imgForwSrc+imgT;
  1464.  
  1465. imgBackSrc= imgD+
  1466. 'iVBORw0KGgoAAAANSUhEUgAAABAAAAAQCAMAAAAoLQ9TAAAAAXNSR0IArs4c6QAAAGBQTFRFEgAAAVixAmTDXmJaBHPcB43ReXxzaYGJCZrkHpbWH5vfKqHgEqn6RaTSm52UMLb/TLv1f7jPV8L/urqvcMr/is3trcrPx8rBz9TKvdzk3t3O5OLU5+XV6ebY7evf9vTrdcYePwAAAAF0Uk5TAEDm2GYAAAABYktHRACIBR1IAAAACXBIWXMAAAsTAAALEwEAmpwYAAAAoklEQVQY003PWxKDIAwF0CCPYouCKIIB7f532UTttPfvnkkyE4Ar+3Ecb8pdoRGM859UY4yUxow/4CpPcM6HuCzLoxaZacV9IY1G0mWGsG0sUs4M1rnU9hRjpCt4AZZ9CyGWOauyg7UB09qC97UqpRhWDFNN3jfFAf0qiFiLcwK4g57KNAw0JUTXqY4gZ2vtgEKwMFBItL47QN/3T8oLxPnVB4gUDOnY6pKLAAAAAElFTkSuQmCC';
  1467. imgBack = imgH+imgBackSrc+imgT;
  1468.  
  1469. imgSwap = "<img border=0 style="+'"margin-bottom: -3px;"'+
  1470. "src=''>";
  1471.  
  1472. imgUse = imgD+ 'iVBORw0KGgoAAAANSUhEUgAAABAAAAAQCAYAAAAf8/9hAAACQklEQVR42mNkoBAwgoiwykntQCoeTW7hqva8SmINeGZv7yCJLHHw4IHnQAOkCBoQXjdnJpBOI9MHsxhDa2b9z4v2xir7/z8QgyCU/gdi/GMA00CKYc7qnQyMwdUz/icGuSFp+s/w+cs3hqPHzzK8fPUGzBcVE2EwMzVk4OTkhGgG4X//GTbtPsTAGFgx5X9iiCfYFpDE+/cfGTZv28Ogr6vJ8A7ItjQ3ZLhy/Q7DxUvXGVxcHRj4+PgY/kIN2LH3IAOjX8mE//Gh3gz/oU7bu+8wg4S4CIOBnjbDlFmLGRLiIxnYWJgYzl+4wvD0+WsGSxtLhr///kPVHmBg9Cns+R8d7At32rJlaxnCw/0Y/vxjYli2dBVDaEQI2DZmoA0r12xg8A30hxgAxIcPAg3wzOv4HxrgDRb4C/TGmtXrGbx9PBmYWFgZVq9YA4wnRoa/f/4xeAf4MOzYsp3B3dcH7oVTR4FecMtq+e/r7QExAIhPnzzFwMsnwKCkqsIgwM0OVgwKnzPnrjB8/PCeQdPAEO7aCyeOMDC6pDf8d3V1AQv8BcXN758Mu3YfZJBXUmQQl5YCGszA8OzJE4ZH9+4z2DnYMPxkZAGrBYXZtbPHGBgdk2v+2zo6AQX+g21jY2FmALqZ4cyZCwxv3rwFKmZgEBYWYjAw1GP4zcDE8O3nb4gBQPFb508wMNonVv7XNzYFGwBzGgsTEwMvFwfI+8DkAzQPGDgfvn1n+PHrLzhdwMDDG5cYGK0iC/8zUABAmcmAEgMA4i8z829X6pgAAAAASUVORK5CYII=';
  1473.  
  1474. imgSave= imgD+
  1475. 'iVBORw0KGgoAAAANSUhEUgAAABAAAAAQCAYAAAAf8/9hAAACgklEQVR42mNkAIL4tTeX/P36xnpBtCXDolN3GYS42Bn8dGUYGBkZGWBg/+0XDFOOPfzDxSOw8enZg5X729N/g8TBKuJWX/v++d0TjnVprgx2vZsY1MT4GebE2jMgg93XnzBMPnKXgYdPhOE/A8OiFRE68XADolZc+v7u+V2O7QUBDMbNyxg0JAQZlqZ6ohiw/fIDhr6DNxgEhSTB/P///9euiTFsARsQtuTs96ePb3PsLgpmOPfoJQM3GyuDgZwYigErHxQw3P/wnoGVhR2kmeH05aivt/bsFQEbELjg5PfHT+5y/Pj5i4GZiYkBG0iIqGZgYvwH5x89O4Xh1t6zCmADPGce+s7LycrBAPIdDmBuHAkM1L8IA870AQ04DzHAdere70CKgwEPcLFNAboAYcCR050Md/ZDDXCcuJMIA7IYGJmQvHCqieHugQsQA2z7tnzXEBfgEOXBbQa7VBSQRDLgdA0Dw4fDEAMsuzZ835nrxfH/zx+GP38QzvwP1DD3VjDDp7+PcZl7AWyAadua7weK/Tm+ff3G8Ov3HxQV7//eYVh8O5Xh3/+/KOL//jEzvHvBaw02wLBp+fcj5cEcL16+Y/j67QeGNZe/rmE4+XYxitj9Rw4Ml7Z9h3hBp27x95NV4RwvXmE34C/Q9lUPKhne/rkL5qsKGzIcOefLcGbXFogBmtXzvx9JUeHgYsWeiEDg1dcnDB3Xe4Bpn5GhUquUIXPzL4azB/YpMLb0T+E6+o1nryLrI2Nmhn8gEyBZEJkEhScwjX3geQMS+s//RfjfvV9Sz0x4/xszLlmyRAmUFIBYGogFgZgbmiZYgJgZFmZADMq+P4H4CxB/AOKnQHwQALc2/z+3odafAAAAAElFTkSuQmCC';
  1476.  
  1477. imgFlags= {
  1478. 'AN': imgD+
  1479. 'iVBORw0KGgoAAAANSUhEUgAAABgAAAAQCAYAAAAMJL+VAAAAAXNSR0IArs4c6QAAAAZiS0dEAP8A/wD/oL2nkwAAAAlwSFlzAAALEwAACxMBAJqcGAAABANJREFUOMutlEtsVHUUh797//c10955F2hKM7QUEUgNoRAqSUGJESO6ACSyIKgJwSXGGFJ2mhijG2JE3WgiiSaCAQlVeagIhAYID4GIgKU4lGdbhs5MZ+bemft0UXGBukF/y7M4X07OOR/8T9n84eebNm3ZuvPBuvJgYcvOg8r573a0zF62uqP/zM/JUdvX8T1f1fRqV+vkQu74gXsqoaUaRlSbt2zaxby1UJs2e929WR1tSzqSdH/15RMnThw7fL+ftP7tj41hueG5eDK12GzJLp3a0vxoImaKCyNlLuWrnBvKU8oXWZoKKVoOZ4suoSyBAIQGWpSIEaFnWoLH2zOkhs9d2Lh8aed9gBh7ZuOVk+acDeVU68JEzJwUNxQ5Kgc4rk/ZsqnYNWJBnZ+eb+bZVp3V0xvoTAvSIkTyXUZqPh4SrqSQ1AQzOmdNsq/8Ovr7wOXTALJ3sX8/jkXZsig7PlU3oFGRWTA5QlRT8YRKBQULQdEDTYKeKRF6uzJ8s7yZBSkBnsvtksXVgs1vt0qse++Dd2a2t6UARFs2Gx/p6FkVeh7pRp2kJpBkyJoqm+el6Zli0Ds3jl338fwAz/MJQvDCEMsL6Z6kkxtzGbICbARJTZBpyhiPTElmDu/t6xNJr3LN6V652XJdYoZOKqKS0AS2H7B9sMqqbBQ5CHA9nyCcAIRB8OcGwdRkNnQm2X/D5nrFQ1NVTEWma1H3vKuH9v0g8oWC0/TYoqfHjUSrUASZBp2ELohrMhU/ZElLFE1iAuD7BEGA7/oTABlCwPF9xj2J4yMOViBIGAIzFiXmFg0B0JROTy1ku54kDCYAhoqpyUQViU8vV2iOSnTEBL7nI4UBvj8xRQhIEmgyHLxR5XzJpx4CqkG6fu/uZ6+99LICYFZG9ymq9pZVsynaDhUnQs0LSRgyqajgzXPjtGs+cxs87oxZxHGYGQlpMWUCwJBlTt2pQKACUaxisbC994W1lmUNCIDh3MDtzOIVr1cCWddVhUxUJxZRiGkybgDjTsipuzbbLxU5ervKj9fKhI7DyvZGnCBkz1CFHTkL3TRpvXn6THHrq2uK+ZF+IPzrk7XBk31M71lbsuqUa3UsR8cxQkxFolETxKMGRmMES5ZBM9hVrPP93hKBW8coj461DJ057p7e/8X1awPfBlD5myoy1uiRm0JdW7FrlGp1qnWXWqBiGhpNTREKnkz5yvnB6i/HLgS3ruYoDef0ejnnDudy+bHSTQfKQPCvLiod2bXbnP/iJ2XbpWg5WEInlPAv9W070r/n692DJ48esF33hqmKetn1w4cyYnPvtrO8ezSc8f6h4vxVr2yNRiJz/kmID53ZK9a/MfWpNR9FFJGduPL/nj8ActPaP8mCkKMAAAAASUVORK5CYII='
  1480. ,'en': imgD+
  1481. 'iVBORw0KGgoAAAANSUhEUgAAABAAAAAQCAYAAAAf8/9hAAAACXBIWXMAAAsTAAALEwEAmpwYAAAAIGNIUk0AAHolAACAgwAA+f8AAIDpAAB1MAAA6mAAADqYAAAXb5JfxUYAAAJJSURBVHjapJNdSJNRGMd/Z3u3OTXXhx8xjRLDFWIk0jAk6WYYfaAZSUWGZkFdRNRFYAYFXhkI2gpKqISC1IsIUiKIchUEUQiuj4uiMT8m0wbNr+n2vu/p4l2K3Yn/m3Pg8Pye8/wf/kJKyWokACuQnjxXojgwowCOwPW2iS37y3k0bOfJwwHmYzFePK5n7Oxl0FRyH9zicGM3CZOFBhccypMox2tJdWZnK0Bay/AGmr+McmJvKUXbaul79hmTzYJ9x3bQdEw2C2V7XOzLUdmVZ2PWvZvm1lcAaQpg/vY9xL2NRdRMvqHUlYmzoRw9oREPjICqoic0qtIiFGVmECp20+kd4IPPD2AWQMFMcOSnxWZDavqyIcfOXARVJ7fLu9w4s4nEwgLpmzdtVQB+HW0kNcOBFo3Cv6VICWYTSEmw6hSYBOjGo7LOwdz0lHEHiMVimCciaNEpRLIWqSMUxWCpKgiBEAIJqI4M5tLtS4Cs++3kZGUlK5cUcFciEwkKBl//t3xBeHIS3KWYAEo87biruxiKWkh568N/447RWZdI3fAlMG8l5Z2PT00deOq6KfHcBjAA+YVO+p6eo8zXy/P+r9R9tCcbCQQCgIMHOuh3llFx2kNn/jjFhWuXAN6rFay/66XtZZjG9zrh0KjxVasCNgsAs9N/OHnkJq0/HLjOH+NadmjRg7jW00tnWOPKQGRxzEH/EOM7XUhNI+of4nckCEDThRaCl2qor66E/p64ABxALrBmhVmYBsYEYE4GybxCgAbExWrj/HcACIPUyGtYcDcAAAAASUVORK5CYII='
  1482. ,'fr': imgD+
  1483. 'iVBORw0KGgoAAAANSUhEUgAAABAAAAAQCAYAAAAf8/9hAAAACXBIWXMAAAsTAAALEwEAmpwYAAAAIGNIUk0AAHolAACAgwAA+f8AAIDpAAB1MAAA6mAAADqYAAAXb5JfxUYAAAFESURBVHjapJM7TsNAEIa/9RqjKECEkGgoKOgoaHKCdByBc3APDsIRaHID01ASRTxEgxI5zsvetb1DkcSJCUaKMtJqZzSrb2f/nVEiwj6mgAA4Wu67mAWmPtDqdB6+f2e73ftK3L+92yJcPT2e+0CzKBzt9uW/1zVurpc1KxBh/vwC0PQBbW3BeJwyHE5rAebto/T9s1MkywC0D2CMJYpmRNG8FuBG49IvPE1u0wUMIEkMg8GMOJ5T9yn5KC59AWxiNgEJvV5GHCf1kvffS1+3Tkh9bw1I0wSRBs65ehU3cpJlpJnbBBi0PsS5+qaSFUBAihyTWwC8hYgG50ApVa6tjlvlPAVOSIxdV2CtJQiEIPDrnxAcVKoxdi2iFSmYTD4r58MwrMSvcfSntgpoARfA8Y6zMAG+FKCXg6R3BBSAVfuO888AocKXohfLXWQAAAAASUVORK5CYII='
  1484. ,'ru': imgD+
  1485. 'iVBORw0KGgoAAAANSUhEUgAAABAAAAAQCAYAAAAf8/9hAAAACXBIWXMAAAsTAAALEwEAmpwYAAAAIGNIUk0AAHolAACAgwAA+f8AAIDpAAB1MAAA6mAAADqYAAAXb5JfxUYAAADkSURBVHjapJM5TgNBEEVfzeZlQIOE5ARxClLOxC04CzFX4ArERAhZsoTEyDDurqa7SGxsIYJZKqnov9p+iZkxJQSogLN9HhIKfBZAY2abUdVFVhlQT5igzoB8AiDPAGKMg5UHTQFwc/fI6rKh/VLsz4b/u9FFXbF5b4+AnXe8rIW2C72qN8uSwtwJoHMwm5NSP0+EbyP4E8DD0z3XZUnqdr0A2XLBawjcHgBOPZYXINILYMlw6o8deFVsvoCq7AewhFf9BWg047n9GOMDFaABroDzgeIt8CZ7J1YjHBkBlanv/DMAwHdYum9dlZQAAAAASUVORK5CYII='
  1486. ,'zh-CN': imgD+
  1487. 'iVBORw0KGgoAAAANSUhEUgAAABAAAAAQCAYAAAAf8/9hAAAACXBIWXMAAAsTAAALEwEAmpwYAAAAIGNIUk0AAHolAACAgwAA+f8AAIDpAAB1MAAA6mAAADqYAAAXb5JfxUYAAAFbSURBVHjapJO/SgNBEIe/vduLCSaKFhb6DCnFzs4XsMgjWlj4AD6BFhYKFqKgTSzSaBLN7f+xuJM7RSHRHywLy8w3Oz9mlIjwHymgA/TrexU54E0Dm+Pj0QSACGRADt39gLnQTZkftHd2uqOBdYmR3uGQYjfhnzPSXJHeYTCK2Lu8AquGJYC9vgFY10AupUdvTekdTYgnGvdYVQ5P0nTa+kW+vYV4D5BrgGAt9nZOmCzwDxlxkqG60D0ImCsN/msbKs+J1gKgAXww+Kcp5nKBKgQ0rA09i/Oi8iR880HAGdMATFnSuX8kzWZ8vvqHJvi7idnGBibPGkBZlvSzDJFURfj6FC1ISyl4SpcagDUWVRRISi2ngPTL8PiArUysW3AW0RqUWmqCRATjWiZaa6HbQxXFcjOYUpVTA1wU4eb15S+r4BSwCewBgxWT58BY1XZ16nsVRcCp/67zxwDGd5ld8bkQAQAAAABJRU5ErkJggg=='
  1488. ,'ar': imgD+'iVBORw0KGgoAAAANSUhEUgAAABAAAAAQCAYAAAAf8/9hAAAACXBIWXMAAAsTAAALEwEAmpwYAAAAIGNIUk0AAHolAACAgwAA+f8AAIDpAAB1MAAA6mAAADqYAAAXb5JfxUYAAAIvSURBVHjapJPNSxRxHMY/M/Ob3bXddV3DBTUh8yJdCiHwIIV0DXqB7nWLDv0NBt06Rt3q0E0PGaFBmGSGUCZKQR6ENV8wddfVfZ35zW/mN9NByd4u4nP5Hr7wgYfneYwoijiODCAGpA7uUaSAugAyN0aHCn9//TAg0BoAYZrYlv0P4cX1oZwAkjrUnMv1ABBFEX6oaU+1cibbARGsVbZZrxWxTRPDMAH4UsgDJAVg+WFATTmUZBWtNVd7L3Ktux9ZaxBvShDriTO2/InhxSlMA042NeOHAYAlADzfpyIbFBplLp/u42bnBRzHYWVjHa01bdlWrnT0sV4t8mppBmFYeL4PgAkgPUnJq9LQHpdyvbizn9koFgl8n3gsxlZ5D39+gcFsDwrNjqwiPXkIcKVHvrxJwa3SIgOcmRkCIUil0+yUSkTxOLsfpmltKHaVQ778A1d6vwMcwlBTk3W2kzaJ5VWyc/Ns1euYmWbK468xZ+cpZBJU3CphqHGls58QgJQSO2njKo+RlY8M3L3Dzq3bdHSdoimdRn9bpO3xEx5tLrDn1MidaMGX8hCgpIeIkqTsBCMLb+gebOfe2Cjm+ASh72M+uM9T1ng28ZKUnSCKQtSBhf0UlCJBREzY2Jbg4fvnTHadpX/gPKZpMLc0zOzqV2xLYBkmIRGeUr8AKtIh5fzmHy2b/v6Oyam3YIDAIi5s5H/qbAAZoBNIH3ELNWDDAKyDIVlHBGhAGced888BANVaBfgg0AbGAAAAAElFTkSuQmCC'
  1489. ,"af":"Namibia" //
  1490. ,"sq":"Albania"
  1491. //,"ar":"United-Arab-Emirates"
  1492. ,"hy":"Armenia"
  1493. ,"az":"Azerbaijan"
  1494. ,"eu":"Spain"
  1495. ,"be":"Belarus"
  1496. ,"bn":"Bangladesh"
  1497. ,"bs":"Bosnian"
  1498. ,"bg":"Bulgaria"
  1499. ,"ca":"Spain"
  1500. ,"ceb":"Philippines"
  1501. ,"ny":"Mozambique"
  1502. /* ,"zh-CN":"Chinese"*/
  1503. ,"hr":"Croatia"
  1504. ,"cs":"Czech-Republic"
  1505. ,"da":"Denmark"
  1506. ,"nl":"Netherlands"
  1507. /*, "en":"English"*/
  1508. ,"eo":"United-Nations"
  1509. ,"et":"Estonia"
  1510. ,"tl":"Philippines"
  1511. ,"fi":"Finland"
  1512. /* ,"fr":"France"*/
  1513. ,"gl":"Ukraine"
  1514. ,"ka":"Georgia"
  1515. ,"de":"Germany"
  1516. ,"el":"Greece"
  1517. ,"gu":"India"
  1518. ,"ht":"Haiti"
  1519. ,"ha":"Nigeria"
  1520. ,"iw":"Israel"
  1521. ,"hi":"India"
  1522. ,"hmn":"Laos"
  1523. ,"hu":"Hungary"
  1524. ,"is":"Iceland"
  1525. ,"ig":"Nigeria"
  1526. ,"id":"Indonezia"
  1527. ,"ga":"Ireland"
  1528. ,"it":"Italy"
  1529. ,"ja":"Japan"
  1530. ,"jw":"Japan"
  1531. ,"kn":"India"
  1532. ,"kk":"Kazakhstan"
  1533. ,"km":"Cambodia"
  1534. ,"ko":"North-Korea"
  1535. ,"lo":"Laos"
  1536. ,"la":"Vatican-City"
  1537. ,"lv":"Latvia"
  1538. ,"lt":"Lithuania"
  1539. ,"mk":"Macedonia"
  1540. ,"mg":"Madagascar"
  1541. ,"ms":"Malaysia"
  1542. ,"ml":"India"
  1543. ,"mt":"Malta"
  1544. ,"mi":"New-Zealand"
  1545. ,"mr":"India"
  1546. ,"mn":"Mongolia"
  1547. ,"my":"Burma" //absent
  1548. ,"ne":"Nepal"
  1549. ,"no":"Norway"
  1550. ,"fa":"Iran"
  1551. ,"pl":"Poland"
  1552. ,"pt":"Brazil"
  1553. ,"pa":"Pakistan"
  1554. ,"ro":"Romania"
  1555. /*, "ru":"Russia"*/
  1556. ,"sr":"Serbia"
  1557. ,"st":"Lesotho"
  1558. ,"si":"Sri-Lanka"
  1559. ,"sk":"Slovakia"
  1560. ,"sl":"Slovenia"
  1561. ,"so":"Somalia"
  1562. ,"es":"Spain"
  1563. ,"su":"Sudan"
  1564. ,"sw":"Mozambique"
  1565. ,"sv":"Sweden"
  1566. ,"tg":"Tajikistan"
  1567. ,"ta":"India"
  1568. ,"te":"India"
  1569. ,"th":"Thailand"
  1570. ,"tr":"Turkey"
  1571. ,"uk":"Ukraine"
  1572. ,"ur":"Pakistan"
  1573. ,"uz":"Uzbekistan"
  1574. ,"vi":"Viet-Nam"
  1575. ,"cy":"Wales"
  1576. ,"yi":"Israel"
  1577. ,"yo":"Nigeria"
  1578. ,"zu":"South-Africa"
  1579. };
  1580. imgFlags['zh-TW'] = imgFlags['zh-CN'];
  1581. imgFlags['to'] = imgForwSrc; imgFlags['from'] = imgBackSrc;
  1582.  
  1583. languagesGoogle = '<option value="auto">Detect language</option>\
  1584. </option><option value="af">Afrikaans</option><option value="sq">Albanian</option><option value="ar">Arabic</option><option value="hy">Armenian</option><option value="az">Azerbaijani</option><option value="eu">Basque</option><option value="be">Belarusian</option><option value="bn">Bengali</option><option value="bs">Bosnian</option><option value="bg">Bulgarian</option><option value="ca">Catalan</option><option value="ceb">Cebuano</option><option value="ny">Chichewa</option><option value="zh-CN">Chinese</option><option value="hr">Croatian</option><option value="cs">Czech</option><option value="da">Danish</option><option value="nl">Dutch</option><option value="en">English</option><option value="eo">Esperanto</option><option value="et">Estonian</option><option value="tl">Filipino</option><option value="fi">Finnish</option><option value="fr">French</option><option value="gl">Galician</option><option value="ka">Georgian</option><option value="de">German</option><option value="el">Greek</option><option value="gu">Gujarati</option><option value="ht">Haitian Creole</option><option value="ha">Hausa</option><option value="iw">Hebrew</option><option value="hi">Hindi</option><option value="hmn">Hmong</option><option value="hu">Hungarian</option><option value="is">Icelandic</option><option value="ig">Igbo</option><option value="id">Indonesian</option><option value="ga">Irish</option><option value="it">Italian</option><option value="ja">Japanese</option><option value="jw">Javanese</option><option value="kn">Kannada</option><option value="kk">Kazakh</option><option value="km">Khmer</option><option value="ko">Korean</option><option value="lo">Lao</option><option value="la">Latin</option><option value="lv">Latvian</option><option value="lt">Lithuanian</option><option value="mk">Macedonian</option><option value="mg">Malagasy</option><option value="ms">Malay</option><option value="ml">Malayalam</option><option value="mt">Maltese</option><option value="mi">Maori</option><option value="mr">Marathi</option><option value="mn">Mongolian</option><option value="my">Myanmar (Burmese)</option><option value="ne">Nepali</option><option value="no">Norwegian</option><option value="fa">Persian</option><option value="pl">Polish</option><option value="pt">Portuguese</option><option value="pa">Punjabi</option><option value="ro">Romanian</option><option value="ru">Russian</option><option value="sr">Serbian</option><option value="st">Sesotho</option><option value="si">Sinhala</option><option value="sk">Slovak</option><option value="sl">Slovenian</option><option value="so">Somali</option><option value="es">Spanish</option><option value="su">Sundanese</option><option value="sw">Swahili</option><option value="sv">Swedish</option><option value="tg">Tajik</option><option value="ta">Tamil</option><option value="te">Telugu</option><option value="th">Thai</option><option value="tr">Turkish</option><option value="uk">Ukrainian</option><option value="ur">Urdu</option><option value="uz">Uzbek</option><option value="vi">Vietnamese</option><option value="cy">Welsh</option><option value="yi">Yiddish</option><option value="yo">Yoruba</option><option value="zu">Zulu</option>\
  1585. ';
  1586.  
  1587. /* */
  1588. try{
  1589. body=
  1590. window.document.body;
  1591.  
  1592. maxHT=GM_getValue('histSize');
  1593. if(!maxHT) maxHT=20;
  1594. maxWC=GM_getValue('histWc');
  1595. if(!maxWC) maxWC=3;
  1596. sourceBH = GM_getValue('sourceBH',3);
  1597. sourceDP = GM_getValue('sourceDP',10);
  1598. if(!sourceDP) sourceDP = 10;
  1599.  
  1600. var av = GM_getValue('version');
  1601. if( av!= version ){
  1602. GM_setValue('version',version);
  1603. }
  1604. sT=GM_getValue('sourceText');
  1605. if (sT){
  1606. try{
  1607. sT=JSON.parse(sT);
  1608. }catch(e){console.log('broken source\n'+e)} ;
  1609. } else sT='';
  1610.  
  1611. // gmail spoils my timeout -- workaround
  1612. // borrowed from dbaron.org/log/20100309-faster-timeouts :
  1613. // Only add setZeroTimeout to the window object, and hide everything
  1614. // else in a closure.
  1615. (function() {
  1616. var timeouts = [];
  1617. var messageName = "zero-timeout-130613";
  1618. // Like setTimeout, but only takes a function argument. There's
  1619. // no time argument (always zero) and no arguments (you have to
  1620. // use a closure).
  1621. function setZeroTimeout(fn) {
  1622. timeouts.push(fn);
  1623. window.postMessage(messageName, "*");
  1624. }
  1625. function handleMessage(event) {
  1626. if (event.source == window && event.data == messageName) {
  1627. event.stopPropagation();
  1628. if (timeouts.length > 0) {
  1629. var fn = timeouts.shift();
  1630. fn();
  1631. } } }
  1632. window.addEventListener("message", handleMessage, true);
  1633. // Add the one thing we want added to the window object.
  1634. window.setZeroTimeout = setZeroTimeout;
  1635. })();
  1636.  
  1637. document.addEventListener('mouseup', showLookupIcon, false);
  1638. document.addEventListener('mousedown', mousedownCleaning, false);
  1639. // http://www.senojflags.com/#flags16
  1640. if( location.href.indexOf(senojflags[0])>-1
  1641. ||location.href.indexOf(senojflags[1])>-1
  1642. ){try{
  1643. var f16=getId('flags16');
  1644. if(f16){
  1645. if(!fCSS) fCSS=
  1646. 'div#flags16 img {cursor: pointer !important}'+
  1647. 'div#flags48,div#flags32 {display:none; visibility: hidden}';
  1648. stickStyle(fCSS);
  1649. _log('inside\n' + location.href);
  1650. insBefore(buildEl('div',{style:'font: bold italic 100% sans-serif; color:red;',
  1651. align:'left'},null,'&nbsp;&nbsp;<u>Click on a country flag icon then choose the language</u>'),
  1652. f16);
  1653. f16.addEventListener('click',flagClick,false);
  1654. }
  1655. }catch(e){console.log('senojflags\n'+e)}
  1656. }
  1657. }catch(e){console.log('nobody\n'+e); }
  1658.  
  1659. function deURI(u,m){
  1660. if(!m) m= "&q=";
  1661. var x = u.indexOf(m);
  1662. if(x>=0) u=u.substr(x + m.length);
  1663. return decodeURIComponent(u).split(' ').slice(0,9).join(' ');
  1664. }
  1665. var uq=location.href.match(/^https:\/\/translate\.google\.[a-z]{2,3}\/translate_tts\?client\=t\&.+?(\&q\=.+)/);
  1666. if(uq && uq[1])
  1667. window.document.title=deURI(uq[1]);
  1668.  
  1669. }
  1670. main();
  1671. }
  1672. //
  1673.