大声朗读 - TTS辅助阅读

极为好用的网页朗读器,集成微软/百度TTS,在浏览器上实时文本转语音 支持移动端| 全平台适用

// ==UserScript==
// @name         大声朗读 - TTS辅助阅读
// @namespace    http://tampermonkey.net/
// @version      0.3.2.2
// @description  极为好用的网页朗读器,集成微软/百度TTS,在浏览器上实时文本转语音 支持移动端| 全平台适用
// @author       GAEE
// @match        http://*/*
// @match        https://*/*
// @exclude      https://example.com/
// @icon         
// @run-at       document-end
// @grant        GM_download
// @connect      azure.microsoft.com
// @connect      ai.baidu.com
// @grant        GM_addStyle
// @grant        GM_setValue
// @grant        GM_getValue
// @grant        GM_openInTab
// @grant        GM_xmlhttpRequest
// @grant        GM_registerMenuCommand
// @grant        GM_unregisterMenuCommand
// @grant        GM_notification
// @grant        unsafeWindow
// @license      none
// @require      https://cdn.staticfile.org/jquery/3.6.0/jquery.min.js
// @require      https://greasyfork.org/scripts/442967-tts-sdk/code/TTS_SDK.js?version=1037522
// @require      https://cdn.jsdelivr.net/npm/microsoft-cognitiveservices-speech-sdk@1.15.1/distrib/browser/microsoft.cognitiveservices.speech.sdk.bundle-min.js
/* globals jQuery, $, waitForKeyElements */
// ==/UserScript==
//this.$ = this.jQuery = jQuery.noConflict(true);
let version = "0.3.2.2 Alpha 实验版";
var Global_TEXT = "";
var MicrosoftTTS_info = {
    person: "zh-CN-XiaoxiaoNeural",
    speed: 1,
    pitch: 1,
    status: false,
};
var BaiduTTS_info = {
    person: 4003,
    speed: 5,
    pitch: 5,
    status: false,
};
var info = {
    type: "microsoft",
};
var setting = {
    version: version,
    speech_type: "all_text",
};
var TTS_GLOBAL,TTS_MORE_GLOBAL;

