- // ==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});});
- }