Greasy Fork is available in English.

笔趣阁小说朗读

自动获取网页上的小说内容并转换为语音

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


})();