(function() {
    'use strict';

    init_voice_setting();

    GM_addStyle('body{user-select:auto !important; -webkit-user-select:auto !important; -moz-user-select:auto !important; -ms-user-select:auto !important; }');
    GM_addStyle('#GAEE_TTS_IFRAME,.div {bottom: 10%;transform: translate(10px);position: fixed;z-index: 1000;background-spanor: transparent;transform: translate(0);}.TTS_Button {display: flex;justify-content: center;align-items: center;height: 31px;width: 20px;border-radius: 8px;padding: 7px 12px;font-size: 12px;spanor: #969696;//border-radius: 50%;box-shadow: 0 2px 10px rgb(0 0 0 / 5%);background-spanor: white;background: rgba(255, 255, 255, 0.9);margin-left: 8px;transform-origin: center;transition: .2s;cursor: pointer;flex-direction: spanumn;}.TTS_Button:hover {background: #e3e5e7;}.TTS_Card {position: fixed;//position:relative;box-sizing: border-box;padding: 18px;width: 360px;height: 200px;border-radius: 8px;background: white;box-shadow: 0 3px 12px rgb(0 0 0 / 20%);font-size: 16px;bottom: 18%;margin-left: 9px;-moz-user-select: none;-khtml-user-select: none;user-select: none;z-index: 1000;}.TTS_Card .close {position: absolute;top: 14px;right: 14px;width: 14px;height: 14px;cursor: pointer;}.TTS_Card .title {margin-bottom: 16px;margin-left: 2px;//spanor: black;//font-size: 16px;line-height: 22px;display: flex;}.TTS_Card .title .title_text {spanor: black;font-size: 16px;margin-right: 30px;}.TTS_Card .title .TTS_Change {display: flex;}.TTS_Card .title .TTS_Change .il {margin-left: 40px;font-size: 16px;font-family: "微软雅黑";color: #969696;border-bottom: 2px solid #ffffff;cursor: pointer;}.TTS_Card .title .TTS_Change .il:hover {border-bottom: 2px dashed #F00;}.TTS_Card .title .TTS_Change .il:focus {outline: none;border-bottom: 2px solid #F00;}.TTS_Card .login-tip-content-item>* {display: flex;align-items: center;margin-bottom: 14px;width: 50%;height: 26px;}.setting {//position:relative;position: absolute;weight: 100%;height: auto;//max-height: 145px;//background:green;//overflow: auto;}.setting .row {margin: auto;max-height: 50px;width: 100%;overflow: auto;// overflow-x: scroll;// overflow-y: hidden;//white-space: nowrap;margin-bottom: 12px;}.setting .col {width: auto;}.setting .span {width: 70px;height: 30px;float: left;margin-right: 1px;//color: red;color: #2C3E50;cursor: pointer;}.setting .setting_down {display: flex;}.setting .speech_set {font-size: 13px;margin-top: 3px;}.setting .slider {width: 170px;height: 20px;margin: 0;transform-origin: 75px 75px;}.setting .others {display: flex;font-size: inherit;margin-left: auto;}.setting .others .more {background-color: #DCDCDC;border: 1px solid #DCDCDC;color: #fff;display: inline-block;font-size: 9px;padding: 2px 18px;height: 20px;cursor: pointer;}.setting .others .listen {background-color: #0078d4;border: 1px solid #0078d4;border-radius: 3px;color: #fff;display: inline-block;font-size: 9px;padding: 5px 8px;cursor: pointer;}.setting .others .more:active {background-color: #C0C0C0;}.setting .others .listen:active {background-color: #0062ad;}::-webkit-scrollbar {width: 4px;height: 4px;background-color: transparent;}::-webkit-scrollbar-track {-webkit-box-shadow: inset 0 0 6px transparent;border-radius: 10px;background-color: white;}::-webkit-scrollbar-thumb {border-radius: 10px;-webkit-box-shadow: inset 0 0 6px rgba(0, 0, 0, .3);background-color: #555;}.setting a {color: #2a5caa;text-decoration: none;}');

    let BUTTON = '<div class="div"><div class="TTS_Button" id="TTS_Button"><svg width="1.7em" height="1.7em" t="1649228019321" class="icon" viewBox="0 0 1024 1024" version="1.1" xmlns="http://www.w3.org/2000/svg" p-id="2481" width="200" height="200"><path id="icon-start"          d="M920.8 600.5V491c0-55.7-11-109.8-32.7-160.9-20.9-49.3-50.9-93.6-88.9-131.6-38.1-38.1-82.4-68-131.6-88.9-51.1-21.7-105.2-32.7-160.9-32.7s-109.8 11-160.9 32.7c-49.3 20.9-93.6 50.9-131.6 88.9-38.1 38.1-68 82.4-88.9 131.6-21.9 51.1-32.9 105.2-32.9 160.9v109.5c-21.7 33.2-33.2 73.2-33.2 115.6 0 42.6 11.6 82.7 33.4 116 17.8 27.2 41.7 47.9 68.5 59.8 9.6 32.2 39.5 55.7 74.7 55.7H265c43 0 78-35 78-78v-307c0-43-35-78-78-78h-29.2c-35.2 0-65.1 23.5-74.7 55.7-4.3 1.9-8.6 4.1-12.8 6.5V491c0-48.1 9.5-94.9 28.2-139 18.1-42.6 44-81 77-113.9 33-33 71.3-58.9 113.9-77 44.1-18.7 90.9-28.2 139-28.2 48.1 0 94.9 9.5 139 28.2 42.6 18.1 81 44 113.9 77 33 33 58.9 71.3 77 113.9 18.7 44.1 28.2 90.9 28.2 139v55.8c-4.2-2.4-8.4-4.6-12.8-6.5-9.6-32.2-39.5-55.7-74.7-55.7h-29.2c-43 0-78 35-78 78v307c0 43 35 78 78 78H777c35.2 0 65.1-23.5 74.7-55.7 26.8-12 50.7-32.7 68.5-59.8 21.9-33.3 33.4-73.4 33.4-116 0.4-42.4-11.1-82.3-32.8-115.6z"          fill="#969696" p-id="2482"></path><path id="icon-playing"          d="M512 816.213333c-23.466667 0-42.666667-19.2-42.666667-42.666666V250.88c0-23.466667 19.2-42.666667 42.666667-42.666667s42.666667 19.2 42.666667 42.666667v522.666667c0 23.466667-19.2 42.666667-42.666667 42.666666zM341.333333 597.333333c-23.466667 0-42.666667-19.2-42.666666-42.666666v-85.333334c0-23.466667 19.2-42.666667 42.666666-42.666666s42.666667 19.2 42.666667 42.666666v85.333334c0 23.466667-19.2 42.666667-42.666667 42.666666zM853.333333 640c-23.466667 0-42.666667-19.2-42.666666-42.666667v-170.666666c0-23.466667 19.2-42.666667 42.666666-42.666667s42.666667 19.2 42.666667 42.666667v170.666666c0 23.466667-19.2 42.666667-42.666667 42.666667zM170.666667 682.666667c-23.466667 0-42.666667-19.2-42.666667-42.666667V384c0-23.466667 19.2-42.666667 42.666667-42.666667s42.666667 19.2 42.666666 42.666667v256c0 23.466667-19.2 42.666667-42.666666 42.666667zM682.666667 727.893333c-23.466667 0-42.666667-19.2-42.666667-42.666666V338.773333c0-23.466667 19.2-42.666667 42.666667-42.666666s42.666667 19.2 42.666666 42.666666v346.88a42.666667 42.666667 0 0 1-42.666666 42.24z"          fill="#969696" p-id="1413"></path></svg></div></div>';
    let CARD = '<div class="TTS_Card" id="TTS_Card"><div class="close" id="card_close"><svg viewBox="0 0 100 100"><path d="M2 2 L98 98 M 98 2 L2 98Z" stroke-width="10px" stroke="#969696" stroke-linecap="round"></path></svg></div><div class="title"><div class="title_text">        TTS</div><div class="TTS_Change"><div id="microsoft" tabindex="1" class="il">          微软</div><div id="baidu" tabindex="2" class="il" style="border-bottom: 2px solid #F00;">          百度</div><div id="about" tabindex="3" class="il">          关于</div></div></div><div class="card"><div class="setting" id="microsoft_card" style="display: none;"><div>          代码正在构建中</div></div><div class="setting" id="baidu_card" style="display: show;"><div class="row"><div class="col"><div class="span" aria-label="4003">              度逍遥</div><div class="span" aria-label="4115">              度小贤</div><div class="span" aria-label="4119">              度小鹿</div><div class="span" aria-label="4100">              度小雯</div><div class="span" aria-label="4106">              度博文</div><div class="span" aria-label="4103">              度米朵</div></div></div><div class="speech_set"><label for="speed" id="speedlabel">语速: 5</label><div class="slider"><input type="range" id="speed" name="speed" min="0" max="15" value="5" class="slider__input" data-bi-id="demo-rate-slider" aria-label="语速"></div></div><div class="speech_set"><label for="pitch" id="pitchlabel">音调: 5</label><div class="slider"><input type="range" id="pitch" name="pitch" min="0" max="15" value="5" class="slider__input" data-bi-id="demo-pitch-slider" aria-label="音调"></div></div></div></div><div class="setting" id="about_card" style="display: none;"><div>        本项目由GAEE维护支持  <a href="https://greasyfork.org/zh-CN/scripts/429810-%E5%A4%A7%E5%A3%B0%E6%9C%97%E8%AF%BB">首页</a></div><div><img style="height:32%;width:32%;" src="https://img.github.luxe/2022/d1807e2d06008.png" alt="90kskmwly1smny0t4dz6vh750k8n.png" title="支持一下" /></div></div></div>';

    if(!document.querySelector("#GAEE_TTS_IFRAME")){
        var b = document.createElement('iframe');
        b.setAttribute("id","GAEE_TTS_IFRAME");
        b.setAttribute("title","GAEE_TTS");
        b.style.cssText = "height:60px; width:70px; border: unset; scrolling:no; display: flex;";
        document.body.appendChild(b);
    }
    var _TTS_ = document.querySelector("#GAEE_TTS_IFRAME");
    TTS_GLOBAL = $($("#GAEE_TTS_IFRAME")[0].contentWindow.document);
    add_TTS_Style(_TTS_.contentWindow.document,'.div {bottom: 10px;transform: translate(10px);position: fixed;z-index: 1000;background-spanor: transparent;transform: translate(0);}.TTS_Button {display: flex;justify-content: center;align-items: center;height: 31px;width: 20px;border-radius: 8px;padding: 7px 12px;font-size: 12px;spanor: #969696;//border-radius: 50%;box-shadow: 0 2px 10px rgb(0 0 0 / 5%);background-spanor: white;background: rgba(255, 255, 255, 0.9);margin-left: 8px;transform-origin: center;transition: .2s;cursor: pointer;flex-direction: spanumn;}.TTS_Button:hover {background: #e3e5e7;}.TTS_Card {position: fixed;//position:relative;box-sizing: border-box;padding: 18px;width: 360px;height: 200px;border-radius: 8px;background: white;box-shadow: 0 3px 12px rgb(0 0 0 / 20%);font-size: 16px;bottom: unset;margin-left: 9px;-moz-user-select: none;-khtml-user-select: none;user-select: none;z-index: 1000;}.TTS_Card .close {position: absolute;top: 14px;right: 14px;width: 14px;height: 14px;cursor: pointer;}.TTS_Card .title {margin-bottom: 16px;margin-left: 2px;//spanor: black;//font-size: 16px;line-height: 22px;display: flex;}.TTS_Card .title .title_text {spanor: black;font-size: 16px;margin-right: 30px;}.TTS_Card .title .TTS_Change {display: flex;}.TTS_Card .title .TTS_Change .il {margin-left: 40px;font-size: 16px;font-family: "微软雅黑";color: #969696;border-bottom: 2px solid #ffffff;cursor: pointer;}.TTS_Card .title .TTS_Change .il:hover {border-bottom: 2px dashed #F00;}.TTS_Card .title .TTS_Change .il:focus {outline: none;border-bottom: 2px solid #F00;}.TTS_Card .login-tip-content-item>* {display: flex;align-items: center;margin-bottom: 14px;width: 50%;height: 26px;}.setting {//position:relative;position: absolute;weight: 100%;height: auto;//max-height: 145px;//background:green;//overflow: auto;}.setting .row {margin: auto;max-height: 50px;width: 100%;overflow: auto;// overflow-x: scroll;// overflow-y: hidden;//white-space: nowrap;margin-bottom: 12px;}.setting .col {width: auto;}.setting .span {width: 70px;height: 30px;float: left;margin-right: 1px;//color: red;color: #2C3E50;cursor: pointer;}.setting .setting_down {display: flex;}.setting .speech_set {font-size: 13px;margin-top: 3px;}.setting .slider {width: 170px;height: 20px;margin: 0;transform-origin: 75px 75px;}.setting .others {display: flex;font-size: inherit;margin-left: auto;}.setting .others .more {background-color: #DCDCDC;border: 1px solid #DCDCDC;color: #fff;display: inline-block;font-size: 9px;padding: 2px 18px;height: 20px;cursor: pointer;}.setting .others .listen {background-color: #0078d4;border: 1px solid #0078d4;border-radius: 3px;color: #fff;display: inline-block;font-size: 9px;padding: 5px 8px;cursor: pointer;}.setting .others .more:active {background-color: #C0C0C0;}.setting .others .listen:active {background-color: #0062ad;}::-webkit-scrollbar {width: 4px;height: 4px;background-color: transparent;}::-webkit-scrollbar-track {-webkit-box-shadow: inset 0 0 6px transparent;border-radius: 10px;background-color: white;}::-webkit-scrollbar-thumb {border-radius: 10px;-webkit-box-shadow: inset 0 0 6px rgba(0, 0, 0, .3);background-color: #555;}.setting a {color: #2a5caa;text-decoration: none;}');
    _TTS_.contentWindow.document.body.innerHTML = BUTTON;

    UIinit();
    rebind();

    var timeOutEvent,mobile_timeOutEvent;
    var longClick = 0;
    var mobile_longClick = 0;
    TTS_GLOBAL.find("#TTS_Button").on({
        mousedown: function(e) {
            if(e && e.preventDefault) {
                e.preventDefault();
            }
            longClick = 0;
            timeOutEvent = setTimeout(function() {
                longClick = 1;
                ShowCard();
            }, 300);
        },
        mouseup: function() {
            clearTimeout(timeOutEvent);
        },
        click: function(e) {
            clearTimeout(timeOutEvent);
            if (longClick == 0) {
                CLICKED();
            }
            return false;
        }
    });
    TTS_GLOBAL.find("#TTS_Button").on({
        touchstart: function(e) {
            mobile_longClick = 0;
            mobile_timeOutEvent = setTimeout(function() {
                mobile_longClick = 1;
                ShowCard();
            }, 300);
        },
        touchmove: function(e) {
            clearTimeout(mobile_timeOutEvent);
            mobile_timeOutEvent = 0;
            e.preventDefault();
        },
        touchend: function(e) {
            clearTimeout(mobile_timeOutEvent);
            if (mobile_timeOutEvent != 0 && mobile_longClick == 0) {
                CLICKED();
            }
            return false;
        }
    });
    function CLICKED(){
        init();
        var TEXT = "";
        if(setting.speech_type == "all_text"){
            TEXT = window.getSelection().toString() || Get_InnerText();
        }else if(setting.speech_type == "next_text"){
            TEXT = Get_InnerText().slice(Get_InnerText().indexOf(window.getSelection().toString()));
        }
        if(info.type == "baidu"){
            if(!BaiduTTS_info.status){
                BaiduTTS(TEXT);
            }
        }else if(info.type == "microsoft"){
            if(!MicrosoftTTS_info.status){
                AzureTTS(TEXT);
            }
        }
    }
})();

