translate.google tooltip

Translates selected text into a `tooltip' via Google translate

当前为 2015-04-28 提交的版本,查看 最新版本

  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.90
  14. //* This is a descendant of lazyttrick's http://userscripts.org/scripts/show/36898.
  15. // 3.7.90 2015-04-28 + dark colouring
  16. // 3.7.8.2 2015-04-26 + new country flags host; * fixes; gothic colouring
  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.  
  71. var GTurl= "https://translate.google"+GTsuffix+"/?";
  72. var dictURL= "https://translate.google"+GTsuffix+"/translate_a/t?client=t";
  73. var ttsURL= "https://translate.google.com/translate_tts?client=t";
  74. var version= 3800;
  75.  
  76. var HREF_NO = 'javascript:void(0)';
  77.  
  78. var llii=0, _log = function(){ /* * /
  79. for (var s=++llii +':', li=arguments.length, i = 0; i<li; i++)
  80. s+=' ' + arguments[i];
  81. console.log(s)
  82. /* */
  83. }
  84. _log("tgtt..");
  85. var URL='*'; var tURL;
  86. var GT_tl='auto';
  87. var body;
  88.  
  89. //{[ hacks
  90. var UA = navigator.userAgent;
  91. 0 && (UA="Mozilla/5.0 (Windows NT 5.1; rv:37.0) Gecko/20100101 Firefox/37.0");
  92. var ano="";
  93. 0 && (ano= "http://anonymouse.org/cgi-bin/anon-www.cgi/");
  94. //}]
  95. var senoj="https://cdn.rawgit.com/trespassersW/UserScripts/master/Flags/",
  96. //"http://www.senojflags.com/",
  97. senojflags = [senoj+"index.html?gtrantoltip#", "http://lh/Flags/"
  98. ],seno=senoj, senoimg=seno //+'images/national-flag-icons/'
  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' ],
  122. l: ['#047' ,'#047' ,'#047' ,'#047' ,'#7CF' ,'#047' ],
  123. g: ['#404040','#404040' ,'#404040','#404040','#ccb' ,'#404040']
  124. };
  125. var BG={
  126. t: ["yellow" ,"grey" ,"blue" ,"green" ,"dark" ,"striped"],
  127. C: ['#FFFFE1','#D1D1D1','#D3ECEC','#C4FFC4' ,'#333' ,
  128. _G+'(-45deg, #DDD, #AAA )'],
  129. T: [_G+"(to right,#FFFFE1,#DDDDAA)", _G+"(to right,#D1D1D1,#A0A097)",
  130. _G+"(to right,#D3ECED,#8CCCCE)", _G+"(to right,#888C4FFC4,#6BEF69)",
  131. _G+"(to right,#777,#373737)", _G+"(to right,#CCC, #888)"],
  132. //'#F9E78F','#C1C1B7','#BCD1D1','#AAEEAA','#E8D0D0','rgba(0,0,0,.1)'
  133. H: [_T,_T,_T,_T,_T,_G+'(to bottom right,rgba(127,127,127,0),rgba(127,127,127,.15))'
  134. ],
  135. F: [G_,G_,G_,G_,W_,G_],
  136. E: ['#F4F4E8','#EEEEEE','#E8E8F4','#E8F4E8','#777' ,'#DDDDDD'],
  137. f: [function(){css(0)}, function(){css(1)}, function(){css(2)},
  138. function(){css(3)}, function(){css(4)}, function(){css(5)}]
  139. }
  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',style: 'position: relative;padding: 0 !important;margin:3px 0 0 0!important;'},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', {style: 'margin-left:.5em;' },[],deURI(url));
  430. // addEl(IFR, 'br');
  431. addEl(IFR, 'iframe',{
  432. width: "100%", height: "48", frameborder: "0",scrolling:"auto", marginheight:"0", marginwidth:"0",
  433. style:'padding-top:3px;overflow-x:hidden;',
  434. src: url
  435. },
  436. null,null);
  437.  
  438. insAfter(IFR,getId('divBottom'));
  439. /*
  440. */
  441. }
  442.  
  443. function ttsRequest(txt,t,e){
  444. var etxt = escAp(txt);
  445. etxt=ttsURL + "&tl=" + t + "&ie=utf-8&q=" + etxt.split(' ').slice(0,19).join(' ');
  446. _log('tts> '+etxt);
  447. if(e)
  448. GM_openInTab(etxt);
  449. else
  450. openInFrame(etxt);
  451. //GM_openInTab(etxt);
  452. // sorry, firefox' decodeAudioData() does NOT support mp3
  453. }
  454.  
  455. function gtRequest(txt,s,t){
  456. var etxt = escAp(txt);
  457. currentURL = GTurl + "langpair=" + s + "|" + t + "&text="+etxt ;
  458. etxt=GTurl + "langpair=" + s + "|" + t + "&text=" + etxt.split(' ').slice(0,9).join(' ');
  459. if( !((s==last_sl && t==last_tl) || (s==last_tl && t==last_sl)) || (divExtract=='')){
  460. var c=':';
  461. //_log(s+c+last_sl+ ' '+t+c+last_tl + ' '+ divExtract );
  462. divExtract = '';
  463. Request(etxt);
  464. }else{
  465. extractResult(null);
  466. }
  467. last_sl = s; last_tl = t;
  468. }
  469. function Request(url,cb){
  470. URL=url; _log('R: '+URL);
  471. var meth=cb? 'POST': 'GET';
  472. GM_xmlhttpRequest({
  473. method: meth,
  474. url: url,
  475. headers: {
  476. "User-Agent": UA
  477. ,"Accept": "text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8"
  478. ,"Accept-Encoding": "gzip, deflate"
  479. //,"Host": "www.google.com"
  480. },
  481. onload: function(resp) {
  482. try{
  483. if(cb)
  484. cb(resp.responseText)
  485. else
  486. extractResult(resp.responseText);
  487. }catch(e){
  488. if(getId('divResult'))
  489. getId('divResult').innerHTML =
  490. '<a id="gttpErrRef" href="#">error processing response text:</a><br>'+e;
  491. getId('gttpErrRef').href=URL.subsr(0,99);
  492. }
  493. }
  494. });
  495. }
  496.  
  497. function quickLookup(){
  498. gt_sl=getId('optSelLangFrom').value;
  499. gt_tl=getId('optSelLangTo').value;
  500. GT_tl=gt_tl;
  501. saveIt();
  502. gtRequest(txtSel,gt_sl,gt_tl);
  503. }
  504. function histLookup(e){
  505. try{
  506. var txt=e.target.textContent, ix=-1;
  507. //.innerHTML??
  508. e.preventDefault();
  509. var ha = getTag('a',getId('divHist'));
  510. // ix=ha.indexOf(e.target);
  511. /**/
  512. for(var i=0,l=ha.length; i<l; i++)
  513. if(e.target == ha[i]){ ix=i; break; }
  514. /**/
  515. if(ix<0) return;
  516. if(e.ctrlKey && e.altKey){ //remove itemm
  517. if(ix==0) return;
  518. killId('divHist');
  519. ht.splice(ix,1);
  520. GM_setValue('hist',JSON.stringify(ht));
  521. history();
  522. return;
  523. }
  524. var lang = ht[ix][2].match(/([a-zA-Z-]+)\|([a-zA-Z-]+)/);
  525. gt_sl=lang[1]; gt_tl=lang[2];
  526. txtSel = txt;
  527. getId('divResult').innerHTML = 'Loading...'
  528. gtRequest(txtSel,gt_sl,gt_tl);
  529. } catch(e){console.log('broken history\n'+e)}
  530. }
  531.  
  532. function fastSwap(){
  533. if(gt_sl != 'auto'){
  534. var t= gt_sl; gt_sl=gt_tl; gt_tl=t;
  535. gtRequest(txtSel,gt_sl,gt_tl);
  536. }
  537. }
  538.  
  539. function badResponce(html,e){
  540. var dr=getId('divResult')
  541. dr.innerHTML = '';
  542. var br=addEl(dr,'a',{'class':'gootranslink'},null,'Bad Google response');
  543. br.href=URL.substr(0,100);
  544. var m=html.match(/\<title\>(.*?)\<\/title\>/);
  545. if(m && m[1])
  546. addEl(dr,'p',{},null,m[1]);
  547. //id="captcha"
  548. m=html.match(/(<img\s.*?\>)/);
  549. if(m && m[1])
  550. addEl(dr,'p',{},null,m[1]);
  551. //_log(html);
  552. return;
  553. }
  554. var ex_sl , ex_tl;
  555. function extractResult(html){
  556. if(html){
  557. var html2 = html.match(/\<body[^\>]*\>([\s\S]+)\<\/body\>/);//[1];//select body content
  558. if(!html2){ // too many lettters!!!11
  559. badResponce(html); return;
  560. }
  561. html2 = html2[1].replace(/\<script[^\<]+\<\/script\>/ig, '');//remove script tags...
  562. //html2 = html.replace(/\<iframe[^\<]+\<\/iframe\>/ig, '');
  563. //cleanup
  564. killId('divExtract');
  565. // killId(res_dict);
  566. //append translated page as hidden frame
  567. var iframe = addEl(document.body,'iframe',
  568. {style:'visibility:hidden;'});
  569. var dX = addEl(iframe.contentWindow.document.body,'div',
  570. {id:'divExtract'}, null, html2);
  571. divExtract = document.importNode(dX, true);
  572. iframe.parentNode.removeChild(iframe);
  573. ex_sl= gt_sl, ex_tl=gt_tl;
  574. }
  575. try{ //gather info
  576. // 2013-10-20
  577. var _sl = detectedLang(gt_sl);
  578. var _tl = detectedLang(gt_tl);
  579. // var _sl = getXId("gt-sl-gms").textContent;
  580. // var _tl = getXId("gt-tl-gms").textContent;
  581. // _sl = _sl.replace(/^.*?\:\s*/,'');
  582. // _tl = _tl.replace(/^.*?\:\s*/,'');
  583. /* ?!11 150415 */ _log('**',_sl+'>'+_tl)
  584. if( 1 || ex_sl !== gt_sl )
  585. gt_sl_gms = _sl, gt_tl_gms =_tl;
  586. else
  587. gt_sl_gms = _tl, gt_tl_gms = _sl; /* ?!11 */
  588. getId('divBottom').removeChild(getId('optionsLink'));
  589. var oL= buildEl('div', {id:'optionsLink', title: 'Settings', 'class':'gootransbutt'},
  590. null, null);
  591. addEl(oL,'a',{id:'optionsFrom','class':'gootranslink'},
  592. ['click', options], gt_sl_gms +' ');
  593. addEl(oL,'a',{id:'optionsFast','class':'gootranslink',
  594. title: 'swap languages'}, ['click', fastSwap], imgSwap);
  595. addEl(oL,'a',{id:'optionsTo','class':'gootranslink ' + (getId('divOpt') ? 'gtlActive':'gtlPassive')},
  596. ['click', options], gt_tl_gms );
  597. getId('divBottom').appendChild(oL);
  598. }catch(e){ console.log('gather\n'+e); }
  599. // var translation = getXId("result_box").textContent;
  600. // first run: resolve tl = auto
  601. if(GT_tl == 'auto')try{
  602. GT_tl=getXId("gt-tl").value;
  603. if(GT_tl) GM_setValue('to', GT_tl);
  604. else GT_tl='en';
  605. gt_tl=GT_tl;
  606. }catch(e){console.log('auto?\n'+e)}
  607.  
  608.  
  609. //parse info
  610. var dR=getId('divResult');
  611. dR.innerHTML = '<a class="gootranslink" href="#'+
  612. '" target="_blank">' + 'translating..' /*translation*/ + '</a>'; // +'<br>&nbsp;';
  613. dR.childNodes[0].setAttribute('href',currentURL); //<a href
  614. dR.childNodes[0].setAttribute('title',deURI(currentURL,"&text="));
  615. dR.style.textAlign = rtl_langs.indexOf(GT_tl) < 0? 'left':'right';
  616. dR.style.direction = rtl_langs.indexOf(GT_tl) < 0? 'ltr' : 'rtl';
  617. dR.lang=GT_tl;
  618. dict();
  619.  
  620. }
  621.  
  622. function getSelection(t){
  623. var txt = '';
  624. //get selected text
  625. if (window.getSelection){
  626. txt = window.getSelection();
  627. }
  628. else if (document.getSelection) {
  629. txt = document.getSelection();
  630. }
  631. else if (document.selection) {
  632. txt = document.selection.createRange().text;
  633. }
  634. inTextArea= t.type=='textarea' ? t : null
  635. if(inTextArea){
  636. txt=t.value.substr(t.selectionStart,t.selectionEnd-t.selectionStart);
  637. }
  638. return trim(txt);
  639. }
  640. function swapLang(){
  641. var to=getId('optSelLangTo').value,from=getId('optSelLangFrom').value;
  642. // if(from!='auto'){
  643. getId('optSelLangTo').value = from;
  644. getId('optSelLangFrom').value = to;
  645. quickLookup();
  646. // }
  647. }
  648.  
  649. function saveIt(){
  650. var bs =getId('gtp-save');
  651. bs && (bs.className ='gootranslink goounsaved');
  652. }
  653.  
  654. function options(evt){
  655. var dO = getId('divOpt');
  656. if(!dO){//show options
  657. dO = buildEl('div', {id:'divOpt' });
  658. var oL=getId('optionsLink');
  659. oL.title='Hide settings';
  660. //
  661. var dA=getId('divHist');
  662. if(dA){
  663. insAfter(dO,dA);
  664. }else if(( dA=getId('divSourceshow')) ){
  665. insAfter(dO,dA);
  666. }else{
  667. insAfter(dO,getId('divResult'));
  668. }
  669. //from
  670. addEl(dO,'a',{'class':'gootransbutt gootranslink',
  671. target:'_blank', href:seno, title: 'choose country flag icon'},
  672. ['click',function(e){
  673. e.preventDefault(); GM_openInTab(senojflags[0]); cleanUp(); return false;}],
  674. imgH+imgFlags['AN']+imgT);
  675. addEl(dO,'span', null, null,' From: ');
  676. var gt_slist = getXId("gt-sl");
  677. gt_slist= gt_slist ? gt_slist.innerHTML+'' : languagesGoogle;
  678. /* console.log(gt_slist) /* !!! */
  679.  
  680. var oF =dO.appendChild(buildEl('select', {id:'optSelLangFrom'}, null, gt_slist));
  681. oF.value = GM_getValue('from', "auto");
  682. oF.addEventListener('change', quickLookup, false);
  683. // swap
  684. addEl(dO,'span', null, null,'&nbsp');
  685. addEl(dO,'a', {id:'opSelectLangSwap',href:HREF_NO, 'class':"gootranslink",
  686. title:'Swap languages',}, ['click', swapLang], imgSwap);
  687. //to
  688. addEl(dO,'span', null, null,' To:');
  689. var gt_tlist = getXId("gt-tl");
  690. gt_tlist= gt_tlist ? gt_tlist.innerHTML+'' : languagesGoogle;
  691. var oT =dO.appendChild(buildEl('select', {id:'optSelLangTo'}, null, gt_tlist));
  692. oT.value = GM_getValue('to', "auto");
  693. oT.addEventListener('change', quickLookup, false);
  694. //use ctrl
  695. addEl(dO,'br');
  696. addEl(dO,'span', null, null,'Use with: ');
  697. var d=addEl(dO,'input', {id:'checkCtrl', type:'checkbox'},
  698. ['change', saveIt], null );
  699. addEl(dO,'span', null, null,' Ctrl &nbsp;&nbsp; ');
  700. d.checked = GM_getValue('ctrl',false);
  701. // use alt
  702. d=addEl(dO,'input', {id:'checkAlt', type:'checkbox',
  703. title:'using Alt is highly recommended'},
  704. ['change', saveIt], null);
  705. addEl(dO,'span', null,
  706. null,' Alt &nbsp;&nbsp;&nbsp; History:&nbsp;');
  707. d.checked = GM_getValue('alt',true);
  708. // history depth
  709. d=addEl(dO,'input', {id:'histSize', type:'textbox', maxlength: 2,
  710. style: "width:2em; ", title: "set to 0 to clear history"});
  711. addEl(dO,'span', null, null,' items &nbsp; of ');
  712. d.value = maxHT;
  713. // max # words in phrase
  714. d=addEl(dO,'input', {id:'histWc', type:'textbox', maxlength: 1,
  715. style: "width:1em; ", title: "max # of words in phrase"});
  716. addEl(dO,'span', null, null,' words');
  717. d.value = maxWC;
  718. // source box params
  719. addEl(dO,'br');
  720. addEl(dO,'span', null, null,'Source box: &nbsp;');
  721. d=addEl(dO,'input', {id:'sourceBH', type:'textbox', maxlength: 1,
  722. style: "width:1em; ", title: "box height"});
  723. d.value = sourceBH;
  724.  
  725. addEl(dO,'span', null, null,' height &nbsp; ');
  726.  
  727. d=addEl(dO,'input', {id:'sourceDP', type:'textbox', maxlength: 2,
  728. style: "width:2em; ", title: "# of lines to keep"});
  729. addEl(dO,'span', null, null,' depth &nbsp; ');
  730. d.value = sourceDP;
  731.  
  732. addEl(dO,'span', null, null," No flags:");
  733. d=addEl(dO,'input', {id:'checkNoflags', type:'checkbox',
  734. title: "don't show country flag icons"});
  735. d.checked = GM_getValue('noFlags');
  736. // colours
  737. for(var b,li=BG.C.length,ii=0;ii<li;ii++){
  738. b=addEl(dO,'span',{'class':'gtBGColor', title:BG.t[ii],
  739. style: 'background:'+ BG.C[ii]+'!important;' +
  740. (ii==0?'margin-left:6px' :'')
  741. }, null,'&nbsp;');
  742. b.addEventListener('click',BG.f[ii]);
  743. }
  744. //save
  745. addEl(dO,'span',null,null,' &nbsp; ');
  746. var oS=
  747. addEl(dO,'a', {href:HREF_NO, id:'gtp-save', 'class':'gootranslink gootransbutt',
  748. title: "save changes"},
  749. ['click', saveOptions], '<b>save</b>');
  750. if(!GM_getValue('from'))
  751. saveIt();
  752. getId('optionsTo').className='gootransbutt gootranslink gtlActive';
  753. //cancel
  754. }
  755. else{//hide options
  756. killId(dO);
  757. getId('optionsLink').title = 'Settings';
  758. var oTL = getId('optionsTo');
  759. oTL.className='gootransbutt gootranslink gtlPassive';
  760. }
  761. }
  762. function showTrans(){
  763. try{
  764. var hOs = GM_getValue('showTrans',false) !== true;
  765. var shhi = hOs?'gtp-trans gtp-block':'gtp-trans gtp-hide';
  766. var tds= document.getElementsByClassName('gtp-trans');
  767. for(var i=0, il=tds.length; i<il; i++){
  768. tds[i].className=shhi;
  769. }
  770. getId('gtp_transOnOff').innerHTML = hOs?"&laquo;&laquo;":"&raquo;&raquo;";
  771. GM_setValue('showTrans',hOs)
  772. } catch(e){console.log('showTrans\n'+e)}
  773. }
  774.  
  775. function detectedLang(da){
  776. if(!da) return '';
  777. var gt_slist = getXId("gt-sl");
  778. //console.log(gt_slist.innerHTML)
  779. gt_slist= gt_slist ? gt_slist.innerHTML+'' : languagesGoogle;
  780. var re= new RegExp('ion value="'+da+'">(.*?)<\/opt');
  781. var ma= gt_slist.match(re);
  782. if(ma && ma[1]) return ma[1]; return da;
  783. }
  784.  
  785. function extractDict(txt){
  786. _log('!dict')
  787.  
  788. try{
  789. if(!txt) return;
  790. if(txt.substr(0,1) !== '[')
  791. throw 'Bad Google responce';
  792. txt=txt.replace(/,(?=,)/g,',""');
  793. txt=txt.replace(/\[(?=,)/g,'["asshole"');
  794. var dA=JSON.parse(txt);
  795. var dL='';
  796. //translation
  797. if( dA && dA[0] && dA[0][0] ){
  798. var dR=getId('divResult');
  799. var tr = dA[0][0][0];
  800. dR.childNodes[0].textContent=tr;
  801. dR.style.textAlign = rtl_langs.indexOf(GT_tl) < 0? 'left':'right';
  802. dR.style.direction = rtl_langs.indexOf(GT_tl) < 0? 'ltr' : 'rtl';
  803. dR.lang=GT_tl;
  804. addHistory(txtSel,tr);
  805. } else {
  806. getId('divResult').innerHTML='Google returns nothing!'; return;
  807. }
  808. // detected lang
  809. if(gt_sl=='auto' && dA[2]){
  810. var oF = getId("optionsFrom");
  811. oF.textContent= oF.textContent+' - '+detectedLang(dA[2]) +' ';
  812. }
  813. if(dA && dA[1] && dA[1][0] ){
  814. var da=dA[1];
  815. dL=buildEl('div',{id: 'gtp_dict'});
  816. var dT=addEl(dL,'table');
  817. var dB=addEl(dT,'tbody');
  818. var showT = 'gtp-trans gtp-hide',showI = "&raquo;&raquo;"
  819. if(GM_getValue('showTrans',false) === true)
  820. showT = 'gtp-trans gtp-block', showI = "&laquo;&laquo;"
  821. for(var i=0,il=da.length; i<il; i++){
  822. var tr=addEl(dB,'tr');
  823. addEl(tr,'td',{'class': 'gtp-pos'}, null, da[i][0]);
  824. for(var j=0,jl=da[i][2].length; j<jl; j++){
  825. tr=addEl(dB,'tr');
  826. addEl(tr,'td',{'class': 'gtp-word'}, null, da[i][2][j][0]);
  827. // console.log(JSON.stringify(da[i][2][j]))
  828. da[i][2][j][1]&&
  829. addEl(tr,'td',{'class': showT}, null, da[i][2][j][1].join(', '));
  830. }
  831. }
  832. var gtdir = (getId('divResult').style.direction=='rtl') ? 'left' : 'right';
  833. addEl(dL,'a',{'class': 'gootransbutt gootranslink', id: 'gtp_transOnOff',
  834. style: 'position: absolute; top: -.5em; '+gtdir+': 1px;'
  835. },['click', showTrans],showI);
  836. killId('gtp_dict');
  837. if(dL) getId('divResult').appendChild(dL);
  838. }
  839. killId('divSourceshow');
  840. killId('divHist');
  841.  
  842. if(GM_getValue('sourceShow',true))
  843. source();
  844. if(GM_getValue('histShow',false))
  845. history();
  846. if(!GM_getValue('histWc') && !getId('divOpt')) // no settings?
  847. options(); // show options
  848. } catch(e){
  849. _log('errexDict: '+e+'\n'+txt.substr(0,100));
  850. badResponce(txt,e);
  851. }
  852. }
  853.  
  854. function onTimerDict(){
  855. var q = dictURL +
  856. "&hl="+ GM_getValue('to','auto') +
  857. "&sl=" + gt_sl + "&tl=" + gt_tl + //+'&multires=1&ssel=0&tsel=0&sc=1';
  858. "&text="+ escAp(txtSel);
  859. //console.log('dict:'+ dictURL);
  860. _log('?dict')
  861. Request(q, extractDict);
  862. }
  863.  
  864. function dict(){
  865. var dR=getId('divResult');
  866. killId('gtp_dict');
  867. // var dD=buildEl('div',{id:"gtp_dict"},null,dict)
  868. // dR.appendChild(dD);
  869. window.setZeroTimeout(onTimerDict);
  870. }
  871.  
  872. function saveSource(){
  873. try{
  874. sT = getId('divSourcetext').value;
  875. GM_setValue('sourceText',JSON.stringify(sT));
  876. }catch(e){console.log('saveSource\n'+e)}
  877. }
  878.  
  879. function source(){
  880. var divSource = getId('divSourceshow');
  881. try{
  882. if(divSource){
  883. killId(divSource);
  884. var sL = getId('sourceLink');
  885. sL.innerHTML = 'Source';
  886. sL.className = 'gootransbutt gootranslink gtlPassive'
  887. sL.title = 'Show source';
  888. killId('imgSourcesave');
  889. GM_setValue('sourceShow',false);
  890. return;
  891. }
  892. GM_setValue('sourceShow',true);
  893. divSource= buildEl('form', {id:'divSourceshow'}, null, null);
  894.  
  895. if(sT){
  896. var sTa= sT.split('\n');
  897. var tS= txtSel + ' \u2192 ' + trim(getId('divResult').childNodes[0].textContent);
  898. if(tS != sTa[0]){
  899. while(sTa.length >= sourceDP) sTa.pop();
  900. sT= tS + '\n' + sTa.join('\n');
  901. }
  902. }else sT=txtSel;
  903. if(!getId('imgSourcesave'))
  904. insAfter(
  905. buildEl('img',{id: 'imgSourcesave', title: 'save source', src: imgSave,
  906. style: 'margin-bottom: -3px;'},
  907. ['click', saveSource], null)
  908. ,getId('sourceLink'));
  909. addEl(divSource,'textarea',
  910. { id:'divSourcetext', rows: sourceDP,
  911. style: "font-family: Tahoma,sans-serif !important; height:"
  912. +(sourceBH+1)+"em;"
  913. }, null, sT),
  914. getId('divBottom');
  915. var sL=getId('sourceLink');
  916. sL.innerHTML = 'Source';
  917. sL.className= 'gootransbutt gootranslink gtlActive';
  918. sL.title = 'Hide source';
  919. }catch(e){console.log('Sourceshow\n'+e)};
  920. insAfter(divSource,getId('divResult'));
  921. }
  922. // ht: [from, to, langpair, hitCount]
  923.  
  924. function history(){
  925. var divHist = getId('divHist');
  926. try{
  927. if(divHist){
  928. killId(divHist);
  929. var hL = getId('historyLink');
  930. hL.innerHTML = 'History'; hL.className= 'gootransbutt gootranslink gtlActive';
  931. hL.title = 'Translation history';
  932. GM_setValue('histShow',false);
  933. return;
  934. }
  935. if(!maxHT) return;
  936. GM_setValue('histShow',true);
  937. divHist = buildEl('div', {id:'divHist'},['click', histLookup], null );
  938. //
  939. for(var i=0, l=ht.length; i<l; i++){
  940. var bkg = ht[i][0].indexOf(' ')>0 ? ' goohistlink' : '';
  941. addEl(divHist,'a', {href:HREF_NO, 'class': 'gootranslink'+bkg, 'titel': ht[i][1]+
  942. ((ht[i][3]>1) ? '\u00A0'+ '['+ht[i][3]+']' : '')},
  943. null, ht[i][0]);
  944. if(i < l-1)
  945. divHist.appendChild(document.createTextNode(' '));
  946. }
  947. //addEl(divHist,'span',null,null,'<br>&nbsp;');
  948. if(getId('divSourceshow'))
  949. insAfter(divHist,getId('divSourceshow'));
  950. else
  951. insAfter(divHist,getId('divResult'));
  952. var hL=getId('historyLink')
  953. // hl.textContent = 'X';
  954. hL.title= 'Hide history';
  955. hL.innerHTML = 'History'; hL.className = 'gootransbutt gootranslink gtlActive';
  956. }catch(e){console.log('hist problem\n'+e)}
  957. }
  958.  
  959. function saveOptions(evt){
  960. try{
  961. var from = getId('optSelLangFrom').value;
  962. var to = getId('optSelLangTo').value;
  963. var ctrl = getId('checkCtrl').checked;
  964. var alt = getId('checkAlt').checked;
  965. var mh = parseInt(getId('histSize').value);
  966. var wc = parseInt(getId('histWc').value);
  967. var nf = getId('checkNoflags').checked;
  968. if(0<= mh && mh <=99 && mh<maxHT){
  969. while(ht && ht.length>mh) ht.pop();
  970. GM_setValue('hist',ht? JSON.stringify(ht):'');
  971. if(getId('divHist')){
  972. killId('divHist');
  973. history();
  974. }
  975. }
  976. maxHT=mh;
  977. var bh = parseInt(getId('sourceBH').value);
  978. if( 0< bh && bh <10) sourceBH = bh;
  979. var dp = parseInt(getId('sourceDP').value);
  980. if( 0< dp && dp <100) sourceDP = dp;
  981. if(1<= wc && wc <=9) maxWC=wc;
  982. GM_setValue('histSize',maxHT)
  983. GM_setValue('histWc',maxWC);
  984. GM_setValue('from', from);
  985. GM_setValue('to', to);
  986. GM_setValue('ctrl', ctrl);
  987. GM_setValue('alt', alt);
  988. GM_setValue('sourceBH', sourceBH);
  989. GM_setValue('sourceDP', sourceDP);
  990. GM_setValue('noFlags',nf)
  991. getId('divDic').removeChild(getId('divOpt'));
  992. getId('optionsLink').title='Settings';
  993. return;
  994. }catch(e){console.log('saveOpnions\n'+e);}
  995. }
  996.  
  997. function addHistory(src,trt){
  998. if (!maxHT) return;
  999. try{
  1000. var hts=GM_getValue("hist");
  1001. if( !hts ){
  1002. ht=[["google translator","Der Ubersetzer","en|de",0]];
  1003. hts=JSON.stringify(ht);
  1004. }
  1005. ht=JSON.parse(hts);
  1006. var st=trim(src+''); var tt = trim(trt+'');
  1007. var wc = (st.split(' ')).length;
  1008. if(wc>maxWC) return;
  1009. var lang=currentURL.match(/langpair=([a-zA-Z-\|]+)/)[1];
  1010. var ix=-1; // find word in hist
  1011. for(var i=0, l=ht.length; i<l; i++)
  1012. if(st==ht[i][0]){ ix=i; break; }
  1013. //if(ix==0) return; // nothing to do
  1014. var hits=0;
  1015. if(ix>=0){
  1016. hits=ht[ix][3];
  1017. if( (gt_sl+'|'+gt_tl) == ht[ix][2] && tt != ht[ix][1]) {
  1018. tt = ht[ix][1]; // don't touch my translasion
  1019. }
  1020. ht.splice(ix,1);
  1021. }
  1022. // if(hits<4)
  1023. hits++; // delete it by your own hands
  1024. if (ht.length>maxHT){
  1025. var minHit=99999; // which item shoud i remove?
  1026. for(i=ht.length-1;i>0;i--)
  1027. if(minHit>ht[i][3]) minHit=ht[i][3]
  1028. ix=ht.length-1;
  1029. for(var i=ix; i>0; i--)
  1030. if(minHit==ht[i][3]){ ix=i; break; }
  1031. ht.splice(ix,1);
  1032. }
  1033. ht.unshift([st,tt,lang,hits]);
  1034. GM_setValue('hist',JSON.stringify(ht));
  1035. } catch(e){console.log('addHist\n'+e);}
  1036. }
  1037. var senFlag = '';
  1038. function selFlag(e){
  1039. if(!isInited) {css(-1); isInited=true; }
  1040. killId('divSelflag');
  1041. setEscHnd();
  1042. var p = belowCursor(e,10,10);
  1043. var dsf = buildEl('div',{id:'divSelflag', style:
  1044. ';top:'+p.t+';left:'+p.l+';right:'+p.r +';bottom: auto'});
  1045. var sel=addEl(dsf,'select',{id: 'optSelFlag'},
  1046. null,languagesGoogle);
  1047. sel.value = GM_getValue('to',' en');
  1048. addEl(dsf,'span',null,null,'<br><br>');
  1049. addEl(dsf,'a', {href:HREF_NO,
  1050. 'class':'gootransbutt gootranslink', title: "use icon"},
  1051. ['click', function(){saveFlag(true)}], '<b>&nbsp; OK &nbsp;</b>');
  1052. addEl(dsf,'a', {href:HREF_NO,
  1053. 'class':'gootransbutt gootranslink'},
  1054. ['click', function(){saveFlag(false)}],
  1055. '<b>&nbsp; cancel &nbsp; </b>');
  1056. //
  1057. senFlag = e.target.src+'';
  1058. var sm = senFlag.match(/.+\/(.+)\.png/);
  1059. if(sm && sm[1]) senFlag= sm[1];
  1060. _log(senFlag);
  1061. if(senFlag) body.appendChild(dsf);
  1062. }
  1063. function saveFlag(tf){
  1064. if(tf && senFlag){
  1065. var s= 'l-'+getId('optSelFlag').value;
  1066. GM_setValue(s,senFlag);
  1067. _log(s+': '+senFlag );
  1068. }else _log('cant save flags' );
  1069. killId('divSelflag'); return;
  1070. }
  1071. var fCSS;
  1072. function flagClick(e){
  1073. e.preventDefault();
  1074. if(e.target.nodeName == 'IMG'){
  1075. _log('hit on: ' + e.target.alt);
  1076. selFlag(e);
  1077. }else (killId('divSelflag'));
  1078. }
  1079. function belowCursor(evt,ho,vo,lr){
  1080. var p={t:'', l:'auto', r:'auto'};
  1081. p.t=(evt.clientY+window.pageYOffset+vo)+'px';
  1082. var l=(evt.clientX+window.pageXOffset+ho)+'px';
  1083. if(lr && lr=='r')
  1084. { p.l=l; return p; }
  1085. var w = window.innerWidth;
  1086. var r=(w-(evt.clientX+window.pageXOffset)+ho);
  1087. // Q: How to detect visibility & thickess of vertical scrollbar?
  1088. r-=8; //
  1089. if(r<0) r= w/4;
  1090. r+='px';
  1091. if(lr && lr=='l')
  1092. { p.r=r; return p; }
  1093. if(evt.clientX < w/2)
  1094. p.l=l;
  1095. else
  1096. p.r=r;
  1097. return p;
  1098. }
  1099. var flagLang;
  1100. function getFlagSrc(lng, where){
  1101. if(!where) where = 'to';
  1102. if(GM_getValue('noFlags')) return imgFlags[where];
  1103. var fl='l-'+lng;
  1104. var flag = GM_getValue(fl,'');
  1105. if(!flag){
  1106. flag=imgFlags[lng];
  1107. if(!flag) return flag = imgFlags[where];
  1108. }
  1109. if(flag.indexOf('http')==0
  1110. //flag.indexOf('file') ==0
  1111. ||flag.indexOf('data:') ==0)
  1112. return flag;
  1113. flagLang=fl;
  1114. // flag= 'http://www.senojflags.com/images/national-flag-icons/'+flag+'-Flag.png';
  1115. flag= senoimg+flag+senoext;//'-Flag.png';
  1116. flagRequest(flag);
  1117. return flag;
  1118. }
  1119. function flagRequest(f){
  1120. _log('load '+f);
  1121. GM_xmlhttpRequest({
  1122. method: 'GET',
  1123. url: f,
  1124. binary: true,
  1125. overrideMimeType: "text/plain; charset=x-user-defined",
  1126. headers: {
  1127. "User-Agent": UA
  1128. ,"Accept": "image/png,image/*;q=0.8,*/*;q=0.5"
  1129. ,"Accept-Encoding": "gzip, deflate"
  1130. },
  1131. onload: function(resp) {
  1132. try{
  1133. flagStore(resp.responseText,f);
  1134. }catch(e){console.log('FlagRqst\n'+e);}
  1135. }
  1136. });
  1137. }
  1138. function flagStore(r,url){
  1139. if(r.indexOf("<head")>=0)
  1140. {
  1141. console.log("Banned!\n"+url);
  1142. GM_setValue(flagLang,url);
  1143. }
  1144. else
  1145. GM_setValue(flagLang,"data:image/png;base64," + b2b64(r));
  1146. }
  1147.  
  1148. function trim(s){
  1149. return (s+'').replace(/\s+/g,' ').replace(/^\s/,'').replace(/\s$/,'');
  1150. }
  1151.  
  1152. function killId(nod){
  1153. if(!nod) return;
  1154. var n = nod;
  1155. if(typeof n == 'string'){
  1156. n= getId(nod);
  1157. }
  1158. if(!n) return;
  1159. if(n.parentNode) n.parentNode.removeChild(n);
  1160. else _log('cant kill: '+nod)
  1161. }
  1162.  
  1163. function addEl(to,type, attrArray, eL, html){
  1164. return to.appendChild(buildEl(type, attrArray, eL, html));
  1165. }
  1166.  
  1167. function buildEl(type, attrArray, eL, html)
  1168. {
  1169. var node = document.createElement(type);
  1170. for (var attr in attrArray) if (attrArray.hasOwnProperty(attr)){
  1171. node.setAttribute(attr, attrArray[attr]);
  1172. }
  1173. if(eL){
  1174. //console.log('buildEl\n'+type+'\n'+JSON.stringify(attrArray)+'\n'+eL[0])
  1175. node.addEventListener(eL[0], eL[1], eL[2]?true:false);
  1176. }
  1177. if(html)
  1178. node.innerHTML = html;
  1179. return node;
  1180. }
  1181.  
  1182. function getId(id, parent){
  1183. if(!parent)
  1184. return document.getElementById(id);
  1185. return parent.getElementById(id);
  1186. }
  1187.  
  1188. function getXId(id){
  1189. var r=Xel('.//*[@id="'+id+'"]',divExtract);
  1190. if(r) return r;
  1191. throw "Xel bug " + id;
  1192. }
  1193.  
  1194. function Xel(XPath, contextNode){
  1195. var a = document.evaluate(XPath, (contextNode || document),
  1196. null, XPathResult.ORDERED_NODE_SNAPSHOT_TYPE, null);
  1197. return (a.snapshotLength ? a.snapshotItem(0) : null);
  1198. }
  1199.  
  1200. /** /
  1201. function Xels(XPath, contextNode){
  1202. var ret=[], i=0;
  1203. var a = document.evaluate(XPath, (contextNode || document),
  1204. null, XPathResult.ORDERED_NODE_SNAPSHOT_TYPE, null);
  1205. while(a.snapshotItem(i))
  1206. ret.push(a.snapshotItem(i++));
  1207. return ret;
  1208. }/**/
  1209. //
  1210.  
  1211. function getTag(name, parent){
  1212. if(!parent)
  1213. return window.document.getElementsByTagName(name);
  1214. return parent.getElementsByTagName(name);
  1215. }
  1216. /*
  1217. * Drag and drop support adapted fom http://www.hunlock.com/blogs/Javascript_Drag_and_Drop
  1218. */
  1219. var savedTarget=null; // The target layer (effectively vidPane)
  1220. var orgCursor=null; // The original mouse style so we can restore it
  1221. var dragOK=false; // True if we're allowed to move the element under mouse
  1222. var dragXoffset=0; // How much we've moved the element on the horozontal
  1223. var dragYoffset=0; // How much we've moved the element on the verticle
  1224. var didDrag=false; //set to true when we do a drag
  1225. function moveHandler(e){
  1226. if (e == null) return;// { e = window.event }
  1227. if ( e.button<=1 && dragOK ){
  1228. savedTarget.style.left = e.clientX - dragXoffset + 'px';
  1229. savedTarget.style.top = e.clientY - dragYoffset + 'px';
  1230. return false;
  1231. }
  1232. }
  1233. function dragCleanup(e) {
  1234. document.removeEventListener('mousemove',moveHandler,false);
  1235. document.removeEventListener('mouseup',dragCleanup,false);
  1236. savedTarget.style.cursor=orgCursor;
  1237. dragOK=false; //its been dragged now
  1238. didDrag=true;
  1239. }
  1240. function dragHandler(e){
  1241. var htype='-moz-grabbing';
  1242. if (e == null) return;//
  1243. var target = e.target;// != null ? e.target : e.srcElement;
  1244. orgCursor=target.style.cursor;
  1245. if(target.nodeName!='DIV' )
  1246. return;
  1247. if( e.ctrlKey || e.altKey || e.shiftKey)
  1248. return; // enable selection inside
  1249. else if(clickedInsideID(target, res_dict))
  1250. return;
  1251. if (target = clickedInsideID(target, 'divDic')) {
  1252. savedTarget=target;
  1253. target.style.cursor=htype;
  1254. dragOK=true;
  1255. dragXoffset = e.clientX-target.offsetLeft;
  1256. dragYoffset = e.clientY-target.offsetTop;
  1257. //set the left before removing the right
  1258. target.style.left = e.clientX - dragXoffset + 'px';
  1259. target.style.right = null;
  1260. document.addEventListener('mousemove',moveHandler,false);
  1261. document.addEventListener('mouseup',dragCleanup,false);
  1262. return false;
  1263. }
  1264. }
  1265. function clickedInsideID(target, id) {
  1266. if (target.getAttribute('id')==id)
  1267. return target;
  1268. if (target.parentNode) {
  1269. while (target = target.parentNode) {
  1270. try{
  1271. if (target.getAttribute('id')==id)
  1272. return target;
  1273. }catch(e){}
  1274. }
  1275. }
  1276. return null;
  1277. }
  1278. //end drag code
  1279. function b2b64(inp) { // binary data --> base64
  1280. var output = [], c1, c2, c3, e1, e2, e3, e4, i = 0;
  1281. var k="ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/=";
  1282. var len = inp.length;
  1283. while( i < len ){
  1284. c1 = inp.charCodeAt(i++); c2 = inp.charCodeAt(i++); c3 = inp.charCodeAt(i++);
  1285. e1 = (c1&255) >> 2;
  1286. e2 = ((c1 & 3) << 4) | ((c2&255) >> 4);
  1287. e3 = ((c2 & 15) << 2) | ((c3&255) >> 6);
  1288. e4 = c3 & 63;
  1289. if( isNaN(c3)) e4 = 64;
  1290. if( isNaN(c2)) e3 = 64;
  1291. output.push( k.charAt(e1) + k.charAt(e2) + k.charAt(e3) + k.charAt(e4));
  1292. } return output.join("");
  1293. }
  1294.  
  1295.  
  1296. function ttrans( s, ttab ){
  1297. for( var c, cc, t='', il=s.length, i=0; i<il; i++ )
  1298. t += (c=ttab[cc=s.charAt(i)]) ? c : cc;
  1299. return t;
  1300. }
  1301. var tabUrlEsc = {
  1302. '#':'%23', '%':'%25', '&':'%26', '.':'%2e', '/':'%2f', '?':'%3f'
  1303. };
  1304. function escAp(s){
  1305. return ttrans( s, tabUrlEsc );
  1306. }
  1307.  
  1308. function stickStyle(css){
  1309. var s=document.createElement("style"); s.type="text/css";
  1310. s.appendChild(document.createTextNode(css));
  1311. return (document.head||document.documentElement).appendChild(s);
  1312. }
  1313.  
  1314. function css(n){
  1315. var k,i=0;
  1316. _log('cssN:',n);
  1317. if(-1 === n){ try{
  1318. k= +(GM_getValue('backG',0));
  1319. if(0<=k && k< BG.C.length) i=k;
  1320. } catch(e){};}
  1321. else GM_setValue('backG', +(i=+n) );
  1322. try{
  1323. if(window.gttpCSS) window.gttpCSS.parentNode.removeChild(window.gttpCSS);
  1324. }catch(e){console.log('cssX:\n'+e)};
  1325. _log('cssS:'+i,BG.C[i]);
  1326. window.gttpCSS=
  1327. stickStyle((
  1328. '#divResult {overflow: auto !important; padding:3px !important; margin: 0 0 3px 0 !important; max-height: 480px !important;}'+
  1329. '#divResult table *{ line-height: .85em !important}'+
  1330. '#divDic, #divDic *, #divSelflag, #divSelflag * {\
  1331. font-family: Tahoma, sans-serif!important;\
  1332. font-size: small!important;\
  1333. font-style: normal!important;\
  1334. font-weight: normal!important;\
  1335. line-height: 1.1;}'+
  1336. '#divDic,#divSelflag {position: absolute; background:'+BG.C[i]+'!important; color:'+FG.t[i]+
  1337. '!important; opacity: 1'+
  1338. ';padding:5px !important; margin:0; z-index:10000; border-radius:3px; border: solid thin gray'+
  1339. ';text-align: left !important;}'+
  1340. '#divDic{/*min-width: 340px !important; min-height:50px;*/ max-width:50%; padding: 3px; margin: 0;}'+
  1341. '#divSelflag{ max-width: 180px; }'+
  1342. '.gootranslink, a.gootranslink '+
  1343. '{color:'+FG.l[i]+'!important; text-decoration: none !important; font: small normal sans-serif !important;'+
  1344. 'cursor:pointer !important; }' +
  1345. 'a.gootranslink:visited {color:'+FG.l[i]+'!important; text-decoration: none !important;}'+
  1346. 'a.gootranslink:hover {color:'+FG.l[i]+'!important; text-decoration: underline !important;}' +
  1347. 'a.gootranslink:active {color:'+FG.l[i]+'!important; text-decoration: underline !important;}' +
  1348. '#gtp_dict tbody> tr {font-size:14px !important; line-height:.9em!important;color:'+FG.t[i]+';'+
  1349. 'background:'+BG.H[i]+
  1350. '}'+
  1351. 'a.goohistlink {background:'+BG.F[i] +';}'+
  1352. '#gtp_dict {margin: 0; position: relative;}'+
  1353. '#gtp_dict ol {padding: 0 .5em 0 0; margin-left: 0.2em;}'+
  1354. '#gtp_dict li {list-style: square inside; display: list-item;}'+
  1355. 'div#gtp_dict tr>td {padding-left: .25em; vertical-align:top; border:0px; color:'+FG.t[i]+'; }'+
  1356. '#optSelLangFrom,#optSelLangTo {max-width: 150px; text-align: left !important; height: 1.5em;\
  1357. }'+
  1358. '#optSelLangFrom,#optSelLangTo,#divDic input[type="textbox"]{background:'+BG.E[i]+'!important;\
  1359. color:'+FG.t[i]+'!important;\
  1360. padding-bottom: 3px !important; margin-bottom: 4px!important;}'+
  1361. '#divExtract{word-spacing: normal !important;}'+
  1362. '#divBottom {position: relative; width: 100%; font-size: smaller; text-decoration:none; }'+
  1363. '#historyLink {display: inline; position: relative; font-size:smaller; text-decoration:none;}'+
  1364. '#sourceLink {display: inline; position: relative; margin-left: 2em; font-size:smaller; text-decoration:none;}'+
  1365. '#imgSourcesave {display: inline; position: relative; margin-left:2px;}'+
  1366. '#optionsLink {display: inline; position: relative; padding-left: 1em; margin-left: 1em; font-size:smaller !important; text-decoration:none !important;}'+
  1367. '#divOpt {position: relative; padding: 5px;'+
  1368. 'border-top: thin solid grey;}'+
  1369. '#divLookup, #divOpt, #divBottom,#divSourcetext,#divHist,#divuse {direction: ltr !important;}'+
  1370. '#divHist {background:'+BG.C[i]+'; position:relative; padding:5px; text-align:left !important;'+
  1371. 'border-top: thin solid grey; color:'+FG.t[i]+';}'+
  1372. 'div#divResult #gtp_dict {background:'+BG.C[i]+'; color:'+FG.t[i]+'; padding:1px!important; border-radius:3px;'+
  1373. 'margin-bottom: .1em!important; overflow-y:auto !important; overflow-x:hidden; font-size:small;}'+
  1374. '#divOpt {background:'+BG.C[i]+'; position:relative; padding:5px; text-align:left !important;}'+
  1375. '#divLookup, #divUse {background-color:transparent; color:'+FG.t[i]+'; position:absolute; padding: 3px;}'+
  1376. 'div#divDic>#divSourceshow {\
  1377. border: none; padding: 0 0 4px 0; margin: 0;}'+
  1378. '#divSourceshow>#divSourcetext{ width:97%; height: 3em; line-height: 1.2em; overflow: auto !important;\
  1379. padding: 0 0 0 4px; margin: 0; border: 0; border-top: 1px solid #AAA}' +
  1380. '.gtlPassive:before{ content:"\u2193";}'+
  1381. '.gtlActive:before{ content:"\u2191" !important;}'+
  1382. '#imgUse, #divGetback, #divGetforw {margin-left: 5px !important; cursor: pointer;}'+
  1383. '#divSourcetext {background:'+BG.E[i]+'; color:'+FG.t[i]+'!important;}'+
  1384. '#divDic .gootransbutt {background:'+BG.T[i]+';'+
  1385. 'border-radius: 3px; margin-top: 5px; }'+
  1386. '.goounsaved {background-color: #EF9024;'+
  1387. 'border-radius: 3px; margin-top: 5px; }'+
  1388. 'td.gtp-pos { color:'+FG.t[i]+'!important; font-weight: bold !important; text-align: left; }'+
  1389. 'td.gtp-pos:before{ content:"\u2666 ";}'+
  1390. 'td.gtp-word {color:'+FG.t[i]+'!important; padding-left: 5px; padding-right: 10px;'+
  1391. 'vertical-align: top; white-space: normal;}'+
  1392. 'td.gtp-trans {/*overflow-x: hidden;*/ vertical-align: top; white-space: normal;'+
  1393. ' width: 100%; color:'+FG.g[i]+'!important}'+
  1394. 'td.gtp-pos, td.gtp-word, td.gtp-trans {padding-bottom: 0px !important; padding-bottom: 1px !important;}'+
  1395. '.gtp-hide {display: none}'+
  1396. '.gtp-block {display: block}'+
  1397. '')
  1398. );
  1399.  
  1400. if(-1 !== n) return;
  1401. stickStyle('\
  1402. #divDic *::-moz-selection {background: #047 !important; color: #FC8 !important; }'+
  1403. '#divUse img, #divDic img, #divLookup img {width: auto; height: auto; }'+ // rt.com :/
  1404. '#divTtsLnk:after{ content:url('+imgPlay+') }'+
  1405. '#divTtsLnk {padding: 0 3px; margin: 0 4px 0 0;}'+
  1406. '#divTtsIfh {width: 100%;overflow-x:hidden;\
  1407. background-color: rgba(127,127,127,.25); padding: 3px 0;\
  1408. }'+
  1409. '#divResult, #divResult div, #divResult table, #divResult tr, #divResult tr td,\
  1410. #divResult a, #divBottom, \
  1411. #divOpt select, #divOpt input, .gootranslink\
  1412. { padding:0 0 0 0; margin: 0 0 0 0; background: none repeat scroll 0 0 transparent;\
  1413. border:medium none; line-height: 0.95; float: none}'+
  1414. '#divOpt {line-height: 2.3 !important;}\
  1415. div#divBottom{padding-top: 3px;}\
  1416. .gootransbutt#optionsLink{margin-top:0; padding-top: 3px; padding-bottom: 1px;}\
  1417. .gtBGColor{border:thin solid blue !important; cursor: pointer;\
  1418. padding-right:6px; margin-right: 2px;}\
  1419. .gootranslink[titel]{position: relative;}\
  1420. .gootranslink[titel]:after {\
  1421. color: #050;\
  1422. content: attr(titel);\
  1423. position: absolute;\
  1424. visibility: hidden;\
  1425. opacity: 0.25;\
  1426. left: 16px;\
  1427. top: 32px;\
  1428. min-width: 0px;\
  1429. width: auto;\
  1430. white-space: normal !important;\
  1431. -moz-hyphens: none !important;\
  1432. text-decoration: none !important;\
  1433. border: 1px #aaa solid;\
  1434. border-radius: 6px;\
  1435. background-color: #dfd;\
  1436. padding: 1px 4px;\
  1437. font-size: 14px;\
  1438. -webkit-transition: all .2s linear .2s;\
  1439. transition: all .2s linear .2s;\
  1440. z-index: 2147483647;\
  1441. }\
  1442. .gootranslink[titel]:hover:after {\
  1443. visibility: visible;\
  1444. opacity: 1;\
  1445. -webkit-transition: all .5s linear .7s;\
  1446. transition: all .5s linear .7s;\
  1447. }\
  1448. ');
  1449. }
  1450. function insAfter(n,e){
  1451. if(e.nextElementSibling){
  1452. e.parentNode.insertBefore(n,e.nextElementSibling);
  1453. }else{
  1454. e.parentNode.appendChild(n);
  1455. }
  1456. }
  1457. function insBefore(n,e){
  1458. e.parentNode.insertBefore(n,e);
  1459. }
  1460. var imgH='<img border=0 src="' , imgD='data:image/png;base64,',imgT='">';
  1461. var imgPlay=imgD+
  1462. 'iVBORw0KGgoAAAANSUhEUgAAAAwAAAAMCAYAAABWdVznAAAACXBIWXMAAC4jAAAuIwF4pT92AAAAGXRFWHRTb2Z0d2FyZQBBZG9iZSBJbWFnZVJlYWR5ccllPAAAAI1JREFUeNpj+H9h4v//D7YRh4FqGf6vsvuPF4PAnnQ4n+H/At3/WDEIgOjTvQg2EDP8ny7/HwWDAIy+sxnBvr0RzGb43y/0H45hAMR+fAQrm+F/B/t/DAASe3gQK5vhfwsDpgaQ2IP9WNkQDTBMsgaYJhgNcgqMfWUZDg3oGveUINh4NSBrnGdGggY0DAAAy70TuBaoTgAAAABJRU5ErkJggg=='
  1463. ;
  1464.  
  1465. //http://www.senojflags.com/images/national-flag-icons/Portugal-Flag.png
  1466. imgForwSrc= imgD+
  1467. 'R0lGODlhEAAQAIQfAAFYsQJkw15iWgRz3AeN0Xl8c2mBiQma5B6W1h+b3yqh4BKp+kWk0pudlDC2/0y79X+4z1fC/7q6r3DK/4rN7a3Kz8fKwc/Uyr3c5N7dzuTi1Ofl1enm2O3r3/b06////yH+EUNyZWF0ZWQgd2l0aCBHSU1QACH5BAEKAB8ALAAAAAAQABAAAAWf4OeNXmdyX6qKpNSc25p6TVEIgl1osljntxxPNrIILhrDZBl5OBaLj6mTa1CWk+YTOpXgsBEM5gFdmDLBiJrC2VCgB85lYJFcmg+MPAPnDP4aGg4ODxsVFBkPBwd/jRuDFBoQDxkVix+NHwBQSBmeFwoEHwEDAQEAAJQVDAwQFxCipqeoABkMiwcWFqKjpqgEBLjAwDIAHwoJyQkIzB8hADs=';
  1468. imgForw = imgH+imgForwSrc+imgT;
  1469.  
  1470. imgBackSrc= imgD+
  1471. 'iVBORw0KGgoAAAANSUhEUgAAABAAAAAQCAMAAAAoLQ9TAAAAAXNSR0IArs4c6QAAAGBQTFRFEgAAAVixAmTDXmJaBHPcB43ReXxzaYGJCZrkHpbWH5vfKqHgEqn6RaTSm52UMLb/TLv1f7jPV8L/urqvcMr/is3trcrPx8rBz9TKvdzk3t3O5OLU5+XV6ebY7evf9vTrdcYePwAAAAF0Uk5TAEDm2GYAAAABYktHRACIBR1IAAAACXBIWXMAAAsTAAALEwEAmpwYAAAAoklEQVQY003PWxKDIAwF0CCPYouCKIIB7f532UTttPfvnkkyE4Ar+3Ecb8pdoRGM859UY4yUxow/4CpPcM6HuCzLoxaZacV9IY1G0mWGsG0sUs4M1rnU9hRjpCt4AZZ9CyGWOauyg7UB09qC97UqpRhWDFNN3jfFAf0qiFiLcwK4g57KNAw0JUTXqY4gZ2vtgEKwMFBItL47QN/3T8oLxPnVB4gUDOnY6pKLAAAAAElFTkSuQmCC';
  1472. imgBack = imgH+imgBackSrc+imgT;
  1473.  
  1474. imgSwap = "<img border=0 style="+'"margin-bottom: -3px;"'+
  1475. "src=''>";
  1476.  
  1477. 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=';
  1478.  
  1479. imgSave= imgD+
  1480. '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';
  1481.  
  1482. imgFlags= {
  1483. 'AN': imgD+
  1484. '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='
  1485. ,'en': imgD+
  1486. '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='
  1487. ,'fr': imgD+
  1488. 'iVBORw0KGgoAAAANSUhEUgAAABAAAAAQCAYAAAAf8/9hAAAACXBIWXMAAAsTAAALEwEAmpwYAAAAIGNIUk0AAHolAACAgwAA+f8AAIDpAAB1MAAA6mAAADqYAAAXb5JfxUYAAAFESURBVHjapJM7TsNAEIa/9RqjKECEkGgoKOgoaHKCdByBc3APDsIRaHID01ASRTxEgxI5zsvetb1DkcSJCUaKMtJqZzSrb2f/nVEiwj6mgAA4Wu67mAWmPtDqdB6+f2e73ftK3L+92yJcPT2e+0CzKBzt9uW/1zVurpc1KxBh/vwC0PQBbW3BeJwyHE5rAebto/T9s1MkywC0D2CMJYpmRNG8FuBG49IvPE1u0wUMIEkMg8GMOJ5T9yn5KC59AWxiNgEJvV5GHCf1kvffS1+3Tkh9bw1I0wSRBs65ehU3cpJlpJnbBBi0PsS5+qaSFUBAihyTWwC8hYgG50ApVa6tjlvlPAVOSIxdV2CtJQiEIPDrnxAcVKoxdi2iFSmYTD4r58MwrMSvcfSntgpoARfA8Y6zMAG+FKCXg6R3BBSAVfuO888AocKXohfLXWQAAAAASUVORK5CYII='
  1489. ,'ru': imgD+
  1490. 'iVBORw0KGgoAAAANSUhEUgAAABAAAAAQCAYAAAAf8/9hAAAACXBIWXMAAAsTAAALEwEAmpwYAAAAIGNIUk0AAHolAACAgwAA+f8AAIDpAAB1MAAA6mAAADqYAAAXb5JfxUYAAADkSURBVHjapJM5TgNBEEVfzeZlQIOE5ARxClLOxC04CzFX4ArERAhZsoTEyDDurqa7SGxsIYJZKqnov9p+iZkxJQSogLN9HhIKfBZAY2abUdVFVhlQT5igzoB8AiDPAGKMg5UHTQFwc/fI6rKh/VLsz4b/u9FFXbF5b4+AnXe8rIW2C72qN8uSwtwJoHMwm5NSP0+EbyP4E8DD0z3XZUnqdr0A2XLBawjcHgBOPZYXINILYMlw6o8deFVsvoCq7AewhFf9BWg047n9GOMDFaABroDzgeIt8CZ7J1YjHBkBlanv/DMAwHdYum9dlZQAAAAASUVORK5CYII='
  1491. ,'zh-CN': imgD+
  1492. 'iVBORw0KGgoAAAANSUhEUgAAABAAAAAQCAYAAAAf8/9hAAAACXBIWXMAAAsTAAALEwEAmpwYAAAAIGNIUk0AAHolAACAgwAA+f8AAIDpAAB1MAAA6mAAADqYAAAXb5JfxUYAAAFbSURBVHjapJO/SgNBEIe/vduLCSaKFhb6DCnFzs4XsMgjWlj4AD6BFhYKFqKgTSzSaBLN7f+xuJM7RSHRHywLy8w3Oz9mlIjwHymgA/TrexU54E0Dm+Pj0QSACGRADt39gLnQTZkftHd2uqOBdYmR3uGQYjfhnzPSXJHeYTCK2Lu8AquGJYC9vgFY10AupUdvTekdTYgnGvdYVQ5P0nTa+kW+vYV4D5BrgGAt9nZOmCzwDxlxkqG60D0ImCsN/msbKs+J1gKgAXww+Kcp5nKBKgQ0rA09i/Oi8iR880HAGdMATFnSuX8kzWZ8vvqHJvi7idnGBibPGkBZlvSzDJFURfj6FC1ISyl4SpcagDUWVRRISi2ngPTL8PiArUysW3AW0RqUWmqCRATjWiZaa6HbQxXFcjOYUpVTA1wU4eb15S+r4BSwCewBgxWT58BY1XZ16nsVRcCp/67zxwDGd5ld8bkQAQAAAABJRU5ErkJggg=='
  1493. ,'ar': imgD+'iVBORw0KGgoAAAANSUhEUgAAABAAAAAQCAYAAAAf8/9hAAAACXBIWXMAAAsTAAALEwEAmpwYAAAAIGNIUk0AAHolAACAgwAA+f8AAIDpAAB1MAAA6mAAADqYAAAXb5JfxUYAAAIvSURBVHjapJPNSxRxHMY/M/Ob3bXddV3DBTUh8yJdCiHwIIV0DXqB7nWLDv0NBt06Rt3q0E0PGaFBmGSGUCZKQR6ENV8wddfVfZ35zW/mN9NByd4u4nP5Hr7wgYfneYwoijiODCAGpA7uUaSAugAyN0aHCn9//TAg0BoAYZrYlv0P4cX1oZwAkjrUnMv1ABBFEX6oaU+1cibbARGsVbZZrxWxTRPDMAH4UsgDJAVg+WFATTmUZBWtNVd7L3Ktux9ZaxBvShDriTO2/InhxSlMA042NeOHAYAlADzfpyIbFBplLp/u42bnBRzHYWVjHa01bdlWrnT0sV4t8mppBmFYeL4PgAkgPUnJq9LQHpdyvbizn9koFgl8n3gsxlZ5D39+gcFsDwrNjqwiPXkIcKVHvrxJwa3SIgOcmRkCIUil0+yUSkTxOLsfpmltKHaVQ778A1d6vwMcwlBTk3W2kzaJ5VWyc/Ns1euYmWbK468xZ+cpZBJU3CphqHGls58QgJQSO2njKo+RlY8M3L3Dzq3bdHSdoimdRn9bpO3xEx5tLrDn1MidaMGX8hCgpIeIkqTsBCMLb+gebOfe2Cjm+ASh72M+uM9T1ng28ZKUnSCKQtSBhf0UlCJBREzY2Jbg4fvnTHadpX/gPKZpMLc0zOzqV2xLYBkmIRGeUr8AKtIh5fzmHy2b/v6Oyam3YIDAIi5s5H/qbAAZoBNIH3ELNWDDAKyDIVlHBGhAGced888BANVaBfgg0AbGAAAAAElFTkSuQmCC'
  1494. ,"af":"Namibia" //
  1495. ,"sq":"Albania"
  1496. //,"ar":"United-Arab-Emirates"
  1497. ,"hy":"Armenia"
  1498. ,"az":"Azerbaijan"
  1499. ,"eu":"Spain"
  1500. ,"be":"Belarus"
  1501. ,"bn":"Bangladesh"
  1502. ,"bs":"Bosnian"
  1503. ,"bg":"Bulgaria"
  1504. ,"ca":"Spain"
  1505. ,"ceb":"Philippines"
  1506. ,"ny":"Mozambique"
  1507. /* ,"zh-CN":"Chinese"*/
  1508. ,"hr":"Croatia"
  1509. ,"cs":"Czech-Republic"
  1510. ,"da":"Denmark"
  1511. ,"nl":"Netherlands"
  1512. /*, "en":"English"*/
  1513. ,"eo":"United-Nations"
  1514. ,"et":"Estonia"
  1515. ,"tl":"Philippines"
  1516. ,"fi":"Finland"
  1517. /* ,"fr":"France"*/
  1518. ,"gl":"Ukraine"
  1519. ,"ka":"Georgia"
  1520. ,"de":"Germany"
  1521. ,"el":"Greece"
  1522. ,"gu":"India"
  1523. ,"ht":"Haiti"
  1524. ,"ha":"Nigeria"
  1525. ,"iw":"Israel"
  1526. ,"hi":"India"
  1527. ,"hmn":"Laos"
  1528. ,"hu":"Hungary"
  1529. ,"is":"Iceland"
  1530. ,"ig":"Nigeria"
  1531. ,"id":"Indonezia"
  1532. ,"ga":"Ireland"
  1533. ,"it":"Italy"
  1534. ,"ja":"Japan"
  1535. ,"jw":"Japan"
  1536. ,"kn":"India"
  1537. ,"kk":"Kazakhstan"
  1538. ,"km":"Cambodia"
  1539. ,"ko":"North-Korea"
  1540. ,"lo":"Laos"
  1541. ,"la":"Vatican-City"
  1542. ,"lv":"Latvia"
  1543. ,"lt":"Lithuania"
  1544. ,"mk":"Macedonia"
  1545. ,"mg":"Madagascar"
  1546. ,"ms":"Malaysia"
  1547. ,"ml":"India"
  1548. ,"mt":"Malta"
  1549. ,"mi":"New-Zealand"
  1550. ,"mr":"India"
  1551. ,"mn":"Mongolia"
  1552. ,"my":"Burma" //absent
  1553. ,"ne":"Nepal"
  1554. ,"no":"Norway"
  1555. ,"fa":"Iran"
  1556. ,"pl":"Poland"
  1557. ,"pt":"Brazil"
  1558. ,"pa":"Pakistan"
  1559. ,"ro":"Romania"
  1560. /*, "ru":"Russia"*/
  1561. ,"sr":"Serbia"
  1562. ,"st":"Lesotho"
  1563. ,"si":"Sri-Lanka"
  1564. ,"sk":"Slovakia"
  1565. ,"sl":"Slovenia"
  1566. ,"so":"Somalia"
  1567. ,"es":"Spain"
  1568. ,"su":"Sudan"
  1569. ,"sw":"Mozambique"
  1570. ,"sv":"Sweden"
  1571. ,"tg":"Tajikistan"
  1572. ,"ta":"India"
  1573. ,"te":"India"
  1574. ,"th":"Thailand"
  1575. ,"tr":"Turkey"
  1576. ,"uk":"Ukraine"
  1577. ,"ur":"Pakistan"
  1578. ,"uz":"Uzbekistan"
  1579. ,"vi":"Viet-Nam"
  1580. ,"cy":"Wales"
  1581. ,"yi":"Israel"
  1582. ,"yo":"Nigeria"
  1583. ,"zu":"South-Africa"
  1584. };
  1585. imgFlags['zh-TW'] = imgFlags['zh-CN'];
  1586. imgFlags['to'] = imgForwSrc; imgFlags['from'] = imgBackSrc;
  1587.  
  1588. languagesGoogle = '<option value="auto">Detect language</option>\
  1589. </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>\
  1590. ';
  1591.  
  1592. /* */
  1593. try{
  1594. //body = window;
  1595. //while(body.parent && body.parent != body) body=body.parent;
  1596. //body = body.document.body;
  1597. body=//document.documentElement.appendChild(document.createElement('section'));
  1598. window.document.body;
  1599.  
  1600. maxHT=GM_getValue('histSize');
  1601. if(!maxHT) maxHT=20;
  1602. maxWC=GM_getValue('histWc');
  1603. if(!maxWC) maxWC=3;
  1604. sourceBH = GM_getValue('sourceBH',3);
  1605. sourceDP = GM_getValue('sourceDP',10);
  1606. if(!sourceDP) sourceDP = 10;
  1607.  
  1608. var av = GM_getValue('version');
  1609. if( av!= version ){
  1610. GM_setValue('version',version);
  1611. }
  1612. sT=GM_getValue('sourceText');
  1613. if (sT){
  1614. try{
  1615. sT=JSON.parse(sT);
  1616. }catch(e){console.log('broken source\n'+e)} ;
  1617. } else sT='';
  1618.  
  1619. // gmail spoils my timeout -- workaround
  1620. // borrowed from dbaron.org/log/20100309-faster-timeouts :
  1621. // Only add setZeroTimeout to the window object, and hide everything
  1622. // else in a closure.
  1623. (function() {
  1624. var timeouts = [];
  1625. var messageName = "zero-timeout-130613";
  1626. // Like setTimeout, but only takes a function argument. There's
  1627. // no time argument (always zero) and no arguments (you have to
  1628. // use a closure).
  1629. function setZeroTimeout(fn) {
  1630. timeouts.push(fn);
  1631. window.postMessage(messageName, "*");
  1632. }
  1633. function handleMessage(event) {
  1634. if (event.source == window && event.data == messageName) {
  1635. event.stopPropagation();
  1636. if (timeouts.length > 0) {
  1637. var fn = timeouts.shift();
  1638. fn();
  1639. } } }
  1640. window.addEventListener("message", handleMessage, true);
  1641. // Add the one thing we want added to the window object.
  1642. window.setZeroTimeout = setZeroTimeout;
  1643. })();
  1644.  
  1645. document.addEventListener('mouseup', showLookupIcon, false);
  1646. document.addEventListener('mousedown', mousedownCleaning, false);
  1647. // http://www.senojflags.com/#flags16
  1648. if( location.href.indexOf(senojflags[0])>-1
  1649. ||location.href.indexOf(senojflags[1])>-1
  1650. ){try{
  1651. if(!fCSS) fCSS=
  1652. 'div#flags16 img {cursor: pointer !important}'+
  1653. 'div#flags48,div#flags32 {display:none; visibility: hidden}'
  1654. stickStyle(fCSS);
  1655. _log('inside\n' + location.href);
  1656. insBefore(buildEl('div',{style:'font: bold italic 90% sans-serif; color:red;',
  1657. align:'left'},null,'&nbsp;&nbsp;<u>Click on a country flag icon then choose the language</u>'),
  1658. getId('flags16').childNodes[0]);
  1659. getId('flags16').addEventListener('click',flagClick,false)
  1660. }catch(e){console.log('senojflags\n'+e)}
  1661. }
  1662. }catch(e){console.log('nobody\n'+e); }
  1663.  
  1664. function deURI(u,m){
  1665. if(!m) m= "&q=";
  1666. var x = u.indexOf(m);
  1667. if(x>=0) u=u.substr(x + m.length);
  1668. return decodeURIComponent(u).split(' ').slice(0,9).join(' ');
  1669. }
  1670. var uq=location.href.match(/^https:\/\/translate\.google\.[a-z]{2,3}\/translate_tts\?client\=t\&.+?(\&q\=.+)/);
  1671. if(uq && uq[1])
  1672. window.document.title=deURI(uq[1]);
  1673.  
  1674. }
  1675. main();
  1676. }
  1677. //