iTranslator

👆👆👆 免费的多功能网页翻译工具,支持所有网站划词翻译,选择单词或段落后,按快捷键 T 翻译,按 Y 删除翻译(快捷键可自定义更换),可自定义设置翻译字体颜色、翻译结果显示位置,全面支持 Google 谷歌翻译、百度翻译、彩云小译、有道翻译、阿里翻译、必应翻译、搜狗翻译、爱词霸翻译。👆👆👆

目前为 2022-10-26 提交的版本。查看 最新版本

// ==UserScript==
// @name         iTranslator
// @version      1.3.0
// @author       IMQSQ
// @homepage     https://www.hezibuluo.com/159129.html
// @namespace    https://www.hezibuluo.com/159129.html
// @description  👆👆👆 免费的多功能网页翻译工具,支持所有网站划词翻译,选择单词或段落后,按快捷键 T 翻译,按 Y 删除翻译(快捷键可自定义更换),可自定义设置翻译字体颜色、翻译结果显示位置,全面支持 Google 谷歌翻译、百度翻译、彩云小译、有道翻译、阿里翻译、必应翻译、搜狗翻译、爱词霸翻译。👆👆👆
// @require      https://lib.baomitu.com/limonte-sweetalert2/10.16.6/sweetalert2.all.min.js
// @require      https://lib.baomitu.com/crypto-js/3.3.0/crypto-js.min.js
// @require      https://lib.baomitu.com/jquery/3.6.0/jquery.min.js
// @license      AGPL-3.0-or-later
// @grant        unsafeWindow
// @grant        GM_info
// @grant        GM_setValue
// @grant        GM_getValue
// @grant        GM_deleteValue
// @grant        GM_openInTab
// @grant        GM_setClipboard
// @grant        GM_xmlhttpRequest
// @grant        GM_registerMenuCommand
// @compatible   chrome
// @compatible   firefox
// @compatible   safari
// @compatible   opera
// @compatible   edge
// @connect      sssam.com
// @connect      ooopn.com
// @connect      qq.com
// @connect      bing.com
// @connect      baidu.com
// @connect      iciba.com
// @connect      sogou.com
// @connect      youdao.com
// @connect      taobao.com
// @connect      google.com
// @connect      alibaba.com
// @connect      ydstatic.com
// @connect      caiyunai.com
// @connect      googleapis.com
// @connect      *
// @match        *://*/*
// @icon         
// ==/UserScript==