function add_TTS_Style(_TTS_,css) {
    var head, style;
    head = _TTS_.getElementsByTagName('head')[0];
    if (!head) { console.log("未能添加TTS样式");return; }
    style = document.createElement('style');
    style.type = 'text/css';
    style.innerHTML = css;
    head.appendChild(style);
}

function change_TTS_Size(w,h){
    $("#GAEE_TTS_IFRAME").width(w);
    $("#GAEE_TTS_IFRAME").height(h);
}

function UIinit() {
    icon_change("init");
    TTS_GLOBAL.find('.span').css('color', '#2C3E50');
    TTS_GLOBAL.find('.span').each(function() {
        if (info.type == "baidu") {
            ShowBaiduCard();
            if ($(this).attr('aria-label') == BaiduTTS_info.person) {
                $(this).css('color', 'red');
            }
        } else if (info.type == "microsoft") {
            ShowMicrosoftCard();
            if ($(this).attr('aria-label') == MicrosoftTTS_info.person) {
                $(this).css('color', 'red');
            }
        }
    });
    TTS_GLOBAL.find("#microsoft_speedlabel").text("\u8bed\u901f: " + MicrosoftTTS_info.speed);
    TTS_GLOBAL.find("#microsoft_pitchlabel").text("\u97f3\u8c03: " + MicrosoftTTS_info.pitch);
    TTS_GLOBAL.find("#microsoft_speed").val(MicrosoftTTS_info.speed*100-100);
    TTS_GLOBAL.find("#microsoft_pitch").val(MicrosoftTTS_info.pitch*50-50);
    TTS_GLOBAL.find("#baidu_speedlabel").text("\u8bed\u901f: " + BaiduTTS_info.speed);
    TTS_GLOBAL.find("#baidu_pitchlabel").text("\u97f3\u8c03: " + BaiduTTS_info.pitch);
    TTS_GLOBAL.find("#baidu_speed").val(BaiduTTS_info.speed);
    TTS_GLOBAL.find("#baidu_pitch").val(BaiduTTS_info.pitch);
}

