// ==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(/"/g, '"').replace(/'/g, "'").replace(/&/g, '&').replace(/</g, '<').replace(/>/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, "&").replace(/</g, "<").replace(/>/g, ">").replace(/"/g, """).replace(/'/g, "'");
// 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});});
}