// ==UserScript== // @id googletranslatorbytoprapid // @name Google select text translator // @name:zh-CN 谷歌点击划词翻译 // @version 2.1 // @namespace https://greasyfork.org/en/scripts/36842 // @author Toprapid // @copyright 2017+,toprapid // @description Translate any selected text into the language you wants with google translator // @description:zh-cn 谷歌点击划词翻译综合插件 // @match http://*/* // @match https://*/* // @grant GM_registerMenuCommand // @grant GM_addStyle // @grant GM_xmlhttpRequest // @grant GM_getValue // @grant GM_setValue // @downloadURL none // ==/UserScript== //版本 var gtVer="2.0"; //缓存已经翻译过的信息内容 var transCache = []; var transMd5Cache=[]; //缓存已经取得的音频数据内容 var transSoundCache=[]; var transSoundMd5Cache=[]; // var sourceTranslateLanguageNameInStorage = "sourceTranslateLanguage"; var targetTranslateLanguageNameInStorage = "targetTranslateLanguage"; var translatorSettingLanguageInStorage="translatorSettingLanguage"; var defaultSourceLanguage = "-1"; var defaultTranslateLanguage = "0"; var currentSourceLanguage = parseInt(GM_getValue(sourceTranslateLanguageNameInStorage, defaultSourceLanguage)); var currentTranslateLanguage= parseInt(GM_getValue(targetTranslateLanguageNameInStorage, defaultTranslateLanguage)); var translatorSettingLanguage = GM_getValue(translatorSettingLanguageInStorage, "0") === "0" ? 0 : 1; //国家与语言代码字符串 //国家 var uLanguageNames = [["中文", "中文", "中文", "英语", "英语", "英语", "英语", "英语", "英语", "英语", "英语", "芬兰语", "英语", "丹麦语", "英语", "希伯来语", "英语", "英语", "英语", "英语", "英语", "英语", "英语", "英语", "英语", "英语", "韩文", "日语", "荷兰语", "荷兰语", "葡萄牙语", "葡萄牙语", "法语", "法语", "法语", "法语", "法语", "西班牙语", "西班牙语", "西班牙语", "西班牙语", "西班牙语", "西班牙语", "西班牙语", "德语", "德语", "德语", "俄语", "意大利语", "希腊语", "挪威语", "匈牙利语", "土耳其语", "捷克语", "斯洛文尼亚语", "波兰语", "瑞典语", "西班牙语"], ["Chinese", "Chinese", "Chinese", "English", "English", "English", "English", "English", "English", "English", "English", "Finnish", "English", "Danish", "English", "Hebrew", "English", "English", "English", "English", "English", "English", "English", "English", "English", "English", "Korean", "Japanese", "Dutch", "Dutch", "Portuguese", "Portuguese", "French", "French", "French", "French", "French", "Spanish", "Spanish", "Spanish", "Spanish", "Spanish", "Spanish", "Spanish", "German", "German", "German", "Russian", "Italian", "Greek", "Norwegian", "Hungarian", "Turkish", "Czech", "Slovenian", "Polish", "Sweden Language", "Spanish"]]; //国家所对应的地区 var uRegionNames = [["简体", "繁体台湾", "繁体香港", "香港", "美国", "英国", "全球", "加拿大", "澳大利亚", "爱尔兰", "芬兰", "芬兰", "丹麦", "丹麦", "以色列", "以色列", "南非", "印度", "挪威", "新加坡", "新西兰", "印度尼西亚", "菲律宾", "泰国", "马来西亚", "阿拉伯", "韩国", "日本", "荷兰", "比利时", "葡萄牙", "巴西", "法国", "卢森堡", "瑞士", "比利时", "加拿大", "拉丁美洲", "西班牙", "阿根廷", "美国", "墨西哥", "哥伦比亚", "波多黎各", "德国", "奥地利", "瑞士", "俄罗斯", "意大利", "希腊", "挪威", "匈牙利", "土耳其", "捷克共和国", "斯洛文尼亚语", "波兰", "瑞典", "智利"],["Simplified", "Traditional Taiwan", "Traditional Hong Kong", "Hong Kong", "America", "United Kingdom", "Global", "Canada", "Australia", "Irish", "Finland", "Finland", "Denmark", "Denmark", "Israel", "Israel", "South Africa", "Indian", "Norway", "Singapore", "New Zealand", "Indonesia", "the Philippines", "Thai", "Malaysia", "Arab", "Korea", "Japan", "Dutch", "Belgium", "Portugal", "Brazil", "French", "Luxembourg", "Switzerland", "Belgium", "Canada", "Latin America", "Spain", "Argentina", "America", "Mexico", "Columbia", "Puerto Rico", "Germany", "Austria", "Switzerland", "Russia", "Italy", "Greece", "Norway", "Hungary", "Turkey", "Czech Republic", "Slovenian", "Poland", "Sweden", "Chile"]]; //最终代码 var uLanguageCode = ["zh-cn", "zh-tw", "zh-hk", "en-hk", "en-us", "en-gb", "en-ww", "en-ca", "en-au", "en-ie", "en-fi", "fi-fi", "en-dk", "da-dk", "en-il", "he-il", "en-za", "en-in", "en-no", "en-sg", "en-nz", "en-id", "en-ph", "en-th", "en-my", "en-xa", "ko-kr", "ja-jp", "nl-nl", "nl-be", "pt-pt", "pt-br", "fr-fr", "fr-lu", "fr-ch", "fr-be", "fr-ca", "es-la", "es-es", "es-ar", "es-us", "es-mx", "es-co", "es-pr", "de-de", "de-at", "de-ch", "ru-ru", "it-it", "el-gr", "no-no", "hu-hu", "tr-tr", "cs-cz", "sl-sl", "pl-pl", "sv-se", "es-cl"]; //google翻译api变量 var UA = navigator.userAgent; var googleDomain = ["translate.google.cn", "translate.google.com"]; var dictURL = "https://" + googleDomain[translatorSettingLanguage] + "/translate_a/single?client=t"; var ttsURL = "http://" + googleDomain[translatorSettingLanguage] + "/translate_tts?client=t"; var audioContext= new AudioContext(); var tokenNameInStorage=["google_value_tk_zh_cn","google_value_tk_en_us"]; var isEnabled = true; var oldText = ""; var speakerIconBase64Data=""; var ANS_TYPE_WORD=0,ANS_TYPE_SENTENCE=1,ANS_TYPE_ZDIC_HTML=2,ANS_TYPE_YOUDAO_WORD=3,ANS_TYPE_YOUDAO_SENTENCE=4; var CACHE_TYPE_GOOGLE_RESULT=0,CACHE_TYPE_ZDIC_HTML=1,CACHE_TYPE_YOUDAO_WORD=2,CACHE_TYPE_YOUDAO_SENTENCE=3; var lingoesServerConfigNameInStorage="lingoesServerConfig"; var defaultLingoesServerConfigValue="0|0|127.0.0.1|11111"; var lingoes_isEnabled=!1,lingoes_showingWay=0,lingoes_serverIp="127.0.0.1",lingoes_serverPort="11111"; var youdaoConfigNameInStorage="youdaoConfig"; var defaultYoudaoConfigValue="0|0|0"; var youdao_isEnabled=!1,youdao_isOnlyForSingleWord=!0; var hotKeyNameInStorage="hotKeyName"; var hotKeyNameForQSInStorage = "QSHotKeyName"; var defaultHotKeyValue= "1|1|1|81"; //Ctrl+Alt+Q enabled var defaultHotKeyValueForQS="1|1|1|68"; //Ctrl+Alt+D enabled var hotKey_Ctrl=!0,hotKey_Alt=!0,hotKey_Code=81, hotkey_isActive = !0, isHotkeyEventRegistered=!1; var hotKey_Ctrl_forQS = !0, hotKey_Alt_forQS = !0, hotKey_Code_forQS = 68, hotkey_isActive_forQS = !0, isHotkeyEventRegistered_forQS = !1; var hotkeyType_forPlugin=0; var hotkeyType_forQS=1; //current target language for playing sound var curTarLanForSpeaker = uLanguageCode[currentTranslateLanguage]; getHotKeyValue(hotkeyType_forPlugin); getHotKeyValue(hotkeyType_forQS); getLingoesConfigValue(); getYoudaoConfigValue(); window.document.body.addEventListener("mouseup", googleTranslate, false); var on = function (node, e, f) { node.addEventListener(e, f, false); }; var $ = function (s) { return document.getElementById('gt_prefs_' + s); }; var setup=function() { var d = document; if ($('setup')) return; var styleNode = GM_addStyle('\ #gt_prefs_setup { position:fixed;z-index:2147483647;top:30px;right:60px;padding:20px 30px;background:#eee;width:600px;border:1px solid black; }\ #gt_prefs_setup * { color:black;text-align:left;line-height:normal;font-size:12px; }\ #gt_prefs_setup a { color:black;text-decoration:underline; }\ #gt_prefs_setup div { text-align:center;font-weight:bold;font-size:14px; }\ #gt_prefs_setup ul { margin:15px 0 15px 0;padding:0;list-style:none;background:#eee;border:0; }\ #gt_prefs_setup input, #gt_prefs_setup select { border:1px solid gray;width:auto;padding:2px;background:white; }\ #gt_prefs_setup li { margin:0;padding:6px 0;vertical-align:middle;background:#eee;border:0 }\ #gt_prefs_setup button { width:150px;margin:0 10px;text-align:center;}\ #gt_prefs_setup textarea { width:98%; height:60px; margin:3px 0; }\ #gt_prefs_setup b { font-weight: bold; font-family: "微软雅黑", sans-serif; }\ #gt_prefs_setup button:disabled { color: graytext; }\ '); var div = d.createElement('div'); div.id = 'gt_prefs_setup'; d.body.appendChild(div); var sb_Title=["谷歌点击划词翻译设置","Google select text translator Setting"], sb_CurStatus = [["(启用中)", "(禁用中)"], ["(Enabled)", "(Disabled)"]], sb_HotkeyForQuickSearch = ["快速查询热键:", "Hotkey for Quick Search:"], sb_HotkeyActivatorForQuickSearch = ["启用热键", "Enable Hotkey"], sb_Usage = ["插件开关热键:", "Hotkey for Plugin switch:"], sb_HotKeyActivator=["启用热键","Enable Hotkey"], sb_CurVersion=["当前版本:","Current Version "], sb_CurAuthor=[",作者:",", Author: "], sb_CurUILanguage=[",Current UI Language:",", 当前界面语言:"], sb_UILanguage=[["Chinese Simplified","English"],["英文","中文"]], sb_SourceLanguage = ["指定源语言:", "Select the source language:"], sb_SourceLanguageSetToAuto = ["自动检测", "Auto Detect"], sb_TargetLanguage=["设置目标翻译语言:","Select the target language to translate:"], sb_Confirmation=[["确定","取消"],["Save","Cancel"]], sb_UseLingoesAPILocalServer=["启用本地灵格斯API服务(需要本机安装灵格斯词典软件,并在其设置中启用API服务器)。","Enable Lingoes Local API Server (Need Lingoes software installed, and enable it's API server)."], sb_LingoesAPIServerIP=["灵格斯API服务器 -- IP地址:","Lingoes Local API Server -- IP:"], sb_LingoesAPIServerPort=["端口号:","Port:"], sb_LingoesShowingWay=[["显示取词窗口","主界面取词","主界面翻译文字","查询方式:"],["Showing query result","Main UI query","Main UI translator","Query way:"]], sb_LingoesErrorWarning=[["输入的IPv4地址不规范!本地地址一般为:127.0.0.1","端口号必须为0-65535间的数字!"],["The Ipv4 address is wrong! For localhost server, use 127.0.0.1 please!","Port number must be a digital between 0 to 65535"]], sb_UseYoudaoSever=["启用有道翻译代替谷歌翻译引擎(注:仅限中英互译):","Use Youdao translator (PS: only used between Chinese and English):"], sb_YoudaoTranslationWay=[["全面启用","仅限单词"],["Totaly enabled","Only for words"]], sb_ErrorWarning=["输入的热键必须为字母或数字!","The hot key must be alphabet or digital!"]; var divHtml = `
[\.-9a-z; \u6c49\u5178\u6f22\u25ce\u7db2\u7f51\uff08\uff09\u3010\u3011\u300a\u300b\u3014\u3015\u300e\u300f\(\)\[\]]+<\/p>/gi,""); callBackFunc(resultHtml, resp.finalUrl, true); return; } } } catch (e) { console.log(e); } callBackFunc(null, null, false); } }); } function getTargetAnswer(resultObj) { if (resultObj) { switch(resultObj[0]){ case CACHE_TYPE_ZDIC_HTML: return [ANS_TYPE_ZDIC_HTML, resultObj[1]]; case CACHE_TYPE_YOUDAO_SENTENCE: return [ANS_TYPE_YOUDAO_SENTENCE, resultObj[1]]; case CACHE_TYPE_YOUDAO_WORD: return [ANS_TYPE_YOUDAO_WORD, resultObj[1]]; case CACHE_TYPE_GOOGLE_RESULT: try { var arr = JSON.parse(resultObj[1]); if (arr[0]) { var len = arr[0].length; //the normal return is len>=2 if (len > 1) { var resultTxt = ""; //if target string is a word var ansType = ANS_TYPE_SENTENCE; if (arr[1]) { resultTxt+=`${arr[2]}\n${arr[0][0][1]}\n`; if (arr[0][1][3] != null) if (arr[0][1][3] != "") resultTxt += `true\n[${arr[0][1][3]}]\n`; resultTxt += `${arr[0][0][0]} ${arr[0][1][2] == null ? "" : "[" + arr[0][1][2] + "]"}\n${arr[0][0][0]}\n`; var len1 = arr[1].length; for (let z1 = 0; z1 < len1; z1++) { resultTxt += `${arr[1][z1][0]} `; var len2 = arr[1][z1][1].length; for (let n1 = 0; n1 < len2; n1++) { resultTxt += `${arr[1][z1][1][n1]}; `; } resultTxt += "\n"; } ansType = ANS_TYPE_WORD; } else { for (let z2 = 0; z2 < len - 1; z2++) { resultTxt += arr[0][z2][0]; } } } else { if (len === 1) resultTxt = arr[0][0][0]; else return null; } return resultTxt === "" ? null : [ansType, resultTxt]; } } catch (e) { console.log("错误:" + e); } break; default: break; } } return null; } function searchInTranslateCache(targetMd5) { for (var n = 0; n < transCache.length; n++) { if (transMd5Cache[n] === targetMd5) return transCache[n]; } return null; } function searchInTransSoundCache(targetMd5) { for (var n = 0; n < transSoundCache.length; n++) { if (transSoundMd5Cache[n] === targetMd5) return transSoundCache[n]; } return null; } function init_google_value_tk() { var url = "https://" + googleDomain[translatorSettingLanguage]; var timeout = setTimeout(function () { this.abort(); }, 2000); GM_xmlhttpRequest({ method: "GET", url: url, onreadystatechange: function (resp) { if (resp.readyState == 4) { clearTimeout(timeout); if (resp.status == 200) { init_google_value_tk_parse(resp.responseText); } } } }); } function init_google_value_tk_parse(responseText) { let indexStr=",tkk:'"; let bIdx=responseText.indexOf(indexStr); if(bIdx>0){ bIdx+=indexStr.length; let eIdx=responseText.indexOf("',",bIdx); if(eIdx>0){ let tkk=responseText.substring(bIdx,eIdx); if(tkk!=null)GM_setValue(tokenNameInStorage[translatorSettingLanguage], tkk); return; } } } // return token for the new API function googleTK(text) { // view-source:https://translate.google.com/translate/releases/twsfe_w_20151214_RC03/r/js/desktop_module_main.js && TKK from HTML var uM = GM_getValue(tokenNameInStorage[translatorSettingLanguage]); if (uM == 'undefined' || uM == null) { init_google_value_tk(); uM = GM_getValue(tokenNameInStorage[translatorSettingLanguage]); } ; var cb = "&"; var k = ""; var Gf = "="; var Vb = "+-a^+6"; var t = "a"; var Yb = "+"; var Zb = "+-3^+b+-f"; var jd = "."; var sM = function (a) { return function () { return a; }; }; 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; }; var vM = function (a) { var b; if (null !== uM) { b = uM; } else { b = sM(String.fromCharCode(84)); var c = sM(String.fromCharCode(75)); b = [b(), b()]; b[1] = c(); b = (uM = window[b.join(c())] || k) || k; } var d = sM(String.fromCharCode(116)), c = sM(String.fromCharCode(107)), d = [d(), d()]; d[1] = c(); c = cb + d.join(k) + Gf; d = b.split(jd); b = Number(d[0]) || 0; for (var e = [], f = 0, g = 0; g < a.length; g++) { var m = a.charCodeAt(g); 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); } a = b || 0; for (f = 0; f < e.length; f++) { a += e[f], a = tM(a, Vb) } a = tM(a, Zb); a ^= Number(d[1]) || 0; 0 > a && (a = (a & 2147483647) + 2147483648); a %= 1E6; return a.toString() + jd + (a ^ b); }; return vM(text); } // Google Translate Request function Request(txt, sl, tl, parse) { var tk = googleTK(txt); var Url = dictURL + "&hl=auto" + "&sl=" + sl + "&tl=" + tl + "&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" + "&tk=" + tk + "&q=" + encodeURI(txt); var method = 'POST'; var Data = ''; var Hdr = { "User-Agent": UA, "Accept": "text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8", "Accept-Encoding": "gzip, deflate" }; var Q = Url.split('&q='); Url = Q[0]; Data = '&q=' + Q[1]; Hdr["Content-Length"] = Data.length + ''; Hdr["Content-Type"] = "application/x-www-form-urlencoded; charset=UTF-8"; GM_xmlhttpRequest({ method: method, url: Url, data: Data, headers: Hdr, onload: function (resp) { try { parse(resp.responseText, tl); } catch (e) { console.log(e); } } }); } function playTTS(lang,text) { text = text.replace(/[«»'"]/g, ' '); var txtMd5 = MD5(text, lang + CACHE_TYPE_GOOGLE_RESULT, null); var buf= searchInTransSoundCache(txtMd5); if (buf){ playSound(buf); return; } var tk = googleTK(text); var Url = ttsURL + "&ie=UTF-8&total=1&idx=0" + "&tl=" + lang + "&q=" + text + "&textlen=" + text.length + "&tk=" + tk; GM_xmlhttpRequest({ method: "GET", url: Url, responseType: 'arraybuffer', onload: function (response) { try { audioContext.decodeAudioData(response.response, function (buffer) { transSoundMd5Cache.push(txtMd5); transSoundCache.push(buffer); playSound(buffer); }); } catch (e) { console.log(e); } } }); } 'use strict'; /* * Add integers, wrapping at 2^32. This uses 16-bit operations internally * to work around bugs in some JS interpreters. */ function safe_add(x, y) { var lsw = (x & 0xFFFF) + (y & 0xFFFF), msw = (x >> 16) + (y >> 16) + (lsw >> 16); return (msw << 16) | (lsw & 0xFFFF); } /* * Bitwise rotate a 32-bit number to the left. */ function bit_rol(num, cnt) { return (num << cnt) | (num >>> (32 - cnt)); } /* * These functions implement the four basic operations the algorithm uses. */ function md5_cmn(q, a, b, x, s, t) { return safe_add(bit_rol(safe_add(safe_add(a, q), safe_add(x, t)), s), b); } function md5_ff(a, b, c, d, x, s, t) { return md5_cmn((b & c) | ((~b) & d), a, b, x, s, t); } function md5_gg(a, b, c, d, x, s, t) { return md5_cmn((b & d) | (c & (~d)), a, b, x, s, t); } function md5_hh(a, b, c, d, x, s, t) { return md5_cmn(b ^ c ^ d, a, b, x, s, t); } function md5_ii(a, b, c, d, x, s, t) { return md5_cmn(c ^ (b | (~d)), a, b, x, s, t); } /* * Calculate the MD5 of an array of little-endian words, and a bit length. */ function binl_md5(x, len) { /* append padding */ x[len >> 5] |= 0x80 << ((len) % 32); x[(((len + 64) >>> 9) << 4) + 14] = len; var i, olda, oldb, oldc, oldd, a = 1732584193, b = -271733879, c = -1732584194, d = 271733878; for (i = 0; i < x.length; i += 16) { olda = a; oldb = b; oldc = c; oldd = d; a = md5_ff(a, b, c, d, x[i], 7, -680876936); d = md5_ff(d, a, b, c, x[i + 1], 12, -389564586); c = md5_ff(c, d, a, b, x[i + 2], 17, 606105819); b = md5_ff(b, c, d, a, x[i + 3], 22, -1044525330); a = md5_ff(a, b, c, d, x[i + 4], 7, -176418897); d = md5_ff(d, a, b, c, x[i + 5], 12, 1200080426); c = md5_ff(c, d, a, b, x[i + 6], 17, -1473231341); b = md5_ff(b, c, d, a, x[i + 7], 22, -45705983); a = md5_ff(a, b, c, d, x[i + 8], 7, 1770035416); d = md5_ff(d, a, b, c, x[i + 9], 12, -1958414417); c = md5_ff(c, d, a, b, x[i + 10], 17, -42063); b = md5_ff(b, c, d, a, x[i + 11], 22, -1990404162); a = md5_ff(a, b, c, d, x[i + 12], 7, 1804603682); d = md5_ff(d, a, b, c, x[i + 13], 12, -40341101); c = md5_ff(c, d, a, b, x[i + 14], 17, -1502002290); b = md5_ff(b, c, d, a, x[i + 15], 22, 1236535329); a = md5_gg(a, b, c, d, x[i + 1], 5, -165796510); d = md5_gg(d, a, b, c, x[i + 6], 9, -1069501632); c = md5_gg(c, d, a, b, x[i + 11], 14, 643717713); b = md5_gg(b, c, d, a, x[i], 20, -373897302); a = md5_gg(a, b, c, d, x[i + 5], 5, -701558691); d = md5_gg(d, a, b, c, x[i + 10], 9, 38016083); c = md5_gg(c, d, a, b, x[i + 15], 14, -660478335); b = md5_gg(b, c, d, a, x[i + 4], 20, -405537848); a = md5_gg(a, b, c, d, x[i + 9], 5, 568446438); d = md5_gg(d, a, b, c, x[i + 14], 9, -1019803690); c = md5_gg(c, d, a, b, x[i + 3], 14, -187363961); b = md5_gg(b, c, d, a, x[i + 8], 20, 1163531501); a = md5_gg(a, b, c, d, x[i + 13], 5, -1444681467); d = md5_gg(d, a, b, c, x[i + 2], 9, -51403784); c = md5_gg(c, d, a, b, x[i + 7], 14, 1735328473); b = md5_gg(b, c, d, a, x[i + 12], 20, -1926607734); a = md5_hh(a, b, c, d, x[i + 5], 4, -378558); d = md5_hh(d, a, b, c, x[i + 8], 11, -2022574463); c = md5_hh(c, d, a, b, x[i + 11], 16, 1839030562); b = md5_hh(b, c, d, a, x[i + 14], 23, -35309556); a = md5_hh(a, b, c, d, x[i + 1], 4, -1530992060); d = md5_hh(d, a, b, c, x[i + 4], 11, 1272893353); c = md5_hh(c, d, a, b, x[i + 7], 16, -155497632); b = md5_hh(b, c, d, a, x[i + 10], 23, -1094730640); a = md5_hh(a, b, c, d, x[i + 13], 4, 681279174); d = md5_hh(d, a, b, c, x[i], 11, -358537222); c = md5_hh(c, d, a, b, x[i + 3], 16, -722521979); b = md5_hh(b, c, d, a, x[i + 6], 23, 76029189); a = md5_hh(a, b, c, d, x[i + 9], 4, -640364487); d = md5_hh(d, a, b, c, x[i + 12], 11, -421815835); c = md5_hh(c, d, a, b, x[i + 15], 16, 530742520); b = md5_hh(b, c, d, a, x[i + 2], 23, -995338651); a = md5_ii(a, b, c, d, x[i], 6, -198630844); d = md5_ii(d, a, b, c, x[i + 7], 10, 1126891415); c = md5_ii(c, d, a, b, x[i + 14], 15, -1416354905); b = md5_ii(b, c, d, a, x[i + 5], 21, -57434055); a = md5_ii(a, b, c, d, x[i + 12], 6, 1700485571); d = md5_ii(d, a, b, c, x[i + 3], 10, -1894986606); c = md5_ii(c, d, a, b, x[i + 10], 15, -1051523); b = md5_ii(b, c, d, a, x[i + 1], 21, -2054922799); a = md5_ii(a, b, c, d, x[i + 8], 6, 1873313359); d = md5_ii(d, a, b, c, x[i + 15], 10, -30611744); c = md5_ii(c, d, a, b, x[i + 6], 15, -1560198380); b = md5_ii(b, c, d, a, x[i + 13], 21, 1309151649); a = md5_ii(a, b, c, d, x[i + 4], 6, -145523070); d = md5_ii(d, a, b, c, x[i + 11], 10, -1120210379); c = md5_ii(c, d, a, b, x[i + 2], 15, 718787259); b = md5_ii(b, c, d, a, x[i + 9], 21, -343485551); a = safe_add(a, olda); b = safe_add(b, oldb); c = safe_add(c, oldc); d = safe_add(d, oldd); } return [a, b, c, d]; } /* * Convert an array of little-endian words to a string */ function binl2rstr(input) { var i, output = ''; for (i = 0; i < input.length * 32; i += 8) { output += String.fromCharCode((input[i >> 5] >>> (i % 32)) & 0xFF); } return output; } /* * Convert a raw string to an array of little-endian words * Characters >255 have their high-byte silently ignored. */ function rstr2binl(input) { var i, output = []; output[(input.length >> 2) - 1] = undefined; for (i = 0; i < output.length; i += 1) { output[i] = 0; } for (i = 0; i < input.length * 8; i += 8) { output[i >> 5] |= (input.charCodeAt(i / 8) & 0xFF) << (i % 32); } return output; } /* * Calculate the MD5 of a raw string */ function rstr_md5(s) { return binl2rstr(binl_md5(rstr2binl(s), s.length * 8)); } /* * Calculate the HMAC-MD5, of a key and some data (raw strings) */ function rstr_hmac_md5(key, data) { var i, bkey = rstr2binl(key), ipad = [], opad = [], hash; ipad[15] = opad[15] = undefined; if (bkey.length > 16) { bkey = binl_md5(bkey, key.length * 8); } for (i = 0; i < 16; i += 1) { ipad[i] = bkey[i] ^ 0x36363636; opad[i] = bkey[i] ^ 0x5C5C5C5C; } hash = binl_md5(ipad.concat(rstr2binl(data)), 512 + data.length * 8); return binl2rstr(binl_md5(opad.concat(hash), 512 + 128)); } /* * Convert a raw string to a hex string */ function rstr2hex(input) { var hex_tab = '0123456789abcdef', output = '', x, i; for (i = 0; i < input.length; i += 1) { x = input.charCodeAt(i); output += hex_tab.charAt((x >>> 4) & 0x0F) + hex_tab.charAt(x & 0x0F); } return output; } /* * Encode a string as utf-8 */ function str2rstr_utf8(input) { return unescape(encodeURIComponent(input)); } /* * Take string arguments and return either raw or hex encoded strings */ function raw_md5(s) { return rstr_md5(str2rstr_utf8(s)); } function hex_md5(s) { return rstr2hex(raw_md5(s)); } function raw_hmac_md5(k, d) { return rstr_hmac_md5(str2rstr_utf8(k), str2rstr_utf8(d)); } function hex_hmac_md5(k, d) { return rstr2hex(raw_hmac_md5(k, d)); } function MD5(string, key, raw) { if (!key) { if (!raw) { return hex_md5(string); } else { return raw_md5(string); } } if (!raw) { return hex_hmac_md5(key, string); } else { return raw_hmac_md5(key, string); } }