function UIinit2(){
    if(setting.speech_type == "all_text"){
        TTS_MORE_GLOBAL.find("#c1").prop("checked",false);
    }else if(setting.speech_type == "next_text"){
        TTS_MORE_GLOBAL.find("#c1").prop("checked",true);
    }
}

function ShowMicrosoftCard() {
    TTS_GLOBAL.find("#microsoft_card").show();
    TTS_GLOBAL.find("#baidu_card").hide();
    TTS_GLOBAL.find("#about_card").hide();
    TTS_GLOBAL.find("#microsoft").css("border-bottom", "2px solid #F00");
    TTS_GLOBAL.find("#baidu").css("border-bottom", "");
    TTS_GLOBAL.find("#about").css("border-bottom", "");
}

function ShowBaiduCard() {
    TTS_GLOBAL.find("#microsoft_card").hide();
    TTS_GLOBAL.find("#baidu_card").show();
    TTS_GLOBAL.find("#about_card").hide();
    TTS_GLOBAL.find("#microsoft").css("border-bottom", "");
    TTS_GLOBAL.find("#baidu").css("border-bottom", "2px solid #F00");
    TTS_GLOBAL.find("#about").css("border-bottom", "");
}

function ShowAboutCard() {
    TTS_GLOBAL.find("#microsoft_card").hide();
    TTS_GLOBAL.find("#baidu_card").hide();
    TTS_GLOBAL.find("#about_card").show();
    TTS_GLOBAL.find("#microsoft").css("border-bottom", "");
    TTS_GLOBAL.find("#baidu").css("border-bottom", "");
    TTS_GLOBAL.find("#about").css("border-bottom", "2px solid #F00");
}

