🏠 返回首頁 

Greasy Fork is available in English.

Google Translator Tooltip Expanded Fork

Translates the selected text into a tooltip automatically. Fork from https://greasyfork.org/scripts/5727/

  1. // ==UserScript==
  2. // @name Google Translator Tooltip Expanded Fork
  3. // @description Translates the selected text into a tooltip automatically. Fork from https://greasyfork.org/scripts/5727/
  4. // @namespace https://greasyfork.org/scripts/16204/
  5. // @homepage https://greasyfork.org/scripts/16204/
  6. // @version 1.29
  7. // @icon http://translate.google.com/favicon.ico
  8. // @include http*
  9. // @include https*
  10. // @include chrome*
  11. // @grant GM_getValue
  12. // @grant GM_xmlhttpRequest
  13. // @grant GM_log
  14. // @grant GM_deleteValue
  15. // @grant GM_addStyle
  16. // @grant GM_openInTab
  17. // @grant GM_registerMenuCommand
  18. // @grant GM_setValue
  19. // ==/UserScript==
  20. var UA = navigator.userAgent;
  21. var googleDomain = "translate.google.com";
  22. var dictURL= "https://" + googleDomain + "/translate_a/single?client=t";
  23. var ttsURL= "http://" + googleDomain + "/translate_tts?client=t";
  24. const HREF_NO = 'javascript:void(0)';
  25. initCrossBrowserSupportForGmFunctions();
  26. var languagesGoogle = '<option value="auto">Detect language</option><option value="af">Afrikaans</option><option value="sq">Albanian</option><option value="ar">Arabic</option><option value="hy">Armenian</option><option value="az">Aerbaijani</option><option value="eu">Basque</option><option value="be">Belarusian</option><option value="bn">Bengali</option><option value="bg">Bulgarian</option><option value="ca">Catalan</option><option value="zh-CN">Chinese (simplified)</option><option value="zh-TW">Chinese (traditional)</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="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="ht">Haitian Creole</option><option value="iw">Hebrew</option><option value="hi">Hindi</option><option value="hu">Hungarian</option><option value="is">Icelandic</option><option value="id">Indonesian</option><option value="ga">Irish</option><option value="it">Italian</option><option value="ja">Japanese</option><option value="ko">Korean</option><option value="lv">Latvian</option><option value="lt">Lithuanian</option><option value="mk">Macedonian</option><option value="ms">Malay</option><option value="mt">Maltese</option><option value="no">Norwegian</option><option value="fa">Persian</option><option value="pl">Polish</option><option value="pt">Portuguese</option><option value="ro">Romanian</option><option value="ru">Russian</option><option value="sr">Serbian</option><option value="sk">Slovak</option><option value="sl">Slovenian</option><option value="es">Spanish</option><option value="sw">Swahili</option><option value="sv">Swedish</option><option value="th">Thai</option><option value="tr">Turkish</option><option value="uk">Ukrainian</option><option value="ur">Urdu</option><option value="vi">Vietnamese</option><option value="cy">Welsh</option><option value="yi">Yiddish</option>';
  27. var body = getTag('body')[0];
  28. var imgLookup;
  29. var txtSel = encodeURIComponent(txtSel); // text selected
  30. var translation2Element = document.createElement('span');
  31. var translation3Element = document.createElement('span');
  32. var currentURL;
  33. var context;
  34. var initialized = false;
  35. images();
  36. css();
  37. document.addEventListener('mouseup', showLookupIcon, false);
  38. document.addEventListener('mousedown', mousedownCleaning, false);
  39. function mousedownCleaning(evt) {
  40. var divDic = getId('divDic');
  41. var divLookup = getId('divLookup');
  42. if (divDic) {
  43. if (!clickedInsideID(evt.target, 'divDic'))
  44. divDic.parentNode.removeChild(divDic);
  45. }
  46. if (divLookup)
  47. divLookup.parentNode.removeChild(divLookup);
  48. }
  49. function showLookupIcon(evt) {
  50. if (evt.ctrlKey && evt.altKey && (!GM_getValue('ctrl') || !GM_getValue('alt')))
  51. return;
  52. // XOR http://www.howtocreate.co.uk/xor.html
  53. if (evt.ctrlKey ? !GM_getValue('ctrl') : GM_getValue('ctrl'))
  54. return;
  55. if (evt.altKey ? !GM_getValue('alt') : GM_getValue('alt'))
  56. return;
  57. if (!initialized) {
  58. images();
  59. css();
  60. initialized = true;
  61. }
  62. var divDic = getId('divDic');
  63. var divLookup = getId('divLookup');
  64. txtSel = getSelection();
  65. // Exit if no text is selected
  66. if (!txtSel || txtSel == "") {
  67. if (divDic) {
  68. if (!clickedInsideID(evt.target, 'divDic'))
  69. divDic.parentNode.removeChild(divDic);
  70. }
  71. if (divLookup)
  72. divLookup.parentNode.removeChild(divLookup);
  73. return;
  74. }
  75. // Possible cleanup
  76. if (divDic) {
  77. if (!clickedInsideID(evt.target, 'divDic'))
  78. divDic.parentNode.removeChild(divDic);
  79. return;
  80. }
  81. // Remove div if exists
  82. if (divLookup) {
  83. divLookup.parentNode.removeChild(divLookup);
  84. }
  85. if(context!=null){
  86. context.close();
  87. context = null;
  88. }
  89. if(GM_getValue('tts', false) == true)context = new AudioContext();
  90. // Div container
  91. divLookup = createElement('div', {
  92. id : 'divLookup',
  93. style : 'background-color:transparent; color:#000000; position:absolute; top:' + (evt.clientY + window.pageYOffset + 10) + 'px; left:' + (evt.clientX + window.pageXOffset + 10) + 'px; padding:0px; z-index:10000; border-radius:2px;'
  94. });
  95. divLookup.appendChild(imgLookup.cloneNode(false));
  96. divLookup.addEventListener('mouseover', function(evt){setTimeout(lookup,parseInt(GM_getValue('delay')))}, false);
  97. body.appendChild(divLookup);
  98. }
  99. // Create the tooltip and launch the Google Translate request to get the translation
  100. function lookup(evt) {
  101. var divR###lt = null;
  102. var divDic = getId('divDic');
  103. var divLookup = getId('divLookup');
  104. var top = divLookup.style.top;
  105. var left = divLookup.style.left;
  106. // No text selected
  107. if (!txtSel || txtSel == "") {
  108. if (divDic = getId('divDic'))
  109. divDic.parentNode.removeChild(divDic);
  110. return;
  111. }
  112. // Cleanup divs
  113. if (divDic = getId('divDic')) {
  114. divDic.parentNode.removeChild(divDic);
  115. }
  116. divLookup.parentNode.removeChild(divLookup);
  117. // Div container
  118. divDic = createElement('div', {
  119. id : 'divDic',
  120. style : 'opacity: 1; font-family: "Open Sans", Arial, Helvetica, sans-serif !important; font-size: ' + GM_getValue('fontsize', 'small') + '; background-color: ' + GM_getValue('backgroundColor', '#EDF4FC') + '; color: ' + GM_getValue('textcolor', 'Gray') + '; position:absolute; top:' + top + '; left:' + left + '; min-width:250px; min-height:50px; max-width:50%; padding:5px; text-align:left; z-index:10000; border-radius:4px; box-shadow: -2px 0px 9px 5px #898D91'
  121. });
  122. divDic.addEventListener('mousedown', dragHandler, false);
  123. body.appendChild(divDic);
  124. // Div r###lt
  125. // This awfull wall of text is the "+" image
  126. divR###lt = createElement('div', {
  127. id : 'divR###lt',
  128. style : 'overflow:auto; padding:3px;'
  129. }, null, '<img src="###JhhjdkplWNzO/5oXI846njjVEIqR2OS2B1pE5PVscajkxhMycqLJgxQCwT40PjfAV4GqNSXYdZXJn5gSkmmmmJu1aZYb14V51do+pTOCmA00AqVB4hG5IJ9PvYnhIFOxmdqhpaI6GeHCtpooisuutmg+Eg62KOMKuqoTaXgicQWoIYq6qiklmoqFV0UoeqqrLbq6quwxirrrLTWauutJ4QAACH5BAkKAAAALAAAAADcABMAAAX/ICCOZGmeaKqubOu+cCzPdG3feK7vfO//wKBwSAQIBoSkcslsOp/QqHRKrVqv2Kx2OhC0BAXHx/EoCzboAcdhcLDdgwJ6nua03YZ8PMFPoBMca215eg98G36IgYNvDgOGh4lqjHd7fXOTjYV9nItvhJaIfYF4jXuIf4CCbHmOBZySdoOtj5eja59wBmYFXitdHhwSFRgKxhobBgUPAmdoyxoI0tPJaM5+u9PaCQZzZ9gP2tPcdM7L4tLVznPn6OQb18nh6NV0fu3i5OvP8/nd1qjwaasHcIPAcf/gBSyAAMMwBANYEAhWYQGDBhAyLihwYJiEjx8fYMxIcsGDAxVA/yYIOZIkBAaGPIK8INJlRpgrPeasaRPmx5QgJfB0abLjz50tSeIM+pFmUo0nQQIV+vRlTJUSnNq0KlXCSq09ozIFexEBAYkeNiwgOaEtn2LFpGEQsKCtXbcSjOmVlqDuhAx3+eg1Jo3u37sZBA9GoMAw4MB5FyMwfLht4sh7G/utPGHlYAV8Nz9OnOBz4c2VFWem/Pivar0aKCP2LFn2XwhnVxBwsPbuBAQbEGiIFg1BggoWkidva5z4cL7IlStfkED48OIYoiufYIH68+cKPkqfnsB58ePj###3Dj199/XE20tv6/27XO3S6z9nPCz9BP3FISDefL/Bt192/uWmAv8BFzAQAQUWWFaaBgqA11hbHWTIXWIVXifNhRlq6FqF1sm1QQYhdiAhbNEYc2KKK1pXnAIvhrjhBh0KxxiINlqQAY4UXjdcjSJyeAx2G2BYJJD7NZQkjCPKuCORKnbAIXsuKhlhBxEomAIBBzgIYXIfHfmhAAyMR2ZkHk62gJoWlNlhi33ZJZ2cQiKTJoG05Wjcm3xith9dcOK5X51tLRenoHTuud2iMnaolp3KGXrdBo7eKYF5p/mXgJcogClmcgzAR5gCKymXYqlCgmacdhp2UCqL96mq4nuDBTmgBasaCFp4sHaQHHUsGvNRiiGyep1exyIra2mS7dprrtA5++z/Z8ZKYGuGsy6GqgTIDvupRGE+6CO0x3xI5Y2mOTkBjD4ySeGU79o44mcaSEClhglgsKyJ9S5ZTGY0Bnzrj+3SiKK9Rh5zjAALCywZBk/ayCWO3hYM5Y8Dn6qxxRFsgAGoJwwgDQRtYXAAragyQOmaLKNZKGaEuUlpyiub+ad/KtPqpntypvvnzR30DBtjMhNodK6Eqrl0zU0/GjTUgG43wdN6Ra2pAhGtAAZGE5Ta8TH6wknd2IytNKaiZ+Or79oR/tcvthIcAPe7DGAs9Edwk6r3qWoTaNzY2fb9HuHh2S343Hs1VIHhYtOt+Hh551rh24vP5YvXSGzh+eeghy76GuikU9FFEainrvrqrLfu+uuwxy777LTXfkIIACH5BAkKAAAALAAAA' +
  130. 'ADcABMAAAX/ICCOZGmeaKqubOu+cCzPdG3feK7vfO//wKBwSAQIBoSkcslsOp/QqHRKrVqv2Kx2OhC0BAWHB2l4CDZo9IDjcBja7UEhTV+3DXi3PJFA8xMcbHiDBgMPG31pgHBvg4Z9iYiBjYx7kWocb26OD398mI2EhoiegJlud4UFiZ5sm6Kdn2mBr5t7pJ9rlG0cHg5gXitdaxwFGArIGgoaGwYCZ3QFDwjU1AoIzdCQzdPV1c0bZ9vS3tUJBmjQaGXl1OB0feze1+faiBvk8wjnimn55e/o4OtWjp+4NPIKogsXjaA3g/fiGZBQAcEAFgQGOChgYEEDCCBBLihwQILJkxIe/3wMKfJBSQkJYJpUyRIkgwcVUJq8QLPmTYoyY6ZcyfJmTp08iYZc8MBkhZgxk9aEcPOlzp5FmwI9KdWn1qASurJkClRoWKwhq6IUqpJBAwQEMBYroAHkhLt3+Ryzh###gAV48Wbgg+waAnoLMgTOm6DwQ8CLBzdGdvjw38V5JTg2lzhyTMeUEwBWHPgzZc4TSOM1bZia6LuqJxCmnOxv7NSsl1mGHHiw5tOuIWeAEHcFATwJME/ApgFBc3MVLEgPvE+Ddb4JokufPmFBAuvPXWu3MIF89wTOmxvOvp179evQtwf2nr6aApPyzVd3jn089e/8xdfeXe/xdZ9/d1ngHf98lbHH3V0LMrgPgsWpcFwBEFBgHmyNXWeYAgLc1UF5sG2wTHjIhNjBiIKZCN81GGyQwYq9uajeMiBOQGOLJ1KjTI40kmfBYNfc2NcGIpI4pI0vyrhjiT1WFqOOLEIZnjVOVpmajYfBiCSNLGbA5YdOkjdihSkQwIEEEWg4nQUmvYhYe+bFKaFodN5lp3rKvJYfnBKAJ+gGDMi3mmbwWYfng7IheuWihu5p32XcSWdSj+stkF95dp64jJ+RBipocHk###6PCiRQ6INookCAAwy0yd2CtNET3Yo7RvihBjFZAOaKDHT43DL4BQnsZMo8xx6uI1oQrHXXhHZrB28G62n/YSYxi+uzP2IrgbbHbiaer7hCiOxDFWhrbmGnLVuus5NFexhFuHLX6gkEECorlLpZo0CWJG4pLjIACykmBsp0eSSVeC15TDJeUhlkowlL+SWLNJpW2WEF87urXzNWSZ6JOEb7b8g1brZMjCg3ezBtWKKc4MvyEtwybPeaMAA1ECRoAQYHYLpbeYYCLfQ+mtL5c9CnfQpYpUtHOSejEgT9ogZ/GSqd0f2m+LR5WzOtHqlQX1pYwpC+WbXKqSYtpJ5Mt4a01lGzS3akF60AxkcTaLgAyRBPWCoDgHfJqwRuBuzdw/1ml3iCwTIeLUWJN0v4McMe7uasCTxseNWPSxc5RbvIgD7geZLbGrqCG3jepUmbbze63Y6fvjiOylbwOITPfIHEFsAHL/zwxBdvPBVdFKH88sw37/zz0Ecv/fTUV2/99SeEAAAh+QQJCgAAACwAAAAA3AATAAAF/yAgjmRpnmiqrmzrvnAsz3Rt33iu73zv/8CgcEgECAaEpHLJbDqf0Kh0Sq1ar9isdjoQtAQFh2cw8BQEm3T6yHEYHHD4oKCuD9qGvNsxT6QTgAkcHHmFeX11fm17hXwPG35qgnhxbwMPkXaLhgZ9gWp3bpyegX4DcG+inY+Qn6eclpiZkHh6epetgLSUcBxlD2csXXdvBQrHGgoaGhsGaIkFDwjTCArTzX+QadHU3c1ofpHc3dcGG89/4+TYktvS1NYI7OHu3fEJ5tpqBu/k+HX7+nXDB06SuoHm0KXhR65cQT8P3FRAMIAFgVMPwDCAwLHjggIHJIgceeFBg44eC/+ITCCBZYKSJ1FCWPBgpE2YMmc+qNCypwScMmnaXAkUJYOaFVyKLOqx5tCXJnMelcBzJNSYKIX2ZPkzqsyjPLku9Zr1QciVErYxaICAgEUOBRJIgzChbt0MLOPFwyBggV27eCUcmxZvg9+/dfPGo5bg8N/Ag61ZM4w4seDF1fpWhizZmoa+GSortgcaMWd/fkP/HY0MgWbTipVV++wY8GhvqSG4XUEgoYTKE+Qh0OCvggULiBckWEZ4Ggbjx5HXVc58IPQJ0idQJ66XanTpFraTe348+XLizRNcz658eHMN3rNPT+C+G/nodqk3t6a+fN3j+u0Xn3nVTQPfdRPspkL/b+dEIN8EeMm2GAYbTNABdrbJ1hyFFv5lQYTodSZABhc+loCEyhxTYYkZopdMMiNeiBxyIFajV4wYHpfBBspUl8yKHu6ooV5APsZjQxyyeNeJ3N1IYod38cgdPBUid6GCKfRWgAYU4IccSyHew8B3doGJHmMLkGkZcynKk2Z50Ym0zJzLbDCmfBbI6eIyCdyJmJmoqZmnBAXy9+Z/yOlZDZpwYihnj7IZpuYEevrYJ5mJEuqiof4l+NYDEXQpXQcMnNjZNDx1oGqJ4S2nF3EsqWrhqqVWl6JIslpAK5MaIqDeqjJq56qN1aTaQaPbHTPYr8Be6Gsyyh6Da7OkmmqP/7GyztdrNVQBm5+pgw3X7aoYKhfZosb6hyUKBHCgQKij1rghkOAJuZg1SeYIIY+nIpDvf/sqm4yNG5CY64f87qdAwSXKGqFkhPH1ZHb2EgYtw3bpKGVkPz5pJAav+gukjB1UHE/HLNJobWc' +
  131. 'SX8jiuicMMBFd2OmKwQFs2tjXpDfnPE1j30V3c7iRHlrzBD2HONzODyZtsQJMI4r0AUNaE3XNHQw95c9GC001MpIxDacFQ+ulTNTZlU3O1eWVHa6vb/pnQUUrgHHSBKIuwG+bCPyEqbAg25gMVV1iOB/IGh5YOKLKIQ6xBAcUHmzjIcIqgajZ+Ro42DcvXl7j0U4WOUd+2IGu7DWjI1pt4DYq8BPm0entuGSQY/4tBi9Ss0HqfwngBQtHbCH88MQXb/zxyFfRRRHMN+/889BHL/301Fdv/fXYZ39CCAAh+QQJCgAAACwAAAAA3AATAAAF/yAgjmRpnmiqrmzrvnAsz3Rt33iu73zv/8CgcEgECAaEpHLJbDqf0Kh0Sq1ar9isdjoQtAQFh2fAKXsKm7R6Q+Y43vABep0mGwwOPH7w2CT+gHZ3d3lyagl+CQNvg4yGh36LcHoGfHR/ZYOElQ9/a4ocmoRygIiRk5p8pYmZjXePaYBujHoOqp5qZHBlHAUFXitddg8PBg8KGsgayxvGkAkFDwgICtPTzX2mftHW3QnOpojG3dbYkNjk1waxsdDS1N7ga9zw1t/aifTk35fu6Qj3numL14fOuHTNECHqU4DDgQEsCCwidiHBAwYQMmpcUOCAhI8gJVzUuLGThAQnP/9abEAyI4MCIVOKZNnyJUqUJxNcGNlywYOQgHZirGkSJ8gHNEky+AkS58qWEJYC/bMzacmbQHkqNdlUJ1KoSz2i9COhmQYCEXtVrCBgwYS3cCf8qTcNQ9u4cFFOq2bPLV65Cf7dxZthbjW+CgbjnWtNgWPFcAsHdoxgWWK/iyV045sAc2S96SDn1exYw17REwpLQEYt2eW/qtPZRQAB7QoC61RW+GsBwYZ/CXb/XRCYLsAKFizEtUAc+G7lcZsjroscOvTmsoUvx15PwccJ0N8yL17N9PG/E7jv9S4hOV7pdIPDdZ+ePDzv2qMXn2b5+wTbKuAWnF3oZbABZY' +
  132. '0lVmD/ApQd9thybxno2GGuCVDggaUpoyBsB1bGGgIYbJCBcuFJiOAyGohIInQSmmdeiBnMF2GHfNUlIoc1rncjYRjW6NgGf3VQGILWwNjBfxEZcAFbC7gHXQcfUYOYdwzQNxo5yUhQZXhvRYlMeVSuSOJHKJa5AQMQThBlZWZ6Bp4Fa1qzTAJbijcBlJrtxeaZ4lnnpZwpukWieGQmYx5ATXIplwTL8DdNZ07CtWYybNIJF4Ap4NZHe0920AEDk035kafieQrqXofK5ympn5JHKYjPrfoWcR8WWQGp4Ul32KPVgXdnqxM6OKqspjIYrGPDrlrsZtRIcOuR86nHFwbPvmes/6PH4frrqbvySh+mKGhaAARPzjjdhCramdoGGOhp44i+zogBkSDuWC5KlE4r4pHJkarXrj++Raq5iLmWLlxHBteavjG+6amJrUkJJI4Ro5sBv9AaOK+jAau77sbH7nspCwNIYIACffL7J4JtWQnen421nNzMcB6AqpRa9klonmBSiR4GNi+cJZpvwgX0ejj71W9yR+eIgaVvQgf0l/A8nWjUFhwtZYWC4hVnkZ3p/PJqNQ5NnwUQrQCGBBBMQIGTtL7abK+5JjAv1fi9bS0GLlJHgdjEgYzzARTwC1fgEWdJuKKBZzj331Y23qB3i9v5aY/rSUC4w7PaLeWXmr9NszMFoN79eeiM232o33EJAIzaSGwh++y012777bhT0UURvPfu++/ABy/88MQXb/zxyCd/QggAIfkECQoAAAAsAAAAANwAEwAABf8gII5kaZ5oqq5s675wLM90bd94ru987//AoHBIBAgGhKRyyWw6n9CodEqtWq/YrHY6ELQEBY5nwCk7xIWNer0hO95wziC9Ttg5b4ND/+Y87IBqZAaEe29zGwmJigmDfHoGiImTjXiQhJEPdYyWhXwDmpuVmHwOoHZqjI6kZ3+MqhyemJKAdo6Ge3OKbEd4ZRwFBV4rc4MPrgYPChrMzAgbyZSJBcoI1tfQoYsJydfe2amT3d7W0OGp1OTl0YtqyQrq0Lt11PDk3KGoG+nxBpvTD9QhwCctm0BzbOyMIwdOUwEDEgawIOCB2oMLgB4wgMCx44IHBySIHClBY0ePfyT/JCB5weRJCAwejFw58kGDlzBTqqTZcuPLmCIBiWx58+VHmiRLFj0JVCVLl0xl7qSZwCbOo0lFWv0pdefQrVFDJtr5gMBEYBgxqBWwYILbtxPsqMPAFu7blfa81bUbN4HAvXAzyLWnoDBguHIRFF6m4LBbwQngMYPXuC3fldbyPrMcGLM3w5wRS1iWWUNlvnElKDZtz/EEwaqvYahQoexEfyILi4RrYYKFZwJ3810QWZ2ECrx9Ew+O3K6F5Yq9zXbb+y30a7olJJ+wnLC16W97Py+uwdtx1NcLWzs/3G9e07stVPc9kHJ0BcLtQp+c3ewKAgYkUAFpCaAmmHqKLSYA/18WHEiZPRhsQF1nlLFWmIR8ZbDBYs0YZuCGpGXWmG92aWiPMwhEOOEEHXRwIALlwXjhio+BeE15IzpnInaLbZBBhhti9x2GbnVQo2Y9ZuCfCgBeMCB+DJDIolt4iVhOaNSJdCOBUfIlkmkyMpPAAvKJ59aXzTQzJo0WoJnmQF36Jp6W1qC4gWW9GZladCiyJd+KnsHImgRRVjfnaDEKuiZvbcYWo5htzefbl5LFWNeSKQAo1QXasdhiiwwUl2B21H3aQaghXnPcp1NagCqYslXAqnV+zYWcpNwVp9l5eepJnHqL4SdBi56CGlmw2Zn6aaiZjZqfb8Y2m+Cz1O0n3f+tnvrGbF6kToApCgAWoNWPeh754JA0vmajiAr4iOuOW7abQXVGNriBWoRdOK8FxNqLwX3oluubhv8yluRbegqGb536yk###oXhyJqPQJIGbLvQhkcwjKs1zBvBwSZIsbcsDCCBAAf4ya+UEhyQoIiEJtfoZ7oxUOafE2BwgMWMqUydfC1LVtiArk0QtGkWEopzlqM9aJrKHfw5c6wKjFkmXDrbhwFockodtMGFLWpXy9JdiXN1ZDNszV4WSLQCGBKoQYHUyonqrHa4ErewAgMmcAAF7f2baIoVzC2p3gUvJtLcvIWqloy6/R04mIpLwDhciI8qLOB5yud44pHPLbA83hFDWPjNbuk9KnySN57Av+TMBvgEAgzzNhJb5K777rz37vvvVHRRxPDEF2/88cgnr/zyzDfv/PPQnxACACH5BAkKAAAALAAAAADcABMAAAX/ICCOZGmeaKqubOu+cCzPdG3feK7vfO//wKBwSAQIBoSkcslsOp/QqHRKrVqv2Kx2OhC0BIUCwcMpO84OT2HDbm8GHLQjnn6wE3g83SA3DB55G3llfHxnfnZ4gglvew6Gf4ySgmYGlpCJknochWiId3kJcZZyDn93i6KPl4eniopwq6SIoZKxhpenbhtHZRxhXisDopwPgHkGDxrLGgjLG8mC0gkFDwjX2AgJ0bXJ2djbgNJsAtbfCNB2oOnn6MmKbeXt226K1fMGi6j359D69ua+QZskjd+3cOvY9XNgp4ABCQNYEDBl7EIeCQkeMIDAseOCBwckiBSZ4ILGjh4B/40kaXIjSggMHmBcifHky5gYE6zM2OAlzGM6Z5rs+fIjTZ0tfcYMSlLCUJ8fL47kCVXmTjwPiKJkUCDnyqc3CxzQmYeAxAEGLGJYiwCDgAUT4sqdgOebArdw507IUNfuW71xdZ7DC5iuhGsKErf9CxhPYgUaEhPWyzfBMgUIJDPW6zhb5M1y+R5GjFkBaLmCM0dOfHqvztXYJnMejaFCBQlmVxAYsEGkYnQV4lqYMNyCtnYSggNekAC58uJxmTufW5w55mwKkg+nLp105uTC53a/nhg88fMTmDfDVl65Xum/IZt/3/zaag3a5W63nll1dvfiWbaaZLmpQIABCVQA2f9lAhTG112PQWYadXE9+FtmEwKWwQYQJrZagxomsOCAGVImInsSbpCBhhwug6KKcXXQQYUcYuDMggrASFmNzjjzzIrh7cUhhhHqONeGpSEW2QYxHsmjhxpgUGAKB16g4IIbMNCkXMlhaJ8GWVJo2I3NyKclYF1GxgyYDEAnXHJrMpNAm/rFBSczPiYAlwXF8ZnmesvoOdyMbx7m4o0S5LWdn4bex2Z4xYmEzaEb5EUcnxbA+WWglqIn6aHPTInCgVbdlZyMqMrIQHMRSiaBBakS1903p04w434n0loBoQFOt1yu2YAnY68RXiNsqh2s2qqxuyKb7Imtmgcrqsp6h8D/fMSpapldx55nwayK/SfqCQd2hcFdAgDp5GMvqhvakF4mZuS710WGIYy30khekRkMu92GNu6bo7r/ttjqwLaua5+HOdrKq5Cl3dcwi+xKiLBwwwom4b0E6xvuYyqOa8IAEghwQAV45VvovpkxBl2mo0W7AKbCZXoAhgMmWnOkEqx2JX5nUufbgJHpXCfMOGu2QAd8eitpW1eaNrNeMGN27mNz0swziYnpSbXN19gYtstzfXrdYjNHtAIYGFVwwAEvR1dfxdjKxVzAP0twAAW/ir2w3nzTd3W4yQWO3t0DfleB4XYnEHCEhffdKgaA29p0eo4fHLng9qoG+OVyXz0gMeWGY7qq3xhiRIEAwayNxBawxy777LTXbjsVXRS' +
  133. 'h++689+7778AHL/zwxBdv/PEnhAAAIfkECQoAAAAsAAAAANwAEwAABf8gII5kaZ5oqq5s675wLM90bd94ru987//AoHBIBAgGhKRyyWw6n9CodEqtWq/YrHY6ELQEhYLD4BlwHGg0ubBpuzdm9Dk9eCTu+MTZkDb4PXYbeIIcHHxqf4F3gnqGY2kOdQmCjHCGfpCSjHhmh2N+knmEkJmKg3uHfgaaeY2qn6t2i4t7sKAPbwIJD2VhXisDCQZgDrKDBQ8aGgjKyhvDlJMJyAjV1gjCunkP1NfVwpRtk93e2ZVt5NfCk27jD97f0LPP7/Dr4pTp1veLgvrx7AL+Q/BM25uBegoYkDCABYFhEobhkUBRwoMGEDJqXPDgQMUEFC9c1LjxQUUJICX/iMRIEgIDkycrjmzJMSXFlDNJvkwJsmdOjQwKfDz5M+PLoSGLQqgZU6XSoB/voHxawGbFlS2XGktAwKEADB0xiEWAodqGBRPSqp1wx5qCamDRrp2Qoa3bagLkzrULF4GCvHPTglRAmKxZvWsHayBcliDitHUlvGWM97F###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###5oCiPqAEkz8tXuLkPeDL3Uhs4fvvwAcv/PDEU9FFEcgnr/zyzDfv/PPQRy/99NRXf0IIACH5BAkKAAAALAAAAADcABMAAAX/ICCOZGmeaKqubOu+cCzPdG3feK7vfO//wKBwSAQIBoSkcslsOp/QqHRKrVqv2Kx2OhC0BIWCw/AoDziOtCHt8BQ28PjmzK57Hom8fo42+P8DeAkbeYQcfX9+gYOFg4d1bIGEjQmPbICClI9/YwaLjHAJdJeKmZOViGtpn3qOqZineoeJgG8CeWUbBV4rAwkGAhIVGL97hGACGsrKCAgbBoTRhLvN1c3PepnU1s2/oZO6AtzdBoPf4eMI3tIJyOnF0YwFD+nY8e3z7+Xfefnj9uz8cVsXCh89axgk7BrAggAwBQsYIChwQILFixIeNIDAseOCBwcSXMy2sSPHjxJE/6a0eEGjSY4MQGK86PIlypUJEmYsaTKmyJ8JW/Ls6HMkzaEn8YwMWtPkx4pGd76E4DMPRqFTY860OGhogwYagBFoKEABA46DEGBAoEBB0AUT4sqdIFKBNbcC4M6dkEEk22oYFOTdG9fvWrtsBxM23MytYL17666t9phwXwlum2lIDHmuSA2IGyuOLOHv38qLMbdFjHruZbWgRXeOe1nC2BUEDiyAMMHZuwoTLAQX3nvDOAUW5Vogru434d4JnAsnPmFB9NBshQXfa9104+Rxl8e13rZxN+CEydtVsFkd+vDjE7C/q52wOvb4s7+faz025frbxefWbSoQIAEDEUCwgf9j7bUlwHN9ZVaegxDK1xYzFMJH24L5saXABhlYxiEzHoKoIV8LYqAMaw9aZqFmJUK4YHuNfRjiXhmk+NcyJgaIolvM8BhiBx3IleN8lH1IWAcRgkZgCgYiaBGJojGgHHFTgtagAFYSZhF7/qnTpY+faVlNAnqJN0EHWa6ozAZjBtgmmBokwMB01LW5jAZwbqfmlNips4B4eOqJgDJ2+imXRZpthuigeC6XZTWIxilXmRo8iYKBCwiWmWkJVEAkfB0w8KI1IvlIpKnOkVpqdB5+h96o8d3lFnijrgprjbfGRSt0lH0nAZG5vsprWxYRW6Suq4UWqrLEsspWg8Io6yv/q6EhK0Fw0GLbjKYn5CZYBYht1laPrnEY67kyrhYbuyceiR28Pso7bYwiXjihjWsWuWF5p/H765HmNoiur3RJsGKNG/jq748XMrwmjhwCfO6QD9v7LQsDxPTAMKsFpthyJCdkmgYiw0VdXF/Om9dyv7YMWGXTLYpZg5wNR11C78oW3p8HSGgul4qyrJppgllJHJZHn0Y0yUwDXCXUNquFZNLKyYXBAVZvxtAKYIQEsmPgDacr0tltO1y/DMwYpkgUpJfTasLGzd3cdCN3gN3UWRcY3epIEPevfq+3njBxq/kqBoGBduvea8f393zICS63ivRBTqgFpgaWZEIUULdcK+frIfAAL2AjscXqrLfu+uuwx05FF0XUbvvtuOeu++689+7778AHL/wJIQAAOwAAAAAAAAAAAA=="/><br/>Loading...');
  134. divDic.appendChild(divR###lt);
  135. // Options link
  136. var optionLink = createElement('a', {
  137. id : 'optionsLink',
  138. href : HREF_NO,
  139. style : 'opacity:0.2; position:absolute; bottom:3px; right:13px; font-size:18px; text-decoration:none!important;background:#528DDF;padding:1px;color:#fff;border-radius:6px 6px 6px 6px;border:2px solid #EEEEEE;font-weight:bold;width:20px;text-align:center;display:block;'
  140. }, {
  141. type: 'click',
  142. handler: openCloseOptions,
  143. capture: false
  144. }, '+');
  145. divDic.appendChild(optionLink);
  146. optionLink.addEventListener('mouseover', function (e) {
  147. e.target.style.opacity = 1.0
  148. });
  149. optionLink.addEventListener('mouseout', function (e) {
  150. e.target.style.opacity = 0.2
  151. });
  152. // Send the Google Translate request
  153. if ((txtSel + " ").search(/^\s*https?:\/\//) > -1) {
  154. divR###lt.innerHTML = '<a href="' + txtSel + '" target="_blank" >' + txtSel + '</a>';
  155. } else if ((txtSel + " ").search(/^\s*\S+(\.\S+)+/) > -1) // site.dom
  156. {
  157. divR###lt.innerHTML = '<a style="color:#888;" href="http://' + txtSel + '" target="_blank" >' + txtSel + '</a>';
  158. } else {
  159. var sl, tl;
  160. sl = GM_getValue('from') ? GM_getValue('from') : "auto";
  161. tl = GM_getValue('to') ? GM_getValue('to') : "auto";
  162. Request(txtSel, sl, tl, extractR###lt);
  163. if (GM_getValue('to2', 'Disabled') != 'Disabled') {
  164. sl = GM_getValue('from') ? GM_getValue('from') : "auto";
  165. tl = GM_getValue('to2') ? GM_getValue('to2') : "auto";
  166. Request(txtSel, sl, tl, extractR###lt2);
  167. } else {
  168. translation2Element.innerHTML = '';
  169. }
  170. if (GM_getValue('to3', 'Disabled') != 'Disabled') {
  171. sl = GM_getValue('from') ? GM_getValue('from') : "auto";
  172. tl = GM_getValue('to3') ? GM_getValue('to3') : "auto";
  173. Request(txtSel, sl, tl, extractR###lt3);
  174. } else {
  175. translation3Element.innerHTML = '';
  176. }
  177. }
  178. }
  179. // Lanched when we select an other language in the setup menu
  180. function quickLookup() {
  181. getId('divDic').style.fontSize = getId('optFontSize').value;
  182. getId('divDic').style.color = getId('optTextColor').value;
  183. getId('divR###lt').innerHTML = 'Loading...';
  184. if((context == null) && (GM_getValue('tts', false) == true) ) context = new AudioContext();
  185. var sl,tl;
  186. sl = getId('optSelLangFrom').value;
  187. tl = getId('optSelLangTo').value;
  188. Request(txtSel, sl, tl, extractR###lt);
  189. if (getId('optSelLangTo2').value != 'Disabled') {
  190. var sl,tl;
  191. sl = getId('optSelLangFrom').value;
  192. tl = getId('optSelLangTo2').value;
  193. Request(txtSel, sl, tl, extractR###lt2);
  194. } else {
  195. translation2Element.innerHTML = '';
  196. }
  197. if (getId('optSelLangTo3').value != 'Disabled') {
  198. var sl,tl;
  199. sl = getId('optSelLangFrom').value;
  200. tl = getId('optSelLangTo3').value;
  201. Request(txtSel, sl, tl, extractR###lt3);
  202. } else {
  203. translation3Element.innerHTML = '';
  204. }
  205. }
  206. function init_google_value_tk() {
  207. var url = "https://" + googleDomain + "/translate_a/element.js";
  208. GM_xmlhttpRequest({
  209. method: "GET",
  210. url: url,
  211. onreadystatechange: function(resp) {
  212. if (resp.readyState == 4) {
  213. clearTimeout(setTimeout( function(){ this.abort(); }, 2000));
  214. if (resp.status == 200) {
  215. init_google_value_tk_parse(resp.responseText);
  216. }
  217. }
  218. }
  219. });
  220. }
  221. function init_google_value_tk_parse(responseText) {
  222. var res = /c\._ctkk='(.+?)'/i.exec(responseText);
  223. if (res != null) {
  224. GM_setValue('google_value_tk', res[1]);
  225. };
  226. }
  227. // return token for the new API
  228. function googleTK(text) {
  229. // view-source:https://translate.google.com/translate/releases/twsfe_w_20160620_RC00/r/js/desktop_module_main.js && TKK from HTML
  230. var uM = GM_getValue('google_value_tk');
  231. if (uM == 'undefined' || uM == null) {
  232. init_google_value_tk();
  233. uM = "427110.1469889687";
  234. } else if (Number(uM.split('.')[0]) !== Math.floor(Date.now() / 3600000)) {
  235. init_google_value_tk();
  236. };
  237. var cb="&";
  238. var k="";
  239. var Gf="=";
  240. var Vb="+-a^+6";
  241. var t="a";
  242. var Yb="+";
  243. var Zb="+-3^+b+-f";
  244. var jd=".";
  245. var sM=function(a){return function(){return a}}
  246. var tM=function(a,b){for(var c=0;c<b.length-2;c+=3){var d=b.charAt(c+2),d=d>=t?d.charCodeAt(0)-87:Number(d),d=b.charAt(c+1)==Yb?a>>>d:a<<d;a=b.charAt(c)==Yb?a+d&4294967295:a^d}return a};
  247. var vM=function(a){
  248. var b;
  249. if(null!==uM) {
  250. b=uM;
  251. }else{
  252. b=sM(String.fromCharCode(84));var c=sM(String.fromCharCode(75));b=[b(),b()];
  253. b[1]=c();
  254. b=(uM=window[b.join(c())]||k)||k
  255. }
  256. var d=sM(String.fromCharCode(116)),c=sM(String.fromCharCode(107)),d=[d(),d()];
  257. d[1]=c();
  258. c=cb+d.join(k)+Gf;
  259. d=b.split(jd);
  260. b=Number(d[0])||0;
  261. for(var e=[],f=0,g=0;g<a.length;g++){
  262. var m=a.charCodeAt(g);
  263. 128>m?e[f++]=m:(2048>m?e[f++]=m>>6|192:(55296==(m&64512)&&g+1<a.length&&56320==(a.charCodeAt(g+1)&64512)?(m=65536+((m&1023)<<10)+(a.charCodeAt(++g)&1023),e[f++]=m>>18|240,e[f++]=m>>12&63|128):e[f++]=m>>12|224,e[f++]=m>>6&63|128),e[f++]=m&63|128)
  264. }
  265. a=b||0;
  266. for(f=0;f<e.length;f++) { a+=e[f],a=tM(a,Vb)};
  267. a=tM(a,Zb);
  268. a^=Number(d[1])||0;
  269. 0>a&&(a=(a&2147483647)+2147483648);
  270. a%=1E6;
  271. return a.toString()+jd+(a^b);
  272. };
  273. return vM(text);
  274. }
  275. // Google Translate Request
  276. function Request(txt, sl, tl, parse) {
  277. var tk=googleTK(txt);
  278. var Url = dictURL +
  279. "&hl=auto" +
  280. "&sl=" + sl + "&tl=" + tl +
  281. "&dt=bd&dt=ex&dt=ld&dt=md&dt=qca&dt=rw&dt=rm&dt=ss&dt=t&dt=at&ie=UTF-8&oe=UTF-8&otf=2&trs=1&inputm=1&ssel=0&tsel=0&source=btn&kc=3"+
  282. "&tk=" + tk +
  283. "&q="+ encodeURI(txt);
  284. var method='POST';
  285. var Data='';
  286. var Hdr= {
  287. "User-Agent": UA,
  288. "Accept": "text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8",
  289. "Accept-Encoding": "gzip, deflate"
  290. }
  291. var Q=Url.split('&q=');
  292. Url=Q[0];
  293. Data='&q='+Q[1];
  294. Hdr["Content-Length"]=Data.length+'';
  295. Hdr["Content-Type"]="application/x-www-form-urlencoded; charset=UTF-8";
  296. GM_xmlhttpRequest({
  297. method: method,
  298. url: Url,
  299. data: Data,
  300. headers: Hdr,
  301. onload: function(resp) {
  302. try{
  303. parse(resp.responseText)
  304. }catch(e){
  305. GM_log(e);
  306. }
  307. }
  308. });
  309. }
  310. function extractR###lt(gTradStringArray) {
  311. var arr = parseTranslationResponse(gTradStringArray);
  312. if (!arr) {
  313. // Handle error case
  314. getId('divR###lt').innerHTML = 'Translation error occurred';
  315. return;
  316. }
  317. /*
  318. Content of the gTrad array :
  319. 0 / 0:Translation 1:Source text
  320. 1 / i:Grammar / 0:Types (word, verb, ...) 1: Other translations
  321. 5 / Array of other translations
  322. */
  323. var translation = '';
  324. // 0 - Full translation
  325. translation += '<small><a style="color:#1a0dab;" target="_blank" href="https://' + googleDomain + '/#' + GM_getValue('from', 'auto') + '/' + GM_getValue('to', 'auto') + '/' + txtSel + '">[' + arr[2] + '] ';
  326. for (var i = 0; i < arr[0].length; i++) { if (typeof arr[0][i][1] != 'undefined' && arr[0][i][1] != null) translation += arr[0][i][1]; }
  327. translation += '</a> <span id="texttospeechbuttonfrom"></span></small><br/>';
  328. translation += '[' + GM_getValue('to', 'auto') + '] ';
  329. for (var i = 0; i < arr[0].length; i++) { if (typeof arr[0][i][0] != 'undefined' && arr[0][i][0] != null) translation += arr[0][i][0]; }
  330. translation += ' <span id="texttospeechbuttonto"></span><br/>';
  331. translation += '<span id="translation2Element"></span>';
  332. translation += '<span id="translation3Element"></span>';
  333. translation += '<a id="toggleShowDetails" style="color:#000;' + (!GM_getValue('details', 'false') ? 'display:none;"' : '"') + '>Show details</a>';
  334. translation += '<span id="divDetails" ' + (GM_getValue('details', 'false') ? 'style="display:none;"' : '') + '><a style="color:#000;" id="toggleHideDetails">Hide details</a><br/>';
  335. // 1 - Grammar
  336. if (typeof arr[1] != 'undefined' && arr[1] != null ||
  337. typeof arr[5] != 'undefined' && arr[5] != null ||
  338. typeof arr[14] != 'undefined' && arr[14] != null) {
  339. translation += '<strong>Translations</strong><br/>';
  340. }
  341. if (typeof arr[1] != 'undefined' && arr[1] != null) {
  342. for (var i = 0; i < arr[1].length; i++) {
  343. translation += arr[1][i][0] + ': ';
  344. translation += arr[1][i][1].join(', ');
  345. translation += '<br/>';
  346. }
  347. }
  348. // 5 - Alternatives
  349. if (GM_getValue('alternatives', 'true')) {
  350. if (typeof arr[5] != 'undefined' && arr[5] != null) {
  351. for (var i = 0; i < arr[5].length; i++) {
  352. if (typeof arr[5][i][2] != 'undefined' && arr[5][i][2] != null) { // 5/i/2 array of alternatives, 5/i/0 the part of the text we are studying
  353. translation += '<i>alternatives: </i>';
  354. for (var j = 0; j < arr[5][i][2].length; j++) {
  355. translation += '<i>' + ((j == 0) ? '' : ', ') + arr[5][i][2][j][0] + '</i>';
  356. }
  357. translation += '<br/>';
  358. }
  359. }
  360. }
  361. }
  362. // 14 - See also
  363. if (typeof arr[14] != 'undefined' && arr[14] != null) {
  364. // for (var i = 0; i < arr[14].length; i++) {
  365. translation += '<i>See also: </i>';
  366. translation += '<i>' + arr[14][0].join(', ') + '</i>';
  367. translation += '<br/>';
  368. // }
  369. }
  370. // if ((typeof arr[1] != 'undefined' && arr[1] != null ||
  371. // typeof arr[5] != 'undefined' && arr[5] != null ||
  372. // typeof arr[14] != 'undefined' && arr[14] != null) &&
  373. // (typeof arr[12] != 'undefined' && arr[12] != null)) {
  374. // translation += '<br/>';
  375. // }
  376. // 12 and 11 - Definitions and Synonyms
  377. if (typeof arr[12] != 'undefined' && arr[12] != null) {
  378. translation += '<strong>Definitions</strong><br/>';
  379. for (var i = 0; i < arr[12].length; i++) {
  380. if (typeof arr[12][i][1] != 'undefined' && arr[12][i][1] != null) { // 11/i/1 array of alternatives, 11/i/0 the part of the text we are studying
  381. for (var j = 0; j < arr[12][i][1].length; j++) {
  382. translation += '' + arr[12][i][0] + ': ';
  383. translation += arr[12][i][1][j][0];
  384. translation += '<br/>';
  385. if (GM_getValue('synonyms', 'true')) {
  386. if (typeof arr[11] != 'undefined' && arr[11] != null) {
  387. if (typeof arr[11][i] != 'undefined' && [11][i] != null) {
  388. for (var k = 0; k < arr[11][i][1].length; k++) {
  389. if (arr[12][i][1][j][1] == arr[11][i][1][k][1]) {
  390. translation += '<i>synonyms:</i> ';
  391. translation += '<i>' + arr[11][i][1][k][0].join(', ') + '</i>';
  392. translation += '<br/>';
  393. break;
  394. }
  395. }
  396. }
  397. }
  398. }
  399. }
  400. }
  401. }
  402. }
  403. translation += '</span>'; // Detail end
  404. getId('divR###lt').innerHTML = '<p style="margin:0px;padding:0px;line-height:150%;text-align:left;">' + translation + '</p>';
  405. getId('translation2Element').appendChild(translation2Element); // Optional second translation
  406. getId('translation3Element').appendChild(translation3Element); // Optional third translation
  407. getId('toggleShowDetails').addEventListener('click', function () {
  408. getId('toggleShowDetails').style.display = 'none';
  409. getId('divDetails').style.display = 'block';
  410. }, false);
  411. getId('toggleHideDetails').addEventListener('click', function () {
  412. getId('toggleShowDetails').style.display = 'inline';
  413. getId('divDetails').style.display = 'none';
  414. }, false);
  415. // Create the Text to speech
  416. var fromText = '';
  417. var toText = '';
  418. for (var i = 0; i < arr[0].length; i++) { if (typeof arr[0][i][1] != 'undefined' && arr[0][i][1] != null) fromText += arr[0][i][1]; }
  419. for (var i = 0; i < arr[0].length; i++) { if (typeof arr[0][i][0] != 'undefined' && arr[0][i][0] != null) toText += arr[0][i][0]; }
  420. addTextTospeechLink(getId('texttospeechbuttonfrom'), arr[2], fromText); // arr[2] contains the detected input language
  421. addTextTospeechLink(getId('texttospeechbuttonto'), GM_getValue('to', 'auto') == 'auto' ? 'en' : GM_getValue('to', 'auto'), toText); // I cannot find a way to get the detected destination language, so if the requested destination is 'auto', I use the english Text to speech language
  422. }
  423. function extractR###lt2(gTradStringArray) {
  424. var arr = parseTranslationResponse(gTradStringArray);
  425. if (!arr) {
  426. // Handle error case
  427. getId('divR###lt').innerHTML = 'Translation error occurred';
  428. return;
  429. }
  430. var translation = '';
  431. translation += '#[' + GM_getValue('to2', 'auto') + '] ';
  432. for (var i = 0; i < arr[0].length; i++) { if (typeof arr[0][i][0] != 'undefined' && arr[0][i][0] != null) translation += arr[0][i][0]; }
  433. translation += ' <span id="texttospeechbuttonto2"></span><br/>';
  434. translation2Element.innerHTML = translation;
  435. var toText2 = '';
  436. for (var i = 0; i < arr[0].length; i++) { if (typeof arr[0][i][0] != 'undefined' && arr[0][i][0] != null) toText2 += arr[0][i][0]; }
  437. addTextTospeechLink(getId('texttospeechbuttonto2'), GM_getValue('to2', 'auto') == 'auto' ? 'en' : GM_getValue('to2', 'auto'), toText2);
  438. }
  439. function extractR###lt3(gTradStringArray) {
  440. var arr = parseTranslationResponse(gTradStringArray);
  441. if (!arr) {
  442. // Handle error case
  443. getId('divR###lt').innerHTML = 'Translation error occurred';
  444. return;
  445. }
  446. var translation = '';
  447. translation += '#[' + GM_getValue('to3', 'auto') + '] ';
  448. for (var i = 0; i < arr[0].length; i++) { if (typeof arr[0][i][0] != 'undefined' && arr[0][i][0] != null) translation += arr[0][i][0]; }
  449. translation += ' <span id="texttospeechbuttonto3"></span><br/>';
  450. translation3Element.innerHTML = translation;
  451. var toText3 = '';
  452. for (var i = 0; i < arr[0].length; i++) { if (typeof arr[0][i][0] != 'undefined' && arr[0][i][0] != null) toText3 += arr[0][i][0]; }
  453. addTextTospeechLink(getId('texttospeechbuttonto3'), GM_getValue('to3', 'auto') == 'auto' ? 'en' : GM_getValue('to3', 'auto'), toText3);
  454. }
  455. function parseTranslationResponse(gTradStringArray) {
  456. // If it's already an array, return it
  457. if (Array.isArray(gTradStringArray)) {
  458. return gTradStringArray;
  459. }
  460. // Remove any leading/trailing whitespace
  461. let str = gTradStringArray.trim();
  462. // Handle common escaping issues
  463. str = str
  464. .replace(/\\x/g, '\\u00') // Convert \x escapes to \u escapes
  465. .replace(/\r?\n/g, '\\n') // Handle actual newlines
  466. .replace(/\\/g, '\\\\') // Escape backslashes
  467. .replace(/\\\\"/g, '\\"') // Fix double-escaped quotes
  468. .replace(/\\\\n/g, '\\n') // Fix double-escaped newlines
  469. .replace(/\\\\/g, '\\'); // Fix over-escaped backslashes
  470. try {
  471. // Try parsing directly
  472. return JSON.parse(str);
  473. } catch (e1) {
  474. try {
  475. // If the string isn't wrapped in brackets, wrap it
  476. if (!str.startsWith('[')) {
  477. str = '[' + str + ']';
  478. }
  479. return JSON.parse(str);
  480. } catch (e2) {
  481. // If both attempts fail, log the error and the string for debugging
  482. console.error('Failed to parse translation response');
  483. console.log('Original string:', gTradStringArray);
  484. console.log('Processed string:', str);
  485. console.error('Parse error:', e2);
  486. return null;
  487. }
  488. }
  489. }
  490. function addTextTospeechLink(element, lang, text) {
  491. if (GM_getValue('tts', false) == false) return;
  492. var img = document.createElement('img');
  493. img.setAttribute('src', "");
  494. img.setAttribute('width', '16');
  495. img.setAttribute('height', '16');
  496. img.setAttribute('align', "top");
  497. element.appendChild(img);
  498. // var context = new AudioContext();
  499. element.addEventListener('click', function() { playTTS(lang, text, context) }, false);
  500. }
  501. function playTTS(lang, text, context) {
  502. text = text.replace(/[«»'"]/g, ' ');
  503. tk = googleTK(text);
  504. Url = ttsURL + "&ie=UTF-8&total=1&idx=0" +
  505. "&tl=" + lang +
  506. "&q=" + text +
  507. "&textlen=" + text.length +
  508. "&tk=" + tk;
  509. window.AudioContext = window.AudioContext || window.webkitAudioContext;
  510. //var context = new AudioContext();
  511. var source = context.createBufferSource();
  512. var soundRequest = GM_xmlhttpRequest({
  513. method: "GET",
  514. url: Url,
  515. responseType: 'arraybuffer',
  516. onload: function(response) {
  517. try {
  518. context.decodeAudioData(response.response, function(buffer) {
  519. source.buffer = buffer;
  520. source.connect(context.destination);
  521. source.start(0);
  522. });
  523. } catch(e) {
  524. GM_log(e);
  525. }
  526. }
  527. });
  528. }
  529. function getSelection() {
  530. var text = null;
  531. //get selected text
  532. if (window.getSelection && !window.opera) // window.getSelection() bugs with Opera 12.16 and ViolentMonkey
  533. {
  534. if (document.activeElement &&
  535. (document.activeElement.tagName.toLowerCase() == "textarea" ||
  536. document.activeElement.tagName.toLowerCase() == "input")) {
  537. text = document.activeElement.value;
  538. text = text.substring (document.activeElement.selectionStart, document.activeElement.selectionEnd);
  539. } else {
  540. text = window.getSelection().toString();
  541. }
  542. } else if (document.getSelection) {
  543. text = document.getSelection().toString();
  544. } else if (document.selection) {
  545. text = document.selection.createRange().text;
  546. }
  547. // text = text.replace(/[«»'"]/g, ' ');
  548. text = text.replace(/&/g,'\u00E6').replace(/</g,'\u227A').replace(/\+/g,'\u2795');
  549. return text;
  550. }
  551. function openCloseOptions(evt) {
  552. var divOptions = getId('divOpt');
  553. if (!divOptions) //Show options
  554. {
  555. divOptions = createElement('div', {
  556. id : 'divOpt',
  557. style : 'border-top:2px solid #5A91D8;position:relative; padding:5px;'
  558. });
  559. getId('divDic').appendChild(divOptions);
  560. getId('optionsLink').style.visibility = 'hidden';
  561. //fields container
  562. divOptionsFields = createElement('p', {style: "margin:0px;padding:0px;line-height:160%;"});
  563. divOptions.appendChild(divOptionsFields);
  564. //from
  565. divOptionsFields.appendChild(createElement('span', null, null, 'From: '));
  566. divOptionsFields.appendChild(createElement('select', {
  567. id : 'optSelLangFrom'
  568. }, null, languagesGoogle));
  569. getId('optSelLangFrom').value = GM_getValue('from') ? GM_getValue('from') : 'auto';
  570. getId('optSelLangFrom').addEventListener('change', quickLookup, false);
  571. //to
  572. divOptionsFields.appendChild(createElement('br'));
  573. divOptionsFields.appendChild(createElement('span', null, null, 'To: &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;'));
  574. divOptionsFields.appendChild(createElement('select', {
  575. id : 'optSelLangTo'
  576. }, null, languagesGoogle));
  577. getId('optSelLangTo').value = GM_getValue('to') ? GM_getValue('to') : 'auto';
  578. getId('optSelLangTo').addEventListener('change', quickLookup, false);
  579. //to2
  580. divOptionsFields.appendChild(createElement('br'));
  581. divOptionsFields.appendChild(createElement('span', null, null, 'To 2: &nbsp;&nbsp;'));
  582. divOptionsFields.appendChild(createElement('select', {
  583. id : 'optSelLangTo2'
  584. }, null, '<option value="Disabled">Disabled</option>' + languagesGoogle));
  585. getId('optSelLangTo2').value = GM_getValue('to2') ? GM_getValue('to2') : 'Disabled';
  586. getId('optSelLangTo2').addEventListener('change', quickLookup, false);
  587. //to3
  588. divOptionsFields.appendChild(createElement('br'));
  589. divOptionsFields.appendChild(createElement('span', null, null, 'To 3: &nbsp;&nbsp;'));
  590. divOptionsFields.appendChild(createElement('select', {
  591. id : 'optSelLangTo3'
  592. }, null, '<option value="Disabled">Disabled</option>' + languagesGoogle));
  593. getId('optSelLangTo3').value = GM_getValue('to3') ? GM_getValue('to3') : 'Disabled';
  594. getId('optSelLangTo3').addEventListener('change', quickLookup, false);
  595. //use text to speech
  596. divOptionsFields.appendChild(createElement('br'));
  597. divOptionsFields.appendChild(createElement('input', {
  598. id : 'checkTTS',
  599. type : 'checkbox',
  600. style: "margin-left:0px;"
  601. }));
  602. divOptionsFields.appendChild(createElement('span', null, null, '<span title="The feature has many issues. You often have to refresh the page to launch the .mp3 file. If you use the langage auto-detection, you have to change the langage in the url of the new tab."> Display Text To Speech</span>'));
  603. getId('checkTTS').checked = GM_getValue('tts');
  604. //hide details
  605. divOptionsFields.appendChild(createElement('br'));
  606. divOptionsFields.appendChild(createElement('input', {
  607. id : 'checkDetails',
  608. type : 'checkbox',
  609. style: "margin-left:0px;"
  610. }));
  611. divOptionsFields.appendChild(createElement('span', null, null, ' Hide details by default'));
  612. getId('checkDetails').checked = GM_getValue('details');
  613. //show alternatives in details
  614. divOptionsFields.appendChild(createElement('br'));
  615. divOptionsFields.appendChild(createElement('input', {
  616. id: 'checkAlternatives',
  617. type: 'checkbox',
  618. style: "margin-left:0px;"
  619. }));
  620. divOptionsFields.appendChild(createElement('span', null, null, ' Show alternatives in details'));
  621. getId('checkAlternatives').checked = GM_getValue('alternatives');
  622. //show synonyms in definitions
  623. divOptionsFields.appendChild(createElement('br'));
  624. divOptionsFields.appendChild(createElement('input', {
  625. id: 'checkSynonyms',
  626. type: 'checkbox',
  627. style: "margin-left:0px;"
  628. }));
  629. divOptionsFields.appendChild(createElement('span', null, null, ' Show synonyms in definitions'));
  630. getId('checkSynonyms').checked = GM_getValue('synonyms');
  631. //font size
  632. divOptionsFields.appendChild(createElement('br'));
  633. divOptionsFields.appendChild(createElement('span', null, null, 'Font size: &nbsp;&nbsp;'));
  634. divOptionsFields.appendChild(createElement('select', {
  635. id : 'optFontSize',
  636. style: 'width:141px'
  637. }, null, '<option value="x-small">Extra small</option><option value="small">Small (default)</option><option value="medium">Medium</option><option value="large">Large</option>'));
  638. getId('optFontSize').value = GM_getValue('fontsize') ? GM_getValue('fontsize') : 'small';
  639. getId('optFontSize').addEventListener('change', quickLookup, false);
  640. //text color
  641. divOptionsFields.appendChild(createElement('br'));
  642. divOptionsFields.appendChild(createElement('span', null, null, 'Text color: '));
  643. divOptionsFields.appendChild(createElement('select', {
  644. id : 'optTextColor',
  645. style: 'width:141px'
  646. }, null, '<option value="Gray">Gray (default)</option><option value="Black">Black</option><option value="White">White</option><option value="CadetBlue">CadetBlue</option><option value="ForestGreen">ForestGreen</option><option value="FireBrick">FireBrick</option>'));
  647. getId('optTextColor').value = GM_getValue('textcolor') ? GM_getValue('textcolor') : 'Gray';
  648. getId('optTextColor').addEventListener('change', quickLookup, false);
  649. //use ctrl
  650. divOptionsFields.appendChild(createElement('br'));
  651. divOptionsFields.appendChild(createElement('input', {
  652. id : 'checkCtrl',
  653. type : 'checkbox',
  654. style: "margin-left:0px;"
  655. }));
  656. divOptionsFields.appendChild(createElement('span', null, null, ' Use Ctrl key'));
  657. getId('checkCtrl').checked = GM_getValue('ctrl');
  658. //use alt
  659. divOptionsFields.appendChild(createElement('br'));
  660. divOptionsFields.appendChild(createElement('input', {
  661. id : 'checkAlt',
  662. type : 'checkbox',
  663. style: "margin-left:0px;"
  664. }));
  665. divOptionsFields.appendChild(createElement('span', null, null, ' Use Alt key'));
  666. getId('checkAlt').checked = GM_getValue('alt');
  667. //delay display
  668. divOptionsFields.appendChild(createElement('br'));
  669. divOptionsFields.appendChild(createElement('span', null, null, 'Delay '));
  670. divOptionsFields.appendChild(createElement('input', {
  671. id: 'delayDisplay',
  672. type: 'text',
  673. style: "height:20px;width:50px;padding:0px;text-align:center;",
  674. }));
  675. divOptionsFields.appendChild(createElement('span', null, null, ' ms'));
  676. getId('delayDisplay').value = GM_getValue('delay') ? GM_getValue('delay') : '0';
  677. //save
  678. divOptionsFields.appendChild(createElement('br'));
  679. divOptionsFields.appendChild(createElement('a', {
  680. href : HREF_NO,
  681. class : "gootranslink"
  682. }, {
  683. type: 'click',
  684. handler: saveOptions,
  685. capture: false
  686. }, 'Save'));
  687. //reset
  688. divOptionsFields.appendChild(createElement('span', null, null, ' - '));
  689. divOptionsFields.appendChild(createElement('a', {
  690. href : HREF_NO,
  691. class : "gootranslink"
  692. }, {
  693. type: 'click',
  694. handler: resetOptions,
  695. capture: false
  696. }, 'Reset'));
  697. //cancel
  698. divOptionsFields.appendChild(createElement('span', null, null, ' - '));
  699. divOptionsFields.appendChild(createElement('a', {
  700. href : HREF_NO,
  701. class : "gootranslink"
  702. }, {
  703. type: 'click',
  704. handler: openCloseOptions,
  705. capture: false
  706. }, 'Cancel'));
  707. } else // Hide options
  708. {
  709. divOptions.parentNode.removeChild(divOptions);
  710. getId('optionsLink').style.visibility = 'visible';
  711. }
  712. }
  713. function saveOptions(evt) {
  714. var backgroundColor = getId('divDic').style.backgroundColor;
  715. var from = getId('optSelLangFrom').value;
  716. var to = getId('optSelLangTo').value;
  717. var to2 = getId('optSelLangTo2').value;
  718. var to3 = getId('optSelLangTo3').value;
  719. var tts = getId('checkTTS').checked;
  720. var details = getId('checkDetails').checked;
  721. var alternatives = getId('checkAlternatives').checked;
  722. var synonyms = getId('checkSynonyms').checked;
  723. var fontsize = getId('optFontSize').value;
  724. var textcolor = getId('optTextColor').value;
  725. var ctrl = getId('checkCtrl').checked;
  726. var alt = getId('checkAlt').checked;
  727. var delay = getId('delayDisplay').value;
  728. GM_setValue('backgroundColor', backgroundColor);
  729. GM_setValue('from', from);
  730. GM_setValue('to', to);
  731. GM_setValue('to2', to2);
  732. GM_setValue('to3', to3);
  733. GM_setValue('tts', tts);
  734. GM_setValue('details', details);
  735. GM_setValue('alternatives', alternatives);
  736. GM_setValue('synonyms', synonyms);
  737. GM_setValue('fontsize', fontsize);
  738. GM_setValue('textcolor', textcolor);
  739. GM_setValue('ctrl', ctrl);
  740. GM_setValue('alt', alt);
  741. GM_setValue('delay', delay);
  742. quickLookup();
  743. getId('divDic').removeChild(getId('divOpt'));
  744. getId('optionsLink').style.visibility = 'visible';
  745. }
  746. function resetOptions(evt) {
  747. GM_deleteValue('backgroundColor');
  748. GM_deleteValue('from');
  749. GM_deleteValue('to');
  750. GM_deleteValue('to2');
  751. GM_deleteValue('to3');
  752. GM_deleteValue('tts');
  753. GM_deleteValue('details');
  754. GM_deleteValue('alternatives');
  755. GM_deleteValue('synonyms');
  756. GM_deleteValue('fontsize');
  757. GM_deleteValue('textcolor');
  758. GM_deleteValue('ctrl');
  759. GM_deleteValue('alt');
  760. GM_deleteValue('delay');
  761. getId('divDic').parentNode.removeChild(getId('divDic'));
  762. }
  763. function css() {
  764. var style = createElement('style', {
  765. type : "text/css"
  766. }, null, "" +
  767. 'a.gootranslink:link {color: #0000FF !important; text-decoration: underline !important;}' +
  768. 'a.gootranslink:visited {color: #0000FF !important; text-decoration: underline !important;}' +
  769. 'a.gootranslink:hover {color: #0000FF !important; text-decoration: underline !important;}' +
  770. 'a.gootranslink:active {color: #0000FF !important; text-decoration: underline !important;}' +
  771. '.picker-wrapper,.slide-wrapper{position:relative;float:left}.picker-indicator,.slide-indicator{position:absolute;left:0;top:0;pointer-events:none}.picker,.slide{cursor:crosshair;float:left}.cp-default{background-color:gray;padding:12px;box-shadow:0 0 40px #000;border-radius:15px;float:left}.cp-default .picker{width:200px;height:200px}.cp-default .slide{width:30px;height:200px}.cp-default .slide-wrapper{margin-left:10px}.cp-default .picker-indicator{width:5px;height:5px;border:2px solid darkblue;-moz-border-radius:4px;-o-border-radius:4px;-webkit-border-radius:4px;border-radius:4px;opacity:.5;-ms-filter:"alpha(opacity=50)";filter:alpha(opacity=50);filter:alpha(opacity=50);background-color:white}.cp-default .slide-indicator{width:100%;height:10px;left:-4px;opacity:.6;-ms-filter:"alpha(opacity=60)";filter:alpha(opacity=60);filter:alpha(opacity=60);border:4px solid lightblue;-moz-border-radius:4px;-o-border-radius:4px;-webkit-border-radius:4px;border-radius:4px;background-color:white}.cp-small{padding:5px;background-color:white;float:left;border-radius:5px}.cp-small .picker{width:100px;height:100px}.cp-small .slide{width:15px;height:100px}.cp-small .slide-wrapper{margin-left:5px}.cp-small .picker-indicator{width:1px;height:1px;border:1px solid black;background-color:white}.cp-small .slide-indicator{width:100%;height:2px;left:0;background-color:black}.cp-fancy{padding:10px;background:-webkit-linear-gradient(top,#aaa 0,#222 100%);float:left;border:1px solid #999;box-shadow:inset 0 0 10px white}.cp-fancy .picker{width:200px;height:200px}.cp-fancy .slide{width:30px;height:200px}.cp-fancy .slide-wrapper{margin-left:10px}.cp-fancy .picker-indicator{width:24px;height:24px;background-image:url(http://cdn1.iconfinder.com/data/icons/fugue/bonus/icons-24/target.png)}.cp-fancy .slide-indicator{width:30px;height:31px;left:30px;background-image:url(http://cdn1.iconfinder.com/data/icons/bluecoral/Left.png)}.cp-normal{padding:10px;background-color:white;float:left;border:4px solid #d6d6d6;box-shadow:inset 0 0 10px white}.cp-normal .picker{width:200px;height:200px}.cp-normal .slide{width:30px;height:200px}.cp-normal .slide-wrapper{margin-left:10px}.cp-normal .picker-indicator{width:5px;height:5px;border:1px solid gray;opacity:.5;-ms-filter:"alpha(opacity=50)";filter:alpha(opacity=50);filter:alpha(opacity=50);background-color:white;pointer-events:none}.cp-normal .slide-indicator{width:100%;height:10px;left:-4px;opacity:.6;-ms-filter:"alpha(opacity=60)";filter:alpha(opacity=60);filter:alpha(opacity=60);border:4px solid gray;background-color:white;pointer-events:none}');
  772. getTag('head')[0].appendChild(style);
  773. }
  774. /*
  775. * Short functions to replace the document.createElement document.getElementById and document.getElementsByTagName
  776. */
  777. function createElement(type, attrArray, eventObj, html) {
  778. var node = document.createElement(type);
  779. for (var attr in attrArray)
  780. if (attrArray.hasOwnProperty(attr)) {
  781. node.setAttribute(attr, attrArray[attr]);
  782. }
  783. if (eventObj) {
  784. node.addEventListener(eventObj.type, eventObj.handler, eventObj.capture || false);
  785. }
  786. if (html)
  787. node.innerHTML = html;
  788. return node;
  789. }
  790. function getId(id, parent) {
  791. if (!parent)
  792. return document.getElementById(id);
  793. return parent.getElementById(id);
  794. }
  795. function getTag(name, parent) {
  796. if (!parent)
  797. return document.getElementsByTagName(name);
  798. return parent.getElementsByTagName(name);
  799. }
  800. /*
  801. * Drag and drop support adapted from http://www.hunlock.com/blogs/Javascript_Drag_and_Drop
  802. */
  803. var savedTarget = null; // The target layer (effectively vidPane)
  804. var orgCursor = null; // The original mouse style so we can restore it
  805. var dragOK = false; // True if we're allowed to move the element under mouse
  806. var dragXoffset = 0; // How much we've moved the element on the horozontal
  807. var dragYoffset = 0; // How much we've moved the element on the verticle
  808. var didDrag = false; // Set to true when we do a drag
  809. function moveHandler(e) {
  810. if (e == null)
  811. return; // { e = window.event }
  812. if (e.button <= 1 && dragOK) {
  813. savedTarget.style.left = e.clientX - dragXoffset + 'px';
  814. savedTarget.style.top = e.clientY - dragYoffset + 'px';
  815. return false;
  816. }
  817. }
  818. function dragCleanup(e) {
  819. document.removeEventListener('mousemove', moveHandler, false);
  820. document.removeEventListener('mouseup', dragCleanup, false);
  821. savedTarget.style.cursor = orgCursor;
  822. dragOK = false; // Its been dragged now
  823. didDrag = true;
  824. }
  825. function dragHandler(e) {
  826. var htype = '-moz-grabbing';
  827. if (e == null)
  828. return; // { e = window.event;} // htype='move';}
  829. var target = e.target; // != null ? e.target : e.srcElement;
  830. orgCursor = target.style.cursor;
  831. if (target.nodeName != 'DIV' && target.nodeName != 'P')
  832. return;
  833. if (target = clickedInsideID(target, 'divDic')) {
  834. savedTarget = target;
  835. target.style.cursor = htype;
  836. dragOK = true;
  837. dragXoffset = e.clientX - target.offsetLeft;
  838. dragYoffset = e.clientY - target.offsetTop;
  839. // Set the left before removing the right
  840. target.style.left = e.clientX - dragXoffset + 'px';
  841. target.style.right = null;
  842. document.addEventListener('mousemove', moveHandler, false);
  843. document.addEventListener('mouseup', dragCleanup, false);
  844. return false;
  845. }
  846. }
  847. function clickedInsideID(target, id) {
  848. if (target.getAttribute('id') == id)
  849. return getId(id);
  850. if (target.parentNode) {
  851. while (target = target.parentNode) {
  852. try {
  853. if (target.getAttribute('id') == id)
  854. return getId(id);
  855. } catch (e) {}
  856. }
  857. }
  858. return null;
  859. }
  860. // End drag code
  861. /*
  862. * Images
  863. */
  864. function images() {
  865. imgLookup = createElement('img', {
  866. border : 0
  867. });
  868. imgLookup.src = '';
  869. }
  870. if (typeof GM_deleteValue == 'undefined') {
  871. GM_addStyle = function (css) {
  872. var style = document.createElement('style');
  873. style.textContent = css;
  874. document.getElementsByTagName('head')[0].appendChild(style);
  875. }
  876. GM_deleteValue = function (name) {
  877. localStorage.removeItem(name);
  878. }
  879. GM_getValue = function (name, defaultValue) {
  880. var value = localStorage.getItem(name);
  881. if (!value)
  882. return defaultValue;
  883. var type = value[0];
  884. value = value.substring(1);
  885. switch (type) {
  886. case 'b':
  887. return value == 'true';
  888. case 'n':
  889. return Number(value);
  890. default:
  891. return value;
  892. }
  893. }
  894. GM_log = function (message) {
  895. console.log(message);
  896. }
  897. GM_openInTab = function (url) {
  898. return window.open(url, "_blank");
  899. }
  900. GM_registerMenuCommand = function (name, funk) {
  901. //todo
  902. }
  903. GM_setValue = function (name, value) {
  904. value = (typeof value)[0] + value;
  905. localStorage.setItem(name, value);
  906. }
  907. }
  908. /*
  909. * Cross browser support for GM functions
  910. * http://userscripts.org/topics/41177
  911. */
  912. function initCrossBrowserSupportForGmFunctions() {
  913. if (typeof GM_deleteValue == 'undefined') {
  914. GM_addStyle = function (css) {
  915. var style = document.createElement('style');
  916. style.textContent = css;
  917. document.getElementsByTagName('head')[0].appendChild(style);
  918. }
  919. GM_deleteValue = function (name) {
  920. localStorage.removeItem(name);
  921. }
  922. GM_getValue = function (name, defaultValue) {
  923. var value = localStorage.getItem(name);
  924. if (!value)
  925. return defaultValue;
  926. var type = value[0];
  927. value = value.substring(1);
  928. switch (type) {
  929. case 'b':
  930. return value == 'true';
  931. case 'n':
  932. return Number(value);
  933. default:
  934. return value;
  935. }
  936. }
  937. GM_log = function (message) {
  938. console.log(message);
  939. }
  940. GM_openInTab = function (url) {
  941. return window.open(url, "_blank");
  942. }
  943. GM_registerMenuCommand = function (name, funk) {
  944. //todo
  945. }
  946. GM_setValue = function (name, value) {
  947. value = (typeof value)[0] + value;
  948. localStorage.setItem(name, value);
  949. }
  950. }
  951. }