您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
自动获取网页上的小说内容并转换为语音
// ==UserScript== // @name 笔趣阁小说朗读 // @namespace http://tampermonkey.net/ // @version 2.3.18 // @description 自动获取网页上的小说内容并转换为语音 // @author Xie // @match *://**/*.html // @license 使用说明:请看下方描述 // @icon  // @grant none // ==/UserScript== (function() { 'use strict'; if(!('speechSynthesis' in window)) { throw alert("对不起,您的浏览器不支持") } let url = window.location.href; // 百度贴吧能匹配成功下面的判断,这里直接拦截 if (url.indexOf('baidu') >= 0) { return; } let allText = document.body.innerText; const allTextKeyWords = ['上一章', '上一页', '下一章', '下一页', '目录', '章节', '小说', '网站即将关闭', '阅读最新内容', '为你提供最快', '为您提供最快', '请收藏本站', '最新章节', '一键直达', '最快更新', '记住本书', '首发域名', '首发网址', '首发地址', '免费阅读', '笔趣阁', '小说网']; if (!allTextKeyWords.some(keyword => allText.includes(keyword))) { // 如果整个页面都不含如上任何元素,则不使用当前插件 console.log('其他网站') return; } // 需要删除的页面标签 var removeEle = ['.lm','.erwm', '.read-titlelinke', '.tjlist', 'center', '.readinline', '.sectionTwo', '.lrghqazd', '.fdpsxavq', '#commendbook', '.readon' ,'#tyJ', '#pAV', '.nr_page', 'iframe']; // 模糊匹配删除:class\id\标签名称 var removeEleFuzzyMatching = ['nav', 'top', 'foot', 'link', 'header', 'img', 'show']; // 根据关键词 过滤 小说内容中的行 let filterKeywords = [ 'https', 'https', 'xswang.la', 'https', 'http', 'www', '.com', 'CòΜ', 'net', '网站即将关闭', '阅读最新内容', '为你提供最快', '为您提供最快', '请收藏本站', '最新章节', '一键直达', '最快更新', '章节错误', '记住本书', '首发域名', '首发网址', '首发地址', '本站地址', '本站网址', '免费阅读', '笔趣阁', '无广告', '小说网', '手机版', '请下载' ]; // 将当前网站的域名添加到过滤名单 filterKeywords.push(window.location.hostname); // 章节名称标签 var titleEle = null; // 小说内容标签 var contentEle = null; // 上一页、上一章 var leftButton = null; // 下一页、下一章 var rightButton = null; let punctuationMarks = ['。', '?', '!', ';', '?', '!', ';']; setTimeout(function() { // 删除无用元素 elementRemove(); // 自动识别标题、内容、上一章、下一章等标签 let flag = autoIdentify(); if(!flag) { return; } // 创建一个新的div元素 let floatingDiv = document.createElement("div"); floatingDiv.innerHTML = ` <div class="xie-floating-div"> <div class="xie-menu"> <div title="展开"> <svg t="1698304201100" class="icon xie-svg-zk" viewBox="0 0 1024 1024" version="1.1" xmlns="http://www.w3.org/2000/svg" p-id="1496"><path d="M512 512m-512 0a512 512 0 1 0 1024 0 512 512 0 1 0-1024 0Z" fill="#D0DDFE" p-id="1497"></path><path d="M512 592m-256 0a256 256 0 1 0 512 0 256 256 0 1 0-512 0Z" fill="#FFFFFF" p-id="1498"></path><path d="M512 384a208 208 0 0 0-147.072 355.072 208 208 0 0 0 294.144-294.144 206.64 206.64 0 0 0-147.072-60.928m0-48a256 256 0 1 1-256 256 256 256 0 0 1 256-256z" fill="#FFFFFF" p-id="1499"></path><path d="M225.136 590.688m59.184 0l-0.016 0q59.184 0 59.184 59.184l0 54.624q0 59.184-59.184 59.184l0.016 0q-59.184 0-59.184-59.184l0-54.624q0-59.184 59.184-59.184Z" fill="#8BACFF" p-id="1500"></path><path d="M792.064 671.904a292.8 292.8 0 0 0-70-355.2l16.176-31.472q8.512 7.456 16.592 15.52a337.392 337.392 0 0 1 71.36 371.2z m-594.256 0A337.392 337.392 0 0 1 269.104 300.752a342.224 342.224 0 0 1 32.16-28.304L320.976 301.616a292.928 292.928 0 0 0-89.024 370.288z" fill="#3075FF" p-id="1501"></path><path d="M680 590.688m59.184 0l-0.016 0q59.184 0 59.184 59.184l0 54.624q0 59.184-59.184 59.184l0.016 0q-59.184 0-59.184-59.184l0-54.624q0-59.184 59.184-59.184Z" fill="#8BACFF" p-id="1502"></path><path d="M197.824 563.376m59.184 0l-0.016 0q59.184 0 59.184 59.184l0 109.248q0 59.184-59.184 59.184l0.016 0q-59.184 0-59.184-59.184l0-109.248q0-59.184 59.184-59.184Z" fill="#216CFF" p-id="1503"></path><path d="M707.2 563.376m59.184 0l-0.016 0q59.184 0 59.184 59.184l0 109.248q0 59.184-59.184 59.184l0.016 0q-59.184 0-59.184-59.184l0-109.248q0-59.184 59.184-59.184Z" fill="#216CFF" p-id="1504"></path><path d="M727.984 331.136l-0.128 0.128a316.8 316.8 0 0 0-431.776-2.672L295.984 328.496a32 32 0 1 1-43.2-47.088 381.056 381.056 0 0 1 518.592 2.672 32 32 0 1 1-43.2 47.088z" fill="#216CFF" p-id="1505"></path><path d="M524 464m28 0l0 0q28 0 28 28l0 248q0 28-28 28l0 0q-28 0-28-28l0-248q0-28 28-28Z" fill="#FFB444" p-id="1506"></path><path d="M604 592m28 0l0 0q28 0 28 28l0 104q0 28-28 28l0 0q-28 0-28-28l0-104q0-28 28-28Z" fill="#FFB444" p-id="1507"></path><path d="M444 544m28 0l0 0q28 0 28 28l0 152q0 28-28 28l0 0q-28 0-28-28l0-152q0-28 28-28Z" fill="#FFB444" p-id="1508"></path><path d="M364 640m28 0l0 0q28 0 28 28l0 40q0 28-28 28l0 0q-28 0-28-28l0-40q0-28 28-28Z" fill="#FFB444" p-id="1509"></path></svg> </div> <div title="收缩"> <svg t="1698286098837" class="icon xie-svg-ss" viewBox="0 0 1024 1024" version="1.1" xmlns="http://www.w3.org/2000/svg" p-id="1815" width="200" height="200"><path d="M333.1 126.5l-0.7 0.7c-12.3 12.3-12.3 32.4 0 44.7l339.9 339.9-340.1 340.1c-12.5 12.5-12.5 32.9 0 45.4s32.9 12.5 45.4 0L740 535s0.1-0.1 0.2-0.1l0.7-0.7c12.3-12.3 12.3-32.4 0-44.7l-363-363c-12.4-12.3-32.5-12.3-44.8 0z" fill="#4D4D4D" p-id="1816"></path></svg> </div> </div> <div class="xie-body"> <div class="xie-form"> <div class="xie-row"> <label class="xie-col-label" for="voiceSelect">声音:</label> <select id="voiceSelect" name="voiceSelect" style="width: 150px"></select> </div> <div class="xie-row"> <label class="xie-col-label" for="volumeInput">音量:</label> <input type="range" id="volumeInput" name="volumeInput" min="0.1" max="1" step="0.1" value="1"> <div class="xie-col-value" id="volumeValue">1</div> </div> <div class="xie-row"> <label class="xie-col-label" for="rateInput">语速:</label> <input type="range" id="rateInput" name="rateInput" min="0.1" max="10" step="0.1" value="1"> <div class="xie-col-value" id="rateValue">1</div> </div> <div class="xie-row"> <label class="xie-col-label" for="pitchInput">音高:</label> <input type="range" id="pitchInput" name="pitchInput" min="0" max="2" step="0.1" value="1"> <div class="xie-col-value" id="pitchValue">1</div> </div> <div class="xie-row"> <label class="xie-col-label">字幕:</label> <input type="radio" id="captions_1" name="captions" value="1" style="margin: 0px 10px;"> <label for="captions_1">打开</label> <input type="radio" id="captions_0" name="captions" value="0" style="margin: 0px 10px;"> <label for="captions_0">关闭</label> </div> <div class="xie-row"> <label class="xie-col-label" title="根据关键词净化小说内容,删除含网址地址的行">净化:</label> <input type="radio" id="purification_1" name="purification" value="1" style="margin: 0px 10px;"> <label for="purification_1">打开</label> <input type="radio" id="purification_0" name="purification" value="0" style="margin: 0px 10px;"> <label for="purification_0">关闭</label> </div> <div class="xie-row"> <label class="xie-col-label">隐身:</label> <input type="radio" id="stealth_0" name="stealth" value="0" style="margin: 0px 10px;"> <label for="stealth_0">隐藏</label> <input type="radio" id="stealth_1" name="stealth" value="1" style="margin: 0px 10px;"> <label for="stealth_1">显示</label> </div> <div class="xie-row"> <label class="xie-col-label">续播:</label> <input type="radio" id="autoplay_1" name="autoplay" value="1" style="margin: 0px 10px;"> <label for="autoplay_1">翻页自动播放</label> <input type="radio" id="autoplay_0" name="autoplay" value="0" style="margin: 0px 10px;"> <label for="autoplay_0">手动播放</label> </div> <div class="xie-row" style="flex-direction: column;font-size: 12px;"> <div class="xie-remake"> <div class="xie-remake-title">快捷键(非全局):</div> <div class="xie-remake-body"> <div>Ctrl + 方向左键:上一页、上一章</div> <div>Ctrl + 方向右键:下一页、下一章</div> <div>Ctrl + 方向下键:播放、暂停</div> </div> </div> </div> </div> <div class="xie-body-button"> <div title="播放"> <svg t="1698285826519" class="icon xie-svg-bf" viewBox="0 0 1024 1024" version="1.1" xmlns="http://www.w3.org/2000/svg" p-id="1349" ><path d="M772.7 217.7a32.2 32.1 0 1 0 64.4 0 32.2 32.1 0 1 0-64.4 0Z" fill="#4D4D4D" p-id="1350"></path><path d="M415.8 679.9c5.9 0 11.5-1.6 16.2-4.5l231.1-134.6c10.9-5.2 18.5-16.3 18.5-29.2 0-11.9-6.4-22.3-16-27.8L439.7 352.2c-5.8-6.7-14.4-10.9-23.9-10.9-17.6 0-31.8 14.4-31.8 32.1 0 0.6 0 1.2 0.1 1.8l-0.4 0.2 0.5 269c-0.1 1.1-0.2 2.2-0.2 3.4 0 17.7 14.3 32.1 31.8 32.1z" fill="#4D4D4D" p-id="1351"></path><path d="M909.8 306.6c-5.4-10.5-16.3-17.8-28.9-17.8-17.8 0-32.2 14.4-32.2 32.1 0 6 1.7 11.7 4.6 16.5l-0.1 0.1c26.9 52.4 42.1 111.8 42.1 174.7 0 211.6-171.6 383.2-383.2 383.2S128.8 723.8 128.8 512.2 300.4 129.1 512 129.1c62.5 0 121.5 15 173.6 41.5l0.2-0.4c4.6 2.6 10 4.1 15.7 4.1 17.8 0 32.2-14.4 32.2-32.1 0-13.1-7.9-24.4-19.3-29.4C653.6 81.9 584.9 64.5 512 64.5 264.7 64.5 64.3 265 64.3 512.2S264.7 959.9 512 959.9s447.7-200.4 447.7-447.7c0-74.1-18-144-49.9-205.6z" fill="#4D4D4D" p-id="1352"></path></svg> </div> <div title="暂停"> <svg t="1698285952364" class="icon xie-svg-zt" viewBox="0 0 1024 1024" version="1.1" xmlns="http://www.w3.org/2000/svg" p-id="1507" ><path d="M910.8 303.6c-5.4-10.5-16.3-17.8-28.9-17.8-17.8 0-32.2 14.4-32.2 32.1 0 6 1.7 11.7 4.6 16.5l-0.1 0.1c26.9 52.4 42.1 111.8 42.1 174.7 0 211.6-171.6 383.2-383.2 383.2S129.8 720.8 129.8 509.2 301.4 126.1 513 126.1c62.5 0 121.5 15 173.6 41.5l0.2-0.4c4.6 2.6 10 4.1 15.7 4.1 17.8 0 32.2-14.4 32.2-32.1 0-13.1-7.9-24.4-19.3-29.4C654.6 78.9 585.9 61.5 513 61.5 265.7 61.5 65.3 262 65.3 509.2S265.7 956.9 513 956.9s447.7-200.4 447.7-447.7c0-74.1-18-144-49.9-205.6z" fill="#515151" p-id="1508"></path><path d="M385.4 352.2V672c0 17.5 14.3 31.9 31.9 31.9 17.6 0 32-14.4 31.9-31.9V352.2c0-17.5-14.3-31.9-31.9-31.9-17.5 0-31.9 14.3-31.9 31.9zM578.9 352.2V672c0 17.5 14.3 31.9 31.9 31.9 17.5 0 31.9-14.4 31.9-31.9V352.2c0-17.5-14.3-31.9-31.9-31.9-17.5 0-31.9 14.3-31.9 31.9z" fill="#515151" p-id="1509"></path><path d="M772.7 217.7a32.2 32.1 0 1 0 64.4 0 32.2 32.1 0 1 0-64.4 0Z" fill="#515151" p-id="1510"></path></svg> </div> <div title="清空播放列表"> <svg t="1698289630950" class="icon xie-svg-qk" viewBox="0 0 1024 1024" version="1.1" xmlns="http://www.w3.org/2000/svg" p-id="2268" width="200" height="200"><path d="M703.355701 958.712041 297.259706 958.712041c-97.55792 0-131.029242-18.10945-131.029242-130.289392L166.230464 390.164141c0-14.950504 12.122085-27.073612 27.073612-27.073612 14.950504 0 27.073612 12.122085 27.073612 27.073612l0 438.258509c0 71.145363-1.797949 76.144214 76.883041 76.144214l406.095995 0c76.05007 0 85.554538-2.142803 85.554538-76.144214L788.911262 390.164141c0-14.950504 12.122085-27.073612 27.073612-27.073612 14.950504 0 27.072589 12.122085 27.072589 27.073612l0 438.258509C843.05644 931.904489 808.528042 958.712041 703.355701 958.712041z" fill="#4d4d4d" p-id="2269"></path><path d="M694.155155 241.261832c-14.950504 0-27.072589-12.122085-27.072589-27.073612l0-8.460696c0-78.231759-14.620999-86.295412-85.554538-86.295412L419.088402 119.432113c-73.961502 0-76.883041 10.482749-76.883041 86.295412l0 8.460696c0 14.950504-12.122085 27.073612-27.073612 27.073612-14.951527 0-27.073612-12.122085-27.073612-27.073612l0-8.460696c0-105.000426 27.601638-140.441613 131.029242-140.441613l162.438603 0c112.575961 0 139.701762 45.289486 139.701762 140.441613l0 8.460696C721.228767 229.139747 709.106682 241.261832 694.155155 241.261832z" fill="#4d4d4d" p-id="2270"></path><path d="M342.205361 823.346027c-7.482415 0-13.537318-6.079462-13.537318-13.537318l0-27.071565c0-7.482415 6.054903-13.537318 13.537318-13.537318 7.482415 0 13.536294 6.053879 13.536294 13.537318l0 27.071565C355.741655 817.266565 349.686752 823.346027 342.205361 823.346027z" fill="#4d4d4d" p-id="2271"></path><path d="M342.205361 728.58992c-7.482415 0-13.537318-6.079462-13.537318-13.537318L328.668043 484.920248c0-7.475252 6.054903-13.536294 13.537318-13.536294 7.482415 0 13.536294 6.061043 13.536294 13.536294l0 230.133378C355.741655 722.510458 349.686752 728.58992 342.205361 728.58992z" fill="#4d4d4d" p-id="2272"></path><path d="M504.64294 823.346027c-7.482415 0-13.536294-6.079462-13.536294-13.537318l0-324.889485c0-7.475252 6.053879-13.536294 13.536294-13.536294s13.537318 6.061043 13.537318 13.536294l0 324.889485C518.180258 817.266565 512.125355 823.346027 504.64294 823.346027z" fill="#4d4d4d" p-id="2273"></path><path d="M667.082566 823.346027c-7.482415 0-13.537318-6.079462-13.537318-13.537318l0-324.889485c0-7.475252 6.054903-13.536294 13.537318-13.536294s13.536294 6.061043 13.536294 13.536294l0 324.889485C680.617837 817.266565 674.563958 823.346027 667.082566 823.346027z" fill="#4d4d4d" p-id="2274"></path><path d="M951.349865 305.560254c0 61.674665-49.994648 111.676475-111.676475 111.676475l-656.522558 0c-61.680804 0-111.676475-50.001811-111.676475-111.676475l0-6.768147c0-61.674665 49.994648-111.676475 111.676475-111.676475l656.522558 0c61.680804 0 111.676475 50.001811 111.676475 111.676475L951.349865 305.560254zM897.203664 298.79313c0-31.772634-25.750477-57.530274-57.530274-57.530274l-656.522558 0c-31.778774 0-57.530274 25.75764-57.530274 57.530274l0 6.768147c0 31.772634 25.7515 57.530274 57.530274 57.530274l656.522558 0c31.779797 0 57.530274-25.75764 57.530274-57.530274L897.203664 298.79313z" fill="#4d4d4d" p-id="2275"></path></svg> </div> </div> </div> </div> <div class="xie-bottom-div"> <div class="xie-bottom-content"></div> </div> <style> .xie-floating-div body, dd, div, dl, dt, fieldset, form, h1, h2, h3, h4, h5, h6, html, img, input, li, ol, p, select, table, td, th, ul { margin: 0; padding: 0; } .xie-floating-div { position: fixed; top: 150px; right: 20px; display: flex; background-color: #fff; border: 1px solid #efefef; border-radius: 8px; font-size: 14px; cursor: default; } .xie-bottom-div { position: fixed; bottom: 15px; left: 0; display: flex; width: 100%; align-items: center; align-content: center; justify-content: center; } .xie-bottom-content { min-height: 50px; background-color: #fff; border: 1px solid #efefef; border-radius: 6px; width: 70%; padding: 10px 15px; display: flex; text-align: center; align-content: center; align-items: center; justify-content: center; color: #67C23A; font-size: 20px; font-weight: bold; line-height: 28px; transition: opacity 1s; /* 添加渐变效果 */ } .xie-menu { display: flex; justify-content: center; align-items: center; padding: 5px; } .xie-body { display: none; flex-flow: column; justify-content: center; align-items: center; /*border-left: 1px solid #efefef;*/ transition: opacity 1s; /* 添加渐变效果 */ } .xie-form { padding: 10px 15px; } .xie-row { display: flex; align-items: center; margin: 5px 0px; } .xie-remake { border: 1px solid #efefef; width: 100%; margin-bottom: 5px; } .xie-remake-title { background-color: #aaa; color: #fff; border-bottom: 1px solid #efefef; padding: 5px; } .xie-remake-body{ padding: 5px; width: 240px; } .xie-remake-body-a { height: 22px; display: flex; align-items: center; padding: 0 5px; text-decoration: underline; } .xie-col-label { font-weight: bold; } .xie-col-value{ width: 30px; text-align: center; } .xie-body-button{ display: flex; justify-content: center; align-items: center; /*border-top: 1px solid #efefef;*/ width: 100%; padding: 0; margin-bottom: 20px; } .xie-svg-zk, .xie-svg-ss { width: 30px; height: 30px; } .xie-svg-bf, .xie-svg-zt, .xie-svg-qk { width: 25px; height: 25px; margin: 0px 10px; } .xie-svg-zk, .xie-svg-zt { display: none; } .xie-svg-ss { display: none; } .xie-svg-ss>path { fill: #303133; } .xie-svg-bf>path { fill: #409EFF; } .xie-svg-zt>path { fill: #E6A23C; } .xie-svg-qk>path { fill: #F56C6C; } .xie-svg-bf:hover { filter: brightness(1.5); /* 改变亮度以更改整个SVG的颜色 */ transition: filter 0.3s; /* 添加过渡效果 */ } </style> `; document.body.appendChild(floatingDiv); const pageTitle = document.title; const voiceSelect = document.getElementById("voiceSelect"); const volumeInput = document.getElementById("volumeInput"); const volumeValue = document.getElementById("volumeValue"); const rateInput = document.getElementById("rateInput"); const rateValue = document.getElementById("rateValue"); const pitchInput = document.getElementById("pitchInput"); const pitchValue = document.getElementById("pitchValue"); const captionsButtons = document.querySelectorAll('input[name="captions"]'); const purificationButtons = document.querySelectorAll('input[name="purification"]'); const stealthButtons = document.querySelectorAll('input[name="stealth"]'); const autoplayButtons = document.querySelectorAll('input[name="autoplay"]'); const xieFloatingDiv = document.querySelector(".xie-floating-div"); const xieBody = document.querySelector(".xie-body"); const bottomContent = document.querySelector(".xie-bottom-content"); const xieSvgZk = document.querySelector(".xie-svg-zk"); const xieSvgSs = document.querySelector(".xie-svg-ss"); const xieSvgBf = document.querySelector(".xie-svg-bf"); const xieSvgZt = document.querySelector(".xie-svg-zt"); const xieSvgQk = document.querySelector(".xie-svg-qk"); var autoplayV = '0'; clickSs(); initInput(); clickQk(); function autoIdentify() { // 需要排除的标签,这些标签直接跳过判断 const excludeEleKeyWords = ['script', 'style', 'font', 'symbol', 'svg', 'img', 'select', 'input', 'option']; // 标题常规标签 const titleEleKeyWords = ['h1', 'h2', 'h3', 'h4', 'h5', 'h6', 'header']; // 标题关键词 const titleKeyWords = ['第', '章', '节', '篇', '卷', '页']; // 内容中允许保留的子元素标签,其它标签都需要删除,然后再获取内容文本进行识别 const contentReserveWords = ['p', 'span', 'br', 'pre', 'b', 'strong', 'h1', 'h2', 'h3', 'h4', 'h5', 'h6', 'em', 'i', 'u']; // 从特定标签中找到的标题 let titleEles = [null, null, null]; // 上次识别到的小说内容中文长度,用于与本次长度做对比,保留最长的内容 let len_ch = 0; // 记录已经打印过的标签,防止重复打印 let tags = []; const allElements = document.querySelectorAll('html > :not(head) *'); for (const element of allElements) { // 复制当前元素以避免直接修改原始元素 const clonedElement = element.cloneNode(true); let tagName = clonedElement.tagName.toLowerCase(); let str = getText(clonedElement.textContent); if (str === '' || excludeEleKeyWords.includes(tagName)) { continue; } // 打印出每种标签名称 if (!tags.some(keyword => tagName.includes(keyword))) { console.log(tagName, element); tags.push(tagName); } try { if (tagName !== 'a') { // 如果没有从特定标签取到标题,则持续从所有标签中暂时获取 // 标题从三个方面分别取值, // 1:含关键词 是关键标签 无子元素 // 2:含关键词 是关键标签 有子元素 // 3:不含关键词 是关键标签 无子元素 // 含关键词 if (titleKeyWords.some(keyword => str.includes(keyword))) { // 是关键标签 if (titleEleKeyWords.includes(tagName)) { // 无子元素 if (titleEles[0] == null && clonedElement.children.length === 0) { titleEles[0] = element; } else if (titleEles[1] == null) { titleEles[1] = element; } } } else if (titleEles[2] == null && titleEleKeyWords.includes(tagName) && clonedElement.children.length === 0) { titleEles[2] = element; } } } catch (e) { console.log('标题获取出错'); console.log(e); console.log(clonedElement); } try { // 检查 div 元素是否包含文本内容 if (!clonedElement.className || !clonedElement.className.includes || !clonedElement.className.includes('xie-')) { // 遍历当前元素的所有直接子元素,删除非 特定元素 的元素 var childNodes = clonedElement.childNodes; for (var i = childNodes.length - 1; i >= 0; i--) { var childNode = childNodes[i]; if (childNode.nodeType === 1 && !contentReserveWords.includes(childNode.nodeName.toLowerCase())) { clonedElement.removeChild(childNode); } } str = getText(clonedElement.textContent); let chineseCharacters = str.match(/[\u4e00-\u9fa5]/g); let len_ch2 = chineseCharacters ? chineseCharacters.length : 0; if (len_ch2 > len_ch) { contentEle = element; len_ch = len_ch2; } } } catch (e) { console.log('内容获取出错'); console.log(e); console.log(clonedElement); } try { if (tagName === 'a') { if (str === '上一章' || str === '上一页') { leftButton = element; } else if (str === '下一章' || str === '下一页') { rightButton = element; } } } catch (e) { console.log('上一章/下一章获取出错'); console.log(e); console.log(clonedElement); } } titleEle = titleEles[0] || titleEles[1] || titleEles[2]; console.log('自动识别标题:', titleEle ? getText(titleEle.textContent) : null, titleEles); console.log('自动识别内容:', contentEle, contentEle ? getText(contentEle.textContent) : null); console.log('自动识别上一章:', leftButton); console.log('自动识别下一章:', rightButton); if (contentEle == null) { console.log('内容获取失败'); return false; } return true; } function getText(str) { try { return str.trim().replace(/[\r\n]+/g, '').toLowerCase(); } catch (e) { console.log(ele); console.log(e); } return ''; } // 小说章节页面净化 function bookContent(type) { if (type === 0) { return; } let ss = []; let stra = contentEle.innerText.split('\n'); // 前一行是否为空 let prevNull = false; for (let i = 0; i < stra.length; i++){ let s = filterStr(stra[i]); let isNull = !s || s==='' || s.trim() === ''; // 如果上一行为空,这一行还为空,跳过 if ((prevNull && isNull)) { continue; } prevNull = isNull; ss.push(s); } // console.log(ss); if (ss.length > 0) { contentEle.innerHTML = ss.join("<br/>") } } // 删除元素 function elementRemove() { // 获取所有的定时器ID let highestTimeoutId = setTimeout(() => {}); let highestIntervalId = setInterval(() => {}); // 清除所有 setTimeout for (let i = 0; i <= highestTimeoutId; i++) { clearTimeout(i); } // 清除所有 setInterval for (let i = 0; i <= highestIntervalId; i++) { clearInterval(i); } // 精确删除 removeEle.forEach(function(className) { let elements = document.querySelectorAll(className); elements.forEach(function(element) { element.remove(); }); }); // 模糊匹配删除 var allElements = document.querySelectorAll('html > :not(head) *'); allElements.forEach(function(element) { console.log(element.tagName, element.id, element.classList, element.style.backgroundImage) // 取消所有标签的背景图 element.style.setProperty('background-image', 'none', 'important'); // 检查标签的 class、id 和标签名称是否包含数组中的任何一个关键词 var shouldDelete = removeEleFuzzyMatching.some(function(keyword) { return Array.from(element.classList).join(' ').includes(keyword) || element.id.includes(keyword) || element.tagName.toLowerCase().includes(keyword); }); // 如果包含任何一个关键词,删除该标签 if (shouldDelete) { const tagName = element.tagName; const className = element.className; const id = element.id; console.log(`被删除的标签: 名称=${tagName}, class=${className}, id=${id}`); element.remove(); } }); document.body.style.paddingBottom = '20px'; } function bottomContentShowHidden(type) { showAndHidden(bottomContent, 'flex', type) } function showAndHidden(ele, showType, type) { if (type === 1) { ele.style.display = showType; ele.style.opacity = 1; xieFloatingDiv.style.opacity = 1; } else { // 将透明度设置为0来触发渐隐 ele.style.opacity = 0; xieFloatingDiv.style.opacity = 0.5; // 使用setTimeout在动画结束后将div隐藏 setTimeout(function() { ele.style.display = 'none'; }, 100); // 这里的1000表示1秒,与CSS中的过渡时间相匹配 } } // 显示或隐藏页面内容 function showAndHiddenPage(val) { if (titleEle != null) { titleEle.style.display = val == 1 ? 'block' : 'none'; } contentEle.style.display = val == 1 ? 'block' : 'none'; if (val == 1) { document.title = pageTitle; } else { document.title = '???'; } } function initInput() { setTimeout(() => { let vs = window.speechSynthesis.getVoices(); for (let i in vs) { let newOption = document.createElement("option"); newOption.text = vs[i].name; // 选项的显示文本 newOption.value = i; // 选项的值 // 将新的 option 元素添加到 select 元素中 voiceSelect.add(newOption); } let voiceV = getItem("xie-input-voice", 2); voiceSelect.selectedIndex = Number(voiceV); voiceSelect.addEventListener("change", function() { setItem("xie-input-voice", voiceSelect.value); }); }, 500); let captionsV = getItem("xie-input-captions", '0'); bottomContentShowHidden(Number(captionsV)) captionsButtons.forEach(function(radioButton) { if (radioButton.value === captionsV) { radioButton.checked = true; } radioButton.addEventListener("change", function() { if (radioButton.checked) { setItem("xie-input-captions", radioButton.value); bottomContentShowHidden(Number(radioButton.value)) } }); }); let purificationV = getItem("xie-input-purification", '0'); bookContent(Number(purificationV)); purificationButtons.forEach(function(radioButton) { if (radioButton.value === purificationV) { radioButton.checked = true; } radioButton.addEventListener("change", function() { if (radioButton.checked) { setItem("xie-input-purification", radioButton.value); bookContent(Number(radioButton.value)) } }); }); let stealthV = getItem("xie-input-stealth", '1'); showAndHiddenPage(stealthV); stealthButtons.forEach(function(radioButton) { if (radioButton.value === stealthV) { radioButton.checked = true; } radioButton.addEventListener("change", function() { if (radioButton.checked) { setItem("xie-input-stealth", radioButton.value); showAndHiddenPage(radioButton.value); } }); }); autoplayV = getItem("xie-input-autoplay", '0'); autoplayButtons.forEach(function(radioButton) { if (radioButton.value === autoplayV) { radioButton.checked = true; } radioButton.addEventListener("change", function() { if (radioButton.checked) { setItem("xie-input-autoplay", radioButton.value); } }); }); let volumeV = getItem("xie-input-volume", 1); volumeValue.textContent = volumeV; volumeInput.value = volumeV; volumeInput.addEventListener("input", function() { volumeValue.textContent = volumeInput.value; setItem("xie-input-volume", volumeInput.value); }); let rateV = getItem("xie-input-rate", 0.8); rateValue.textContent = rateV; rateInput.value = rateV; rateInput.addEventListener("input", function() { rateValue.textContent = rateInput.value; setItem("xie-input-rate", rateInput.value); }); let pitchV = getItem("xie-input-pitch", 1); pitchValue.textContent = pitchV; pitchInput.value = pitchV; pitchInput.addEventListener("input", function() { pitchValue.textContent = pitchInput.value; setItem("xie-input-pitch", pitchInput.value); }); xieSvgZk.addEventListener("click", () => { clickZk(); }); xieSvgSs.addEventListener("click", () => { clickSs(); }); xieSvgBf.addEventListener("click", () => { clickBf(); }); xieSvgZt.addEventListener("click", () => { clickZt() }); xieSvgQk.addEventListener("click", () => { clickQk(); }); document.addEventListener('keydown', function(event) { // 检查是否按下了Ctrl键 if (event.ctrlKey) { switch (event.key) { case 'ArrowUp': console.log('按下了Ctrl + 方向上键:'); break; case 'ArrowDown': console.log('按下了Ctrl + 方向下键:播放、暂停', xieSvgBf.style.display); xieSvgBf.style.display == 'none' ? clickZt() : clickBf(); break; case 'ArrowLeft': console.log('按下了Ctrl + 方向左键:上一页、上一章'); if (leftButton == null) { clickQk(); contents.push('未找到上一章元素,快捷键跳转失败!!!') mySpeechSynthesis(0); } else { leftButton.click(); } break; case 'ArrowRight': console.log('按下了Ctrl + 方向右键:下一页、下一章'); if (rightButton == null) { clickQk(); contents.push('未找到下一章元素,快捷键跳转失败!!!') mySpeechSynthesis(0); } else { rightButton.click(); } break; default: // 如果不是上下左右键,可以不处理或执行其他操作 break; } } }); try { // 向用户请求自动播放权限 navigator.mediaDevices.getUserMedia({ audio: true, video: false }) .then(function(stream) { // 用户已授权,可以播放音频或视频 setTimeout(() => { let autoplayV = getItem("xie-input-autoplay", '0'); if (autoplayV == 1) { console.log('开始自动播放'); // xieSvgBf.click(); var clickEvent = new Event('click', { bubbles: true, cancelable: true, }); xieSvgBf.dispatchEvent(clickEvent); } }, 1500); }) .catch(function(error) { console.error('获取用户媒体权限失败:', error); }); } catch (e) { // 失败了也尝试自动播放 setTimeout(() => { let autoplayV = getItem("xie-input-autoplay", '0'); if (autoplayV == 1) { console.log('开始自动播放'); // xieSvgBf.click(); var clickEvent = new Event('click', { bubbles: true, cancelable: true, }); xieSvgBf.dispatchEvent(clickEvent); } }, 1500); } } function getItem(key, defValue) { let v = localStorage.getItem(key); if (v == null) { v = defValue; setItem(key, v); } return v; } function setItem(key, value) { localStorage.setItem(key, value); } // 展开 function clickZk() { xieSvgZk.style.display = 'none'; xieSvgSs.style.display = 'unset'; showAndHidden(xieBody, 'flex', 1); } // 收缩 function clickSs() { xieSvgSs.style.display = 'none'; showAndHidden(xieBody, null, 0); xieSvgZk.style.display = 'unset'; } function bfzt(type) { if (type) { xieSvgBf.style.display = 'none'; xieSvgZt.style.display = 'unset'; } else { xieSvgZt.style.display = 'none'; xieSvgBf.style.display = 'unset'; } } // 播放 function clickBf() { if (window.speechSynthesis.speaking) { console.log('继续播放'); window.speechSynthesis.resume(); //继续 } else { console.log('开始播放'); getContent(); } bfzt(true); } // 暂停 function clickZt() { console.log('暂停'); window.speechSynthesis.pause(); } /* 其它方法 resume() 重新开始 stop() 立即终止 正在播放语音:window.speechSynthesis.speaking && !window.speechSynthesis.pause 处于暂停中:window.speechSynthesis.paused * */ // 清空 function clickQk() { console.log('清空'); bfzt(false); // 清除所有语音播报创建的队列 window.speechSynthesis.cancel(); setCurrentText('未开始朗读!!!'); } var contents = []; function getContent() { clickQk(); // console.log('获取播放内容') const contentReserveWords = ['p', 'span', 'br', 'pre', 'b', 'strong', 'h1', 'h2', 'h3', 'h4', 'h5', 'h6', 'em', 'i', 'u']; var childNodes = contentEle.childNodes; for (var i = childNodes.length - 1; i >= 0; i--) { var childNode = childNodes[i]; if (childNode.nodeType === 1 && !contentReserveWords.includes(childNode.nodeName.toLowerCase())) { contentEle.removeChild(childNode); } } let stra = contentEle.innerText.split('\n'); // console.log(stra) if (titleEle && titleEle.textContent) { contents.push(titleEle.textContent.replace(/[\/-]/g, '/')) } for (let i = 0; i < stra.length; i++){ let s = filterStr(stra[i]); // 如果这一行为空 或 含需要过滤的关键词 或 第一行与标题内容相同 跳过 if (s === '' || (contents.length > 0 && contents[0].replace(' ', '') === s.replace(' ', '') && i === 0)) { continue; } contents.push(s); //console.log(s) } if (contents.length > 0) { let s = '本次朗读结束!' if (autoplayV == 1) { if (rightButton == null) { s += '未找到下一章元素,自动跳转失败' } else { s += '即将自动跳转下一页' } } else { clickZk(); } contents.push(s) mySpeechSynthesis(0); } console.log(contents) } function filterStr(s) { if (s) { s = getText(s); filterKeywords.forEach(function (v) { if (s.indexOf(v) !== -1) { // 定义标点符号数组 let si = s.indexOf(v); let li = -1; // 从字符串末尾开始向前遍历 for (let i = si - 1; i >= 0; i--) { // 如果当前字符是标点符号,则返回该下标 if (punctuationMarks.includes(s[i])) { li = i; break; } } console.log(s, li, v) if (li === -1) { s = ''; return; } else { s = s.substring(0, li + 1); } } }); } return s; } function setCurrentText(message) { // console.log('当前播放:', message); bottomContent.innerHTML = message; } function mySpeechSynthesis(i) { let message = contents[i]; let msg = new SpeechSynthesisUtterance(); // 文本 msg.text = message; // 声音 msg.voice = window.speechSynthesis.getVoices()[Number(getItem("xie-input-voice", null))]; // 音量:0~1,默认1, msg.volume = Number(getItem("xie-input-volume", 1)); // 语速:0.1~10,默认1 msg.rate = Number(getItem("xie-input-rate", 1)); // 音高:0~2,默认1 msg.pitch = Number(getItem("xie-input-pitch", 1)); // 开始 msg.onstart = ()=> { if (i === 0) { clickSs(); } // console.log('开始') setCurrentText(message); } // 暂停 msg.onpause = ()=> { // console.log('暂停'); bfzt(false); } // 结束回调 msg.onend = ()=> { if (!window.speechSynthesis.pending) { console.log('结束', contents.length, i); if (i >= contents.length - 1) { bfzt(false); setCurrentText('播放完成!!!'); if (autoplayV == 1) { if (rightButton == null) { clickZk(); } else { setTimeout(function() { rightButton.click(); }, 1000); } } else { clickZk(); } } else { mySpeechSynthesis(i + 1); } } }; window.speechSynthesis.speak(msg); // console.log(msg) } }, 1500); })();