function rebind() {
    TTS_GLOBAL.find("#microsoft_speed").change(function() {
        var i = $(this).val();
        var n = (i - (-100)) / (200 - (-100)) * 3;
        var speedValue = Math.abs(n) < 1 ? n.toPrecision(2) : n.toPrecision(3);
        TTS_GLOBAL.find("#microsoft_speedlabel").text("\u8bed\u901f: " + speedValue);
        MicrosoftTTS_info.speed = speedValue;
    });
    TTS_GLOBAL.find("#microsoft_pitch").change(function() {
        var i = $(this).val();
        var n = (i - (-50)) / (50 - (-50)) * 2;
        var pitchValue = Math.abs(n) < 1 ? n.toPrecision(2) : n.toPrecision(3);
        TTS_GLOBAL.find("#microsoft_pitchlabel").text("\u97f3\u8c03: " + pitchValue);
        MicrosoftTTS_info.pitch = pitchValue;
    });
    TTS_GLOBAL.find("#baidu_speed").change(function() {
        var speedValue = $(this).val();
        TTS_GLOBAL.find("#baidu_speedlabel").text("\u8bed\u901f: " + speedValue);
        BaiduTTS_info.speed = speedValue;
    });
    TTS_GLOBAL.find("#baidu_pitch").change(function() {
        var pitchValue = $(this).val();
        TTS_GLOBAL.find("#baidu_pitchlabel").text("\u97f3\u8c03: " + pitchValue);
        BaiduTTS_info.pitch = pitchValue;
    });
    TTS_GLOBAL.find('.span').on('click', function(e) {
        TTS_GLOBAL.find('.span').css('color', '#2C3E50');
        TTS_GLOBAL.find(this).css('color', 'red');
        var id = $(this).parent().parent().parent().attr('id');
        if (id == "baidu_card") {
            BaiduTTS_info.person = $(this).attr('aria-label');
            info.type = "baidu";
        } else if (id == "microsoft_card") {
            MicrosoftTTS_info.person = $(this).attr('aria-label');
            info.type = "microsoft";
        }
    });
    TTS_GLOBAL.find(".more").click(function() {
        Show_more_card();
    });
    TTS_GLOBAL.find(".listen").click(function() {
        TryListen();
    });
    TTS_GLOBAL.find("#card_close").click(function() {
        TTS_GLOBAL.find("#TTS_Card").remove();
        change_TTS_Size(70,60);
        voice_setting();
    });
    TTS_GLOBAL.find("#microsoft").click(function() {
        ShowMicrosoftCard();
    });
    TTS_GLOBAL.find("#baidu").click(function() {
        ShowBaiduCard();
    });
    TTS_GLOBAL.find("#about").click(function() {
        ShowAboutCard();
    });
}

function rebind2(){
    TTS_MORE_GLOBAL.find("#c1").click(function() {
        if($(this).prop("checked")){
            setting.speech_type = "next_text";
        }else{
            setting.speech_type = "all_text";
        }
    });
    TTS_MORE_GLOBAL.find("#card_close").click(function() {
        $("#GAEE_TTS_MORE").remove();
        voice_setting();
    });
}

function ShowCard() {
    if (TTS_GLOBAL.find("#TTS_Card").length > 0) {
        return;
    }
    change_TTS_Size(387,275);
    let CARD =
        `<div class="TTS_Card" id="TTS_Card"><div class="close" id="card_close"><svg viewBox="0 0 100 100"><path d="M2 2 L98 98 M 98 2 L2 98Z" stroke-width="10px" stroke="#969696" stroke-linecap="round"></path></svg></div><div class="title"><div class="title_text">        TTS</div><div class="TTS_Change"><div id="microsoft" tabindex="1" class="il">          微软</div><div id="baidu" tabindex="2" class="il" style="border-bottom: 2px solid #F00;">          百度</div><div id="about" tabindex="3" class="il">          关于</div></div></div><div class="card"><div class="setting" id="microsoft_card" style="display: none;"><div class="row"><div class="col"><div class="span" aria-label="zh-CN-XiaoxiaoNeural">              晓晓</div><div class="span" aria-label="zh-CN-XiaochenNeural">              晓辰</div><div class="span" aria-label="zh-CN-YunxiNeural">              云希</div><div class="span" aria-label="zh-CN-YunyangNeural">              云扬</div><div class="span" aria-label="zh-CN-XiaohanNeural">              晓涵</div><div class="span" aria-label="zh-CN-XiaoqiuNeural">              晓秋</div><div class="span" aria-label="zh-CN-XiaoxuanNeural">              晓萱</div><div class="span" aria-label="zh-CN-XiaoyanNeural">              晓颜</div><div class="span" aria-label="zh-CN-XiaoyouNeural">              晓悠</div><div class="span" aria-label="zh-CN-XiaoshuangNeural">              晓双</div><div class="span" aria-label="zh-CN-YunyeNeural">              云野</div></div></div><div class="setting_down"><div class="setting_sp"><div class="speech_set"><label for="speed" id="microsoft_speedlabel">语速: 1.00</label><div class="slider"><input type="range" id="microsoft_speed" name="speed" min="-100" max="200" value="0" class="slider__input" data-bi-id="demo-rate-slider" aria-label="语速"></div></div><div class="speech_set"><label for="pitch" id="microsoft_pitchlabel">音调: 0.00</label><div class="slider"><input type="range" id="microsoft_pitch" name="pitch" min="-50" max="50" value="0" class="slider__input" data-bi-id="demo-pitch-slider" aria-label="音调"></div></div></div><div class="others"><div style="margin-right:80px;width:100%;display: flex;flex-wrap: wrap-reverse;"><button class="more" type="button">更多</button></div><button class="listen" type="button">试听</button></div></div></div><div class="setting" id="baidu_card" style="display: show;"><div class="row"><div class="col"><div class="span" aria-label="4003">              度逍遥</div><div class="span" aria-label="4115">              度小贤</div><div class="span" aria-label="4119">              度小鹿</div><div class="span" aria-label="4100">              度小雯</div><div class="span" aria-label="4106">              度博文</div><div class="span" aria-label="4103">              度米朵</div></div></div><div class="setting_down"><div class="setting_sp"><div class="speech_set"><label for="speed" id="baidu_speedlabel">语速: 5</label><div class="slider"><input type="range" id="baidu_speed" name="speed" min="0" max="15" value="5" class="slider__input" data-bi-id="demo-rate-slider" aria-label="语速"></div></div><div class="speech_set"><label for="pitch" id="baidu_pitchlabel">音调: 5</label><div class="slider"><input type="range" id="baidu_pitch" name="pitch" min="0" max="15" value="5" class="slider__input" data-bi-id="demo-pitch-slider" aria-label="音调"></div></div></div><div class="others"><div style="margin-right:80px;width:100%;display: flex;flex-wrap: wrap-reverse;"><button class="more" type="button">更多</button></div><button class="listen" type="button">试听</button></div></div></div></div><div class="setting" id="about_card" style="display: none;"><div style="font-family: 'Lucida Console', 'Courier New', monospace;">        本项目由GAEE维护支持<a href="https://greasyfork.org/zh-CN/scripts/429810-%E5%A4%A7%E5%A3%B0%E6%9C%97%E8%AF%BB" target="_parent">首页</a></div><div><img style="height:auto;width:31%;" src="https://img.github.luxe/2022/d1807e2d06008.png" alt="90kskmwly1smny0t4dz6vh750k8n.png" title="支持一下" /></div></div></div>`;
    TTS_GLOBAL.find("body").append(CARD);
    UIinit();
    rebind();
}