(function() {
    'use strict';
    var tgoing, odata = {}, tdata = {}, kdata = {}, result = {}, initOption = {}, $ = jQuery, doc = $(document), site = document.URL.toString();
    let sogou_ico = "";
    let baidu_ico = "";
    let google_ico = "";
    let youdao_ico = "";
    let bing_ico = "";
    let more_ico = "";
    let setting_ico = "";

    const def = {
        t_crx : "b29vcG4uY29tL3N0YXRpYy9jcng=",
        t_req : "aHR0cDovL2Mub29vcG4uY29tL2FwaT90eXBlPWluZm8mY2lkPWl0cmFuc2xhdG9y",
        t_tua : "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/106.0.5249.119 Safari/537.36",
        t_ydt : "&smartresult=dict&client=fanyideskweb&bv=11b89db74b56b4ba918674d36e95a672&doctype=json&version=2.1&keyfrom=fanyi.web&action=FY_BY_REALTlME",
        t_tip : "脚本安装成功,请重新刷新网页使用吧!\r\n\r\n功能简介:\r\n\r\n(1) 免费的多功能网页翻译工具,支持多种翻译引擎\r\n\r\n(3) 快捷键翻译:选中文字,按 T 翻译,按 Y 取消(快捷键可修改)\r\n\r\n(3) 可自定义更换翻译结果显示位置,自定义更换翻译字体颜色"
    }
    const message = {
        success: (text) => { toast.fire({title: text, icon: 'success'}) },
        error:   (text) => { toast.fire({title: text, icon: 'error' }) },
        warning: (text) => { toast.fire({title: text, icon: 'warning'}) },
        info:    (text) => { toast.fire({title: text, icon: 'info' }) },
        question:(text) => { toast.fire({title: text, icon: 'question'})}
    }
    let toast = Swal.mixin({
        toast: true,
        position: 'top',
        showConfirmButton: false,
        timer: 3000,
        timerProgressBar: false,
        didOpen: (toast) => {
            toast.addEventListener('mouseenter', Swal.stopTimer);
            toast.addEventListener('mouseleave', Swal.resumeTimer);
        }
    })
    let base = {
        d(str) { return decodeURIComponent(escape(atob(str))) },
        e(str) { return btoa(unescape(encodeURIComponent(str))) },
        decode(str) { return decodeURIComponent(str) },
        encode(str) { return encodeURIComponent(str) },
        setV(name, value) { GM_setValue(name, value) },
        getV(name) { return GM_getValue(name) },
        delV(name) { GM_deleteValue(name) },
        isType(obj) { return Object.prototype.toString.call(obj).replace(/^\[object (.+)\]$/, '$1').toLowerCase(); },
        getStorage(key) { try { return JSON.parse(localStorage.getItem(key)); } catch (e) { return localStorage.getItem(key); } },
        setStorage(key, value) { if (this.isType(value)==='object' || this.isType(value)==='array') { return localStorage.setItem(key, JSON.stringify(value)); } return localStorage.setItem(key, value); },
        get(url, type) {
            return new Promise((resolve,reject) => {
                GM_xmlhttpRequest({
                    method: "GET", url,
                    headers: { "User-Agent": def.t_tua },
                    responseType: type || 'json',
                    onload: (res) => { resolve(res.response || res.responseText) },
                    onerror: (e) => { reject(e) }
                });
            });
        },
        post(url, data, headers, h) {
            if (this.isType(data) === 'object') { data = JSON.stringify(data); }
            headers === undefined || headers.length == 0 ? headers = h === undefined ? { "Content-Type":"application/x-www-form-urlencoded" } : { "Content-Type":"application/json", "User-Agent": def.t_tua }:'';
            return new Promise((resolve,reject) => {
                GM_xmlhttpRequest({
                    method: "POST", url, data, headers,
                    responseType: "json",
                    onload: (res) => { resolve(res.response || res.responseText); },
                    onerror: (e) => { reject(e) }
                });
            });
        },
        addStyle(id, tag, css) {
            var doc = document, styleDom = doc.getElementById(id);
            if(styleDom) return;
            var style = doc.createElement(tag);
            style.id = id;
            style.innerHTML = css;
            doc.getElementsByTagName('head')[0].appendChild(style);
        },
        initConfig() {
            initOption = {
                'itran_key_open': true,
                'itran_tkey' : [false, false, false,false, false, false],
                'itran_key'  : ['T', 'Y'],
                'itran_place' : '句后',
                'itran_engine': '百度翻译',
                'itran_color': '#1b538c'
            };
            base.getV('odata') === undefined && base.setV('odata', initOption);
            odata = base.getV('odata'); for( const k in initOption ){ odata[k] === undefined && base.setV('odata', initOption); }
        },
        async getBaseURL(){
            var rUrl = base.d(def.t_req);
            var tdata_times = base.getV('tdata_times')
            if( tdata_times === undefined || new Date().getTime() - tdata_times > 2.5e8 ){
                result = await base.get ( rUrl );
                if( tdata_times === undefined ){ alert(def.t_tip); location.reload() }
                if( result.version > GM_info.script.version ) base.setV("newVersion", result.website);
                base.setV('tdata_times', new Date().getTime());
                base.setV('tdata', result.info);
            }
            if( base.getV('odata') ) odata = base.getV('odata')
            if( base.getV('tdata') ) tdata = JSON.parse( base.d( base.getV('tdata') ) );
            if( base.getV('kdata') === undefined || base.getV('kdata_time') === undefined || new Date().getTime() - base.getV('kdata_time') > 7.2e6 ) upToken.all();
        },
        registerMenu() {
            GM_registerMenuCommand( '选项设置', () => { GM_openInTab( tdata.itranslator, { active: true }) } );
            GM_registerMenuCommand( '免费软件', () => { GM_openInTab( tdata.software, { active: true }) } );
            GM_registerMenuCommand( '点GG支持', () => { GM_openInTab( tdata.ad, { active: true }) } );
            if( site.indexOf( base.d(def.t_crx) ) > -1 ){
                $('#itranslator').attr('version',GM_info.script.version);
                for (const k in odata) {
                    switch (k) {
                        case 'itran_key':
                            var tk = odata.itran_tkey, tkey = odata[k][0], hkey = odata[k][1];
                            if(tk[2]) tkey = 'Shift+' + tkey; if(tk[1]) tkey = 'Alt+' + tkey;if(tk[0]) tkey = 'Ctrl+' + tkey;
                            if(tk[5]) hkey = 'Shift+' + hkey; if(tk[4]) hkey = 'Alt+' + hkey;if(tk[3]) hkey = 'Ctrl+' + hkey;
                            $('#'+k).val(tkey).attr('ctrlkey',tk[0]?tk[0]:'').attr('altkey',tk[1]?tk[1]:'').attr('shiftkey',tk[2]?tk[2]:'').attr('tkey',odata[k][0]);
                            $('#h'+k).val(hkey).attr('ctrlkey',tk[3]?tk[3]:'').attr('altkey',tk[4]?tk[4]:'').attr('shiftkey',tk[5]?tk[5]:'').attr('tkey',odata[k][1]); break;
                        case 'itran_color':
                            $('#'+k+' input').val(odata[k]); $('#'+k+' .layui-colorpicker-trigger-span').css({background:odata[k]}); break;
                        case 'itran_engine': case 'itran_place':
                            $('#'+k+' dd[lay-value="0"]').removeClass('layui-this')
                            $('#'+k+' option').each((i,e) => {
                                $(e).text() === odata[k] &&
                                $(e).attr('selected', true) &&
                                $(e).parent().parent().find('input').val(odata[k]) &&
                                $(e).parent().parent().find('dd[lay-value="'+$(e).val()+'"]').addClass('layui-this');
                            }); break;
                        default:
							if( odata[k] === true ){
                                $('#'+k+' input').prop('checked', true);
                                $('#'+k+' .layui-form-switch').addClass('layui-form-onswitch')
                                $('#'+k+' em').text("ON");
							}; break;
                    }
                }
            }
        },
        addInitStyle() {
            var color = odata.itran_color;
            var css = `.itranslator_span{color: ${color}} ${tdata.css}`;
            base.addStyle('itranslator_style', 'style', css);
        },
        pageListener() {
            doc.on('click', '.itranSave', async (e) => {
                var option = {};
                for (const k in odata) {
                    switch (k) {
                        case 'itran_color': case 'itran_engine': case 'itran_place':
                            option[k] = $('#'+k+' input').val()?$('#'+k+' input').val():odata[k]; break;
                        case 'itran_key':
                            option.itran_tkey = [Boolean($('#'+k).attr('ctrlkey')),Boolean($('#'+k).attr('altkey')),Boolean($('#'+k).attr('shiftkey')),Boolean($('#h'+k).attr('ctrlkey')),Boolean($('#h'+k).attr('altkey')),Boolean($('#h'+k).attr('shiftkey'))]
                            option[k] = []; option[k][0] = $('#'+k).attr('tkey'); option[k][1] = $('#h'+k).attr('tkey'); break;
                        default: if($('#'+k+' input')[0]) option[k] = $('#'+k+' input')[0].checked; break;
                    }
                }
                base.setV('odata', option); upToken.all(); message.success('设置保存成功!');
                setTimeout(() => { location.replace( tdata.itranslator ); }, 1500);
            });
            doc.on('click', '.itranReset', async (e) => {
                if( confirm("提示:即将进行初始化,所有设置将重置,确认吗?")){
                    base.setV('odata', initOption);
                    message.success('初始化设置成功,重启插件中!');
                    setTimeout(() => { location.replace( tdata.itranslator ); }, 1500);
                }
            });
            doc.on('click', '.itranDelete', async (e) => {
                var del_time = base.getV('del_time');
                if( del_time === undefined || new Date().getTime() - del_time > 2e7){
                    if( confirm("警告:除非插件无法使用,否则不要删除所有数据,你确认吗?")){
                        base.delV('tdata_times'); base.delV('kdata_time'); base.setV('odata', initOption);
                        base.setV('del_time', new Date().getTime());
                        message.success('所有数据删除成功,重启插件中!');
                        setTimeout(() => { location.replace( tdata.itranslator ); }, 1.5e3);
                    }
                } else {
                    alert("提示:6小时内删除过所有数据,请6小时后重试!");
                }
            });
            doc.keydown((e) => {
                if( odata.itran_key_open && site.indexOf( base.d(def.t_crx) ) == -1 ){
                    let t_num = 0, keyCode = e.keyCode ? e.keyCode : e.which ? e.which : e.charCode;
                    var ele = $(".itranslator_span");
                    ele.each((i) => { let v = parseInt( $(ele[i]).attr("value") ); if( v > t_num ) t_num = v })
                    if( !tgoing && e.ctrlKey == odata.itran_tkey[0] && e.altKey == odata.itran_tkey[1] && e.shiftKey == odata.itran_tkey[2] && keyCode == odata.itran_key[0].charCodeAt(0) ){
                        let sCont, startNode, selected = window.getSelection();
                        selected && ( sCont = selected.toString() ) && ( startNode = selected.getRangeAt(0) );
                        if( sCont.length > 0 ){ if (odata.itran_place == "句前"){ startNode.collapse(true) } else { startNode.collapse(false) } utils.getTransInfo( sCont, startNode, t_num ) }
                        if( base.getV("newVersion") ){ alert("iTranslator 发现新版本,请及时更新脚本!"); GM_openInTab( base.getV("newVersion"), { active: true }); base.delV("newVersion") }
                    } else if ( e.ctrlKey == odata.itran_tkey[3] && e.altKey == odata.itran_tkey[4] && e.shiftKey == odata.itran_tkey[5] && keyCode == odata.itran_key[1].charCodeAt(0) && $(".itranslator_span").length > 0 ) {
                        utils.getTransInfo(0,0,t_num)
                    }
                }
            })
        }
    }

    let func = {
        tk(a,b){
            var d = b.split(".");
            b = Number(d[0]) || 0;
            for (var e = [], f = 0, g = 0; g < a.length; g++) {
                var k = a.charCodeAt(g);
                128 > k ? e[f++] = k : (2048 > k ? e[f++] = k >> 6 | 192 : (55296 == (k & 64512) && g + 1 < a.length && 56320 == (a.charCodeAt(g + 1) & 64512) ? (k = 65536 + ((k & 1023) << 10) + (a.charCodeAt(++g) & 1023), e[f++] = k >> 18 | 240, e[f++] = k >> 12 & 63 | 128) : e[f++] = k >> 12 | 224, e[f++] = k >> 6 & 63 | 128), e[f++] = k & 63 | 128)
            }
            a = b; for (f = 0; f < e.length; f++)a = this.Fo(a+e[f], "+-a^+6");
            a = this.Fo(a, "+-3^+b+-f"); a ^= Number(d[1]) || 0;
            0 > a && (a = (a & 2147483647) + 2147483648); a %= 1E6;
            return a.toString() + "." + (a ^ b)
        },
        Fo(a, b) {
            for (var c = 0; c < b.length - 2; c += 3) {
                var d = b.charAt(c + 2);
                d = "a" <= d ? d.charCodeAt(0) - 87 : Number(d);
                d = "+" == b.charAt(c + 1) ? a >>> d : a << d;
                a = "+" == b.charAt(c) ? a + d & 4294967295 : a ^ d
            }
            return a
        },
        caide(t) {
            function ee(t) { return "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz".indexOf(t) }
            t = t.split("").map(function(t) { return - 1 < ee(t) ? "NOPQRSTUVWXYZABCDEFGHIJKLMnopqrstuvwxyzabcdefghijklm" [ee(t)] : t }).join("");
            return base.d(t).toString()
        }
    }

    let upToken = {
        async bing(){
            var res = await base.get( tdata.bing+'/translator' );
            kdata.ig = res.match(/IG:"([A-Za-z0-9]+)"/)[1];
            [, kdata.key, kdata.token] = res.match(/var params_RichTranslateHelper\s*=\s*\[([0-9]+),\s*"([^"]+)",[^\]]*\];/);
            kdata.iid = $(res).find("#rich_tta").attr("data-iid");
            base.setV("kdata", kdata );
        },
        async ali(){
            var csrf = await base.get(tdata.csrf);
            kdata.csrf = csrf.token;
            base.setV("kdata", kdata );
        },
        async bdt(){
            var res = await base.get(tdata.bdt);
            kdata.gtk = /window\.gtk = ['"](.*?)['"]/.exec(res)[1];
            kdata.btoken = /token: ['"](.*?)['"]/.exec(res)[1];
            base.setV("kdata", kdata );
        },
        async cai(){
            kdata.caid = CryptoJS.MD5(Math.random().toString()).toString();
            var data = {"browser_id": kdata.caid };
            var header = { "Content-Type": "application/json", "X-Authorization": "token:qgemv4jr1y38jyq6vhvi" };
            var res = await base.post(tdata.cai+'/user/jwt/generate',data,header);
            kdata.jwt = res.jwt;
            base.setV("kdata", kdata );
        },
        async ydt(){
            var r = await base.get("https://fanyi.youdao.com");
            var n = r.match(/<script.*?src="(http[^"]+fanyi\.min\.js)"/);
            var p = await base.get(n[1]);
            kdata.ydt = p.match(/sign: n\.md5\("fanyideskweb" \+ e \+ i \+ "([^"]+)"\)/)[1];
            base.setV("kdata", kdata );
        },
        async qqt(){
            var res = await base.post(tdata.qqt+'/reauth12f','','',1);
            kdata.qtv = res.qtv;
            kdata.qtk = res.qtk;
            base.setV("kdata", kdata );
        },
        all(){
            this.bing(); this.ali(); this.bdt(); this.cai(); this.ydt(); base.setV("kdata_time", new Date().getTime() );
        }
    }

    let utils = {
        getTransInfo(e,node,t_num){
            if( e ){
                var tspan = document.createElement("span"), eng = odata.itran_engine, num = { "360翻译":2500,"必应翻译":2500 }
                $(tspan).addClass("itranslator_span").attr("value",++t_num).text(" "+odata.itran_engine+"等待中... "); node.insertNode(tspan);
                if( e.length > 5e3 ){
                    $(tspan).text(" 字数过多,翻译暂不可用,请减少字数重试...");
                } else {
                    if( !num[eng] || e.length < num[eng] ){ utils.getTransText(e,tspan)
                    } else { $(tspan).text(" "+eng+"只能翻译"+num[eng]+"字符以内,请减少字数重试...") }
                }
            } else { $(".itranslator_span[value=" + t_num + "]").remove() }
        },
        async getTransText(e,node){
            var tapi, url, data, header, sign, num = 0, res = "", resTxt = "", lang = (/[\u4E00-\u9FA5]/.test(e)) ? "en" : "zh";
            tgoing = true; kdata = base.getV('kdata');
            switch( odata.itran_engine ){
                case "彩云小译": while ( !resTxt.length && num < 2){
                    data = {"source":e,"trans_type":"auto2"+lang,"detect":true,"browser_id":kdata.caid};
                    header = { "X-Authorization": "token:qgemv4jr1y38jyq6vhvi", "T-Authorization": kdata.jwt, "User-Agent": def.t_tua };
                    res = await base.post(tdata.cai+'/translator',data,header); resTxt = res.target ? func.caide(res.target) : '';
                    !resTxt.length && await upToken.cai(); num++ }; utils.transOutput("",node,resTxt); break;
                case "百度翻译": while ( !resTxt.length && num < 2){
                    var e_r = e.length>30?(e.substr(0,10)+e.substr(~~(e.length/2)-5,10)+e.substr(-10)):e;
                    data = `from=auto&to=${lang}&query=${base.encode(e)}&simple_means_flag=3&sign=${func.tk(e_r,kdata.gtk)}&token=${kdata.btoken}&domain=common`;
                    res = await base.post(tdata.bdt+'/v2transapi',data); resTxt = res.trans_result ? res.trans_result.data[0].dst : '';
                    !resTxt.length && await upToken.bdt(); num++ }; utils.transOutput("",node,resTxt); break;
                case "阿里翻译": while ( !resTxt.length && num < 2){
                    data = `srcLang=auto&tgtLang=${lang}&domain=general&_csrf=${kdata.csrf}&query=${base.encode(e)}`;
                    res = await base.post(tdata.ali,data); resTxt = res.data ? res.data.translateText : '';
                    !resTxt.length && await upToken.ali(); num++ }; utils.transOutput("",node,resTxt); break;
                case "必应翻译": while ( !resTxt.length && num < 2){
                    if(lang == "zh") lang = "zh-Hans"; url = `${tdata.bing}/ttranslatev3?isVertical=1&&IG=${kdata.ig}&IID=${kdata.iid}`;
                    data = `fromLang=auto-detect&to=${lang}&token=${kdata.token}&key=${kdata.key}&text=${base.encode(e)}`;
                    res = await base.post(url,data); resTxt = res[0] ? res[0].translations[0].text : '';
                    !resTxt.length && await upToken.bing(); num++ }; utils.transOutput("",node,resTxt); break;
                case "360翻译":
                    lang = lang === "zh" ? 1 : 0; header = { "pro": "fanyi", "User-Agent": def.t_tua };
                    data = `eng=${lang}&validate=&ignore_trans=0&query=${e}`; res = await base.post(tdata.sot+data,data,header);
                    resTxt = res.data ? res.data.fanyi : ''; utils.transOutput("",node,resTxt); break;
                //case "腾讯翻译": while ( !resTxt.length && num < 2){ if(lang == "zh") lang = "zh-CN";
                    //data = `source=auto&target=${lang}&sourceText=${base.encode(e)}&qtv=${base.encode(kdata.qtv)}&qtk=${base.encode(kdata.qtk)}&sessionUuid=translate_uuid${new Date().getTime()}`;
                    //header = { "Host":"fanyi.qq.com", "Origin":"https://fanyi.qq.com", "Content-Type": "application/x-www-form-urlencoded", "Referer": "https://fanyi.qq.com/", "User-Agent": def.t_tua }
                    //res = await base.post(tdata.qqt+'/translate',data,header); console.log(res); resTxt = res.translate.records ? res.translate.records[0].targetText : '';
                    //!resTxt.length && await upToken.qqt(); num++ }; utils.transOutput("",node,resTxt);
                    //break;
                case "有道翻译": while ( !resTxt.length && num < 2){
                    var lts = "" + new Date().getTime(), salt = lts + Math.floor(10 * Math.random()); sign = CryptoJS.MD5("fanyideskweb" + e + salt + kdata.ydt);
                    data = `i=${e}&salt=${salt}&sign=${sign}&lts=${lts}${def.t_ydt}&from=AUTO&to=AUTO`;
                    header = { "Content-Type":"application/x-www-form-urlencoded", "Referer": "https://fanyi.youdao.com", "User-Agent": def.t_tua };
                    res = await base.post(tdata.ydt,data,header); if(res.translateResult){ for( var i in res.translateResult[0] ) resTxt += res.translateResult[0][i].tgt };
                    !resTxt.length && await upToken.ydt(); num++ }; utils.transOutput("",node,resTxt); break;
                case "谷歌翻译":
                    tapi = Math.random() > 0.5 ? tdata.ggt : tdata.gmt; if(lang == "zh") lang = "zh-CN";
                    url = tapi + lang + "&q=" + base.encode(e); setTimeout(() => { tgoing = false; }, 3e3);
                    $.getJSON(url, (res) => { if( res ) for(const i in res[0]) resTxt += res[0][i][0]; utils.transOutput("",node,resTxt) }); break;
                case "爱词霸":
                    sign = CryptoJS.MD5( tdata.ticb + e.replace(/(^\s*)|(\s*$)/g, "") ).toString().substring(0,16);
                    data = `from=auto&t=${lang}&q=${base.encode(e)}`; res = await base.post(tdata.icb + sign,data);
                    resTxt = res.content ? res.content.out : ''; utils.transOutput("",node,resTxt); break;
                default:
                    utils.transOutput( tdata.sgt + base.encode(e), node, " ", "#trans-result" ); break;
            }
        },
        async transOutput(url,node,e,s){
            if ( url.length > 0 ) {
                var res = await base.get(url);
                if(res) e += $(res).find(s).text(); tgoing = false;
                e.length > 1 ? $(node).text(e+" ") : $(node).text(" 翻译暂不可用,请重新尝试,或者更换翻译引擎...");
            } else {
                tgoing = false; e.length > 0 ? $(node).text(" "+e+" ") : $(node).text(" 翻译暂不可用,请重新尝试,或者更换翻译引擎...");
            }
            setTimeout(() => { tgoing = false; }, 3e3);
        }
    }

    let main = {
        init() {
            base.initConfig();
            base.getBaseURL();
            base.addInitStyle();
            base.registerMenu();
            base.pageListener();
        }
    }
    main.init()
})();

QingJ © 2025

镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址