function Show_more_card(){
    if(!document.querySelector("#GAEE_TTS_MORE")){
        var a = document.createElement('div');
        a.setAttribute("id","GAEE_TTS_MORE");
        document.body.appendChild(a);
        var b = document.createElement('iframe');
        b.setAttribute("id","GAEE_TTS_MORE_IFRAME");
        b.setAttribute("title","GAEE_TTS_MORE_IFRAME");
        b.style.cssText = "position: fixed;display: block;border-radius: 8px;border: unset; scrolling:no;left: 50%;top: 50%;z-index: 10001;-webkit-transform: translate(-50%, -50%);-ms-transform: translate(-50%, -50%);-o-transform: translate(-50%, -50%);-moz-transform: translate(-50%, -50%);transform: translate(-50%, -50%);background-color: #fff;";
        a.appendChild(b);
        let CARD = `<div class="card"><div class="title"><div class="title_text">TTS 高级设置</div><div class="close" id="card_close"><svg viewBox="0 0 100 100"><path d="M2 2 L98 98 M 98 2 L2 98Z" stroke-width="10px" stroke="#969696" stroke-linecap="round"></path></svg></div></div><div class="line"><div class="info" style="float:left; margin-right:20px">从选中位置开始朗读</div><div class="click" style="float:right;"><input type="checkbox" name="checkbox" id="c1" class="chooseBtn" /><label for="c1" class="choose-label"></label></div></div></div>`;
        $($("#GAEE_TTS_MORE_IFRAME")[0].contentWindow.document).find("body").append(CARD);
    }
    var _TTS_ = document.querySelector("#GAEE_TTS_MORE_IFRAME");
    TTS_MORE_GLOBAL = $($("#GAEE_TTS_MORE_IFRAME")[0].contentWindow.document);
    add_TTS_Style(_TTS_.contentWindow.document,'.card {user-select: none;}.chooseBtn {display: none;}.choose-label {box-shadow: #ccc 0px 0px 0px 1px;width: 40px;height: 20px;display: inline-block;border-radius: 20px;position: relative;background-color: #bdbdbd;overflow: hidden;}.choose-label:before {content: "";position: absolute;left: 0;width: 20px;height: 20px;display: inline-block;border-radius: 20px;background-color: #fff;z-index: 20;-webkit-transition: all 0.5s;transition: all 0.5s;}.chooseBtn:checked+label.choose-label:before {left: 20px;}.chooseBtn:checked+label.choose-label {background-color: #51ccee;}.close {position: absolute;top: 14px;right: 14px;width: 14px;height: 14px;cursor: pointer;}.title {margin-bottom: 16px;margin-left: 2px;line-height: 22px;display: flex;}.title .title_text {spanor: black;font-size: 16px;margin-right: 30px;}');
    UIinit2();
    rebind2();
}

function Get_InnerText(){
    var text = document.body.innerText;
    text = escapeHtml(text);
    text = cleanHtml(text);
    text = text.replace(/<[^<>]+>/g,"");
    //console.log(text);
    return text;
}
function escapeHtml(String) {
    return String.replace(/&quot;/g, '"').replace(/&#039;/g, "'").replace(/&amp;/g, '&').replace(/&lt;/g, '<').replace(/&gt;/g, '>');
}
function cleanHtml(String){
    let s_a = /<script[^<>]*>/g;
    let s_b = /<\/script>/g;
    var text = "";
    text = String.split(s_a)[0];
    for (var i = 1; i < String.split(s_a).length; i++) {
        var a = String.split(s_a)[i];a=a.split(s_b)[1];
        text+=a;
    }
    //console.log(text);
    return text;
}

function init(){
    if(!document.querySelector("#GAEE_TTS")){
        var audio = document.createElement('audio');
        audio.setAttribute("id","GAEE_TTS");
        //audio.setAttribute("controls","controls");
        audio.setAttribute("hidden","true");
        document.body.appendChild(audio);
        audio.addEventListener('ended', function () {
            console.log("end");
            icon_change("end");
            BaiduTTS_info.status = false;
            if(Global_TEXT.length != 0){
                BaiduTTS(Global_TEXT);
            }
        }, false);
    }
}

function play(data){
    $(document).ready(function() {
        BaiduTTS_info.status = true;
        console.log("start");
        icon_change("play");
        toPlay(data);
        TTS_GLOBAL.find("#TTS_Button").click(function() {
            if(BaiduTTS_info.status && info.type == "baidu"){
                console.log("pause");
                icon_change("end");
                var a = document.getElementById('GAEE_TTS');
                a !== null && a.pause();
                a = null;
                BaiduTTS_info.status = false;
            }
        });
        TTS_GLOBAL.find("#TTS_Button").on("touchstart", function() {
            if(BaiduTTS_info.status && info.type == "baidu"){
                console.log("pause");
                icon_change("end");
                var a = document.getElementById('GAEE_TTS');
                a !== null && a.pause();
                a = null;
                BaiduTTS_info.status = false;
            }
        });
    });
}

function toPlay(data){
    var audioBlob = toBlob(data);
    var blobUrl = window.URL.createObjectURL(audioBlob);
    document.getElementById('GAEE_TTS').src = blobUrl;
    var audio = document.getElementById('GAEE_TTS');
    audio.play();
    if (audio.paused) {
        audio.paused=false;
        audio.play();
    }
}

function toBlob(dataurl) {
    var arr = dataurl.split(','),
        mime = arr[0].match(/:(.*?);/)[1],
        bstr = atob(arr[1]),
        n = bstr.length,
        u8arr = new Uint8Array(n);
    while (n--) {
        u8arr[n] = bstr.charCodeAt(n);
    }
    return new Blob([u8arr], {
        type: mime
    });
}

function icon_change(type){
    if(type=="play"){
        TTS_GLOBAL.find("#icon-start").hide();
        TTS_GLOBAL.find("#icon-playing").show();
    }else if(type=="end" || type=="init"){
        TTS_GLOBAL.find("#icon-start").show();
        TTS_GLOBAL.find("#icon-playing").hide();
    }
}

function BaiduTTS(TEXT){
    var str = TEXT;
    var slen = 150;
    var elen = 200;
    var len = elen;
    var _a = str.slice(slen,elen);
    if(_a.indexOf("。")!=-1){
        len = slen + _a.indexOf("。") + 1;
        Global_TEXT = str.slice(len,TEXT.length);
    }else{
        Global_TEXT = str.slice(len,TEXT.length);
    }
    var BaiduTTS_postdata = "type=tns&per="+BaiduTTS_info.person+"&spd="+BaiduTTS_info.speed+"&pit="+BaiduTTS_info.pitch+"&vol=5&aue=6&tex="+encodeURIComponent(str.slice(0,len));
    GM_xmlhttpRequest({
        url:"https://ai.baidu.com/aidemo",
        method :"POST",
        data:BaiduTTS_postdata,
        headers: {
            'Content-Type': 'application/x-www-form-urlencoded',
            'User-Agent': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/100.0.4896.60 Safari/537.36 Edg/100.0.1185.29',
            'Referer': 'https://ai.baidu.com/tech/speech/tts_online'
        },
        onload:function(r){
            //console.log(r.responseText);
            let result = JSON.parse(r.responseText);
            if(result.errno=="0"){
                play(result.data);
                BaiduTTS_info.status = false;
            }else if(result.errno=="110"){
                if(!BaiduTTS_info.status){
                    BaiduTTS_info.status = true;
                    BaiduTTS("当前文本中可能存在敏感内容,百度TTS已拒绝了该请求");
                }else{
                    BaiduTTS_info.status = false;
                }
            }
            else{
                console.log("请求失败:"+r.responseText);
                BaiduTTS_info.status = false;
            }
        }
    });
}

function AzureTTS(TEXT){
    M_TTS(TEXT,"97421c82cec5dff4eb742cc0246691d8820c81f04ab72b6edf4284924ef1a7a5");
    return;
    GM_xmlhttpRequest({
        url: "https://azure.microsoft.com/zh-cn/services/cognitive-services/text-to-speech/?Voice",
        method: "GET",
        headers: {
            'Content-Type': 'application/x-www-form-urlencoded',
            'User-Agent': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/100.0.4896.60 Safari/537.36 Edg/100.0.1185.29'
        },
        onload: function(r) {
            var a = r.responseText.split("global.instanceId = '")[1].split("';")[0];
            M_TTS(TEXT,a);
        }
    });
}

var a = null;
function M_TTS(TEXT, token) {
    var str = TEXT;
    var slen = 450;
    var elen = 500;
    var len = elen;
    var _a = str.slice(slen,elen);
    if(_a.indexOf("。")!=-1){
        len = slen + _a.indexOf("。") + 1;
        Global_TEXT = str.slice(len,TEXT.length);
    }else{
        Global_TEXT = str.slice(len,TEXT.length);
    }
    TEXT = str.slice(0,len);
    $(document).ready(function() {
        MicrosoftTTS_info.status = true;
        console.log("start");
        var s = window.SpeechSDK;
        function play(){
            var config = s.SpeechTranslationConfig.fromEndpoint(new URL('wss://eastus.api.speech.microsoft.com/cognitiveservices/websocket/v1?TrafficType=AzureDemo')),
                synthesizer,
                audioConfig;
            config.speechSynthesisOutputFormat = s.SpeechSynthesisOutputFormat.Audio24Khz96KBitRateMonoMp3;
            a = new s.SpeakerAudioDestination();
            icon_change("play");
            a.onAudioEnd = function() {
                console.log("end");
                icon_change("end");
                MicrosoftTTS_info.status = false;
                if(Global_TEXT.length != 0){
                    AzureTTS(Global_TEXT);
                }
            };
            audioConfig = s.AudioConfig.fromSpeakerOutput(a);
            synthesizer = new s.SpeechSynthesizer(config, audioConfig);
            synthesizer.synthesisCompleted = function () {
                synthesizer.close();
                synthesizer = null;
            };
            synthesizer.SynthesisCanceled = function (s, e) {
                var r;
                r = s.CancellationDetails.fromResult(e);
                r.reason === s.CancellationReason.Error;
            };
            synthesizer.speakSsmlAsync(TextFormat(TEXT), function() {}, function(n) {
                console.log("Error:" + n);
                MicrosoftTTS_info.status = false;
            });
        }
        play();
        TTS_GLOBAL.find("#TTS_Button").click(function() {
            if(MicrosoftTTS_info.status && info.type == "microsoft"){
                console.log("pause");
                icon_change("end");
                a !== null && a.pause();
                a = null;
                MicrosoftTTS_info.status = false;
            }
        });
        TTS_GLOBAL.find("#TTS_Button").on("touchstart", function() {
            if(MicrosoftTTS_info.status && info.type == "microsoft"){
                console.log("pause");
                icon_change("end");
                a !== null && a.pause();
                a = null;
                MicrosoftTTS_info.status = false;
            }
        });
        function TextFormat(TEXT) {
            // var n = '<prosody rate="{SPEED}" pitch="{PITCH}">{TEXT}<\/prosody>',
            //   t;
            // n = n.replace("{SPEED}", "0" + "%");
            // n = n.replace("{PITCH}", "0" + "%");
            // u.selectedIndex === 0 && (p.hidden || o.selectedIndex === 0) || (n = "<mstts:express-as {STYLE_ATTRIBUTE} {ROLE_PLAY_ATTRIBUTE}>{0}<\/mstts:express-as>".replace("{0}", n), n = u.selectedIndex !== 0 ? n.replace("{STYLE_ATTRIBUTE}",
            //   'style="' + u[u.selectedIndex].value + '"') : n.replace("{STYLE_ATTRIBUTE}", ""), n = p.hidden || o.selectedIndex === 0 ? n.replace("{ROLE_PLAY_ATTRIBUTE}", "") : n.replace("{ROLE_PLAY_ATTRIBUTE}", 'role="' + o[o
            //   .selectedIndex].value + '"'));
            // n = '<voice name="{VOICE}">{0}<\/voice>'.replace("{0}", n);
            // n = n.replace("{VOICE}", i[Config.selectedIndex].value);
            // t = d.value.replace(/&/g, "&amp;").replace(/</g, "&lt;").replace(/>/g, "&gt;").replace(/"/g, "&quot;").replace(/'/g, "&apos;");
            // k.hidden || (t = '<lang xml:lang="{SECONDARY_LOCALE}">{0}<\/lang>'.replace("{0}", t), t = t.replace("{SECONDARY_LOCALE}", c[c.selectedIndex].value));
            // n = '<speak xmlns="http://www.w3.org/2001/10/synthesis" xmlns:mstts="http://www.w3.org/2001/mstts" xmlns:emo="http://www.w3.org/2009/10/emotionml" version="1.0" xml:lang="en-US">{0}<\/speak>'.replace("{0}", n);
            // n = n.replace("{TEXT}", t);
            var n =
                `<speak xmlns="http://www.w3.org/2001/10/synthesis" xmlns:mstts="http://www.w3.org/2001/mstts" xmlns:emo="http://www.w3.org/2009/10/emotionml" version="1.0" xml:lang="en-US"><voice name="{|_PERSON_|}"><prosody rate="{|_SPEED_|}%" pitch="{|_PITCH_|}%">{|_TEXT_|}</prosody></voice></speak>`;
            n = n.replace("{|_PERSON_|}", MicrosoftTTS_info.person);
            n = n.replace("{|_SPEED_|}", MicrosoftTTS_info.speed*100-100);
            n = n.replace("{|_PITCH_|}", MicrosoftTTS_info.pitch*50-50);
            n = n.replace("{|_TEXT_|}", TEXT);
            return n;
        }
    });
}

function TryListen(){
    init();
    let TEXT = "测试成功! 当前版本为 "+version+"。很高兴与你相遇!";
    if (info.type == "baidu") {
        BaiduTTS(TEXT);
    } else if (info.type == "microsoft") {
        AzureTTS(TEXT);
    }
}

function init_voice_setting(){
    var Info_ALL = [MicrosoftTTS_info, BaiduTTS_info, info, setting];
    if(GM_getValue("GAEE_TTS_voice_info") == null || GM_getValue("GAEE_TTS_voice_info")[3] == null){
        GM_setValue("GAEE_TTS_voice_info", Info_ALL);
    }else{
        Info_ALL = GM_getValue("GAEE_TTS_voice_info");
        MicrosoftTTS_info = Info_ALL[0];
        BaiduTTS_info = Info_ALL[1];
        info = Info_ALL[2];
        setting = Info_ALL[3];
    }
    MicrosoftTTS_info.status = false;
    BaiduTTS_info.status = false;
}

function voice_setting(){
    var Info_ALL = [MicrosoftTTS_info, BaiduTTS_info, info, setting];
    GM_setValue("GAEE_TTS_voice_info", Info_ALL);
}

registerMenuCommand();
function registerMenuCommand() {
    GM_registerMenuCommand(`🏁 \u7248\u672c\u4fe1\u606f ${version}`, function () {window.GM_openInTab('https://greasyfork.org/zh-CN/scripts/429810', {active: true,insert: true,setParent: true});});
    //menu_ID[menu_ID.length] = GM_registerMenuCommand('💬 \u53cd\u9988 & \u5efa\u8bae', function () {window.GM_openInTab('', {active: true,insert: true,setParent: true});});
}