// ==UserScript== // @name 小说朗读 // @namespace http://tampermonkey.net/ // @version 2.1.1 // @description 小说阅读,朗读 // @author FHT // @icon  // @grant GM_addElement // @grant GM_addStyle // @grant unsafeWindow // @license MIT // @exclude https://www.hujuge.com/*/index.html // @match https://www.hujuge.com/*/*html // @match https://www.qidian.com/chapter/*/*/ // @match https://www.douyinxs.com/bqg/*/*.html // @downloadURL https://update.greasyfork.icu/scripts/460474/%E5%B0%8F%E8%AF%B4%E6%9C%97%E8%AF%BB.user.js // @updateURL https://update.greasyfork.icu/scripts/460474/%E5%B0%8F%E8%AF%B4%E6%9C%97%E8%AF%BB.meta.js // ==/UserScript== (function() { 'use strict'; const rule = [ { url: '域名', nextSelector: '下一页', prevSelector: '上一页', indexSelector: '目录', titleSelector: '章节名', bookTitleSelector: '书名', contentSelector: '正文', contentReplace: [['去广告正则', ""], [] ] },{ url: 'www.hujuge.com', nextSelector: '#next_url', prevSelector: '#prev_url', indexSelector: '#info_url', titleSelector: '.title', bookTitleSelector: '.layout-tit>a:nth-of-type(2)', contentSelector: '#content', contentReplace: [ ['/\n/g', ' '], [/章节错误.*/g,'']] }, { url: 'www.douyinxs.com', nextSelector: '#next', prevSelector: '#prev', indexSelector: '.bottem1 a:nth-child(2)', titleSelector: '.bookname h1', bookTitleSelector: '.con_top>a:nth-of-type(3)', contentSelector: '#content', contentReplace: [['/\n/g', ' ']] },{ url: 'www.qidian.com', nextSelector: '.nav-btn-group>a:nth-last-of-type(1)', prevSelector: '.nav-btn-group>a:nth-of-type(1)', indexSelector: '.nav-btn-group>a:nth-of-type(2)', titleSelector: '.chapter-wrapper h1', bookTitleSelector: '#r-breadcrumbs>a:nth-of-type(4)', contentSelector: '.content', contentReplace: [[/\s\s/g, '\n'],[/\d+<\/span>/g, ''] ] }, ] GM_addElement('script', { src: 'https://cdn.staticfile.net/vue/3.3.4/vue.global.min.js', type: 'text/javascript' }); GM_addElement('script', { src: 'https://cdn.staticfile.net/axios/1.6.5/axios.min.js', type: 'text/javascript' }); window.onload = () => { let body = document.querySelector('body') let htmlData = body.innerHTML const synth = window.speechSynthesis; synth.cancel() const utterThis = new SpeechSynthesisUtterance(); body.innerHTML = '' GM_addElement(body, 'div', { id:'app' }); const { createApp } = Vue createApp({ data() { return { bookData: { bookName: null, //书名 chapter: [], //章节 content: [], //正文 next: null, //下一章url prev: null, //目录url menu: null, //上一章url }, voicesData: { voicesList: [], //语音列表 voicesIndex: 5, //默认语音 }, readData: { readIndex: 0, //当前章节下标 }, speakData: { speakState: 0, //语音是否存在 speakingState: 1, //语音暂停播放 speakIndex: 0, //语音播放的章节下标 focusText: null //是否有选中文字 }, data: '', //隐藏div数据 state: 1, //监听防抖 } }, created() { this.data = htmlData }, mounted() { this.setData() //获取朗读引擎 const _this = this synth.onvoiceschanged = function () { _this.voicesData.voicesList = [] let arr = synth.getVoices() for (let i = 0; i < arr.length; i++) { if (arr[i].lang == 'zh-CN') { _this.voicesData.voicesList.push(arr[i]) } } } utterThis.onend = (event) => { this.speakData.focusText = null let tag = document.querySelector('.right_' + this.speakData.speakIndex) tag.innerHTML = tag.innerHTML.replace(/<\/?span.*?>/g, '') this.speakData.speakIndex++ if (this.speakData.speakIndex < this.bookData.chapter.length) { tag = document.querySelector('.right_' + this.speakData.speakIndex) tag.scrollIntoView(true) utterThis.voice = this.voicesData.voicesList[this.voicesData.voicesIndex] utterThis.text = tag.innerText synth.speak(utterThis) } else { axios({ url: this.bookData.next, method: 'get' }).then(res => { this.data = /]*>([\s\S]*)<\/body>/.exec(res.data)[1] this.setData() setTimeout(() => { tag = document.querySelector('.right_' + _this.speakData.speakIndex) tag.scrollIntoView(true) utterThis.voice = _this.voicesData.voicesList[_this.voicesData.voicesIndex] utterThis.text = tag.innerText synth.speak(utterThis) }, 1); }) } }; utterThis.onboundary = (event) => { let div = document.querySelector('.right_' + this.speakData.speakIndex) let str let read_text = [] if (this.speakData.focusText) { let t1 = div.innerText.split(this.speakData.focusText)[0] let t2 = this.speakData.focusText + div.innerText.split(this.speakData.focusText)[1] str = t2.substr(event.charIndex, event.charLength) read_text[0] = t2.slice(0, event.charIndex) read_text[1] = t2.slice(event.charIndex) div.innerHTML = t1 + read_text[0] + read_text[1].replace(str, "" + str + "") } else { str = div.innerText.substr(event.charIndex, event.charLength) read_text[0] = div.innerText.substr(0, event.charIndex) read_text[1] = div.innerText.substr(event.charIndex) div.innerHTML = read_text[0] + read_text[1].replace(str, "" + str + "") } if (document.querySelector('.activ').offsetTop > document.querySelector('.right').scrollTop + document.querySelector('.right').clientHeight) { document.querySelector('.right').scrollTop = document.querySelector('.activ').offsetTop - 50 } } //监听滚动条 const right = document.querySelector('.right') right.addEventListener('scroll', () => { if (right.scrollHeight - right.scrollTop - right.clientHeight <= 400) { this.state++ if (this.state == 2) { this.getData({ url: this.bookData.next, method: 'get' }) } } //判断当前页面在第几章 switch (document.querySelectorAll('.right>div').length) { case 1: this.readData.readIndex = 0; break; case 2: right.scrollTop >= document.querySelectorAll('.right>div')[1].offsetTop ? this.readData.readIndex = 1 : this.readData.readIndex = 0 break; default: for (let index = 0; index < document.querySelectorAll('.right>div').length - 1; index++) { if (document.querySelectorAll('.right>div')[index].offsetTop <= right.scrollTop && right.scrollTop < document.querySelectorAll('.right>div')[index + 1].offsetTop) { this.readData.readIndex = index break } else { right.scrollTop < document.querySelectorAll('.right>div')[1].offsetTop ? this.readData.readIndex = 0 : this.readData.readIndex = index + 1 } } break; } }) }, methods: { selectChange() { this.voicesData.voicesIndex = document.querySelector('#voiceSelect').value }, // 语音功能按钮 play() { this.speakData.speakIndex = this.readData.readIndex utterThis.voice = this.voicesData.voicesList[this.voicesData.voicesIndex] if (window.getSelection().toString()) { this.speakData.focusText = window.getSelection().toString() utterThis.text = this.speakData.focusText + document.querySelector('.right_' + this.speakData.speakIndex).innerText.split(this.speakData.focusText)[1] } else { utterThis.text = document.querySelector('.right_' + this.speakData.speakIndex).innerText } synth.speak(utterThis) this.speakData.speakState = synth.speaking }, del() { synth.cancel() this.speakData.speakState = synth.speaking this.speakData.speakingState = !synth.paused }, suspend() { if (synth.speaking) { synth.pause() this.speakData.speakingState = synth.paused } else ( alert('没有播放文本') ) }, recovery() { if (synth.speaking) { synth.resume() this.speakData.speakingState = !synth.paused } }, // 设置数据 setData() { rule.forEach(element => { if (window.location.host == element.url) { this.$nextTick(() => { const name = document.querySelector(element.bookTitleSelector) const chapter = document.querySelector(element.titleSelector) const menu = document.querySelector(element.indexSelector) const prev = document.querySelector(element.prevSelector) const next = document.querySelector(element.nextSelector) const content = document.querySelector(element.contentSelector) element.contentReplace.forEach(e => { content.innerHTML = content.innerHTML.replace(e[0], e[1]) }) this.bookData.bookName = name ? name.innerHTML : '未找到书名' this.bookData.chapter.push({ msg: chapter ? chapter.innerText : '未找到章节名', src: this.bookData.next || window.location.href }) this.bookData.menu = menu ? menu.href : '未找到' this.bookData.prev = this.bookData.prev ? this.bookData.prev : prev.href this.bookData.next = next ? next.href : '未找到' this.bookData.content.push(content ? content.innerText : '未找到内容') this.data = '' }) } }); }, // 请求数据 getData(params) { axios({ url: params.url, method: params.method }).then(res => { this.state = 1 this.data = /]*>([\s\S]*)<\/body>/.exec(res.data)[1] this.setData() }) }, //点击章节列表 click_left_chapter(params) { this.readData.readIndex = params const tag = '.right > div:nth-of-type(' + (params + 1) + ')' document.querySelector(tag).scrollIntoView(true) }, //根据标签内容选择标签 tagContains(params) { const arr = document.querySelectorAll(params.tag) for (let index = 0; index < arr.length; index++) { const element = arr[index]; if (element.innerText == params.msg) { return element } } } }, watch: { 'readData.readIndex'() { window.history.replaceState('', '', this.bookData.chapter[this.readData.readIndex].src) } }, template: `
{{bookData.bookName}}
{{ i.msg }}
{{ bookData.chapter[j].msg }}
`, }).mount('#app') } // css const css = ` * { margin: 0; padding: 0; } #app { width: 100vw; height: 100vh; display: flex; overflow: hidden; background: url(https://qidian.gtimg.com/qd/images/read.qidian.com/theme/theme_1_bg_2x.0.3.png); } .left { width: 250px; height: 100%; overflow: hidden; background: rgb(70, 70, 70); color: white; font-size: 14px; cursor: pointer; padding: 8px; } .left a{ color: white; } .left_bookname { text-align: center; font-size: 24px; font-weight: bold; white-space: nowrap; overflow: hidden; text-overflow: ellipsis; padding: 10px 10px 15px; border-bottom: 2px solid black; } .left_select { display: flex; justify-content: space-evenly; padding: 10px; border-bottom: 1px solid black; } .left_chapter { overflow: auto; height: 85%; } .left_chapter div { padding: 10px; border-bottom: 1px solid black; white-space: nowrap; overflow: hidden; text-overflow: ellipsis; } .right { overflow: auto; width: 100%; height: 100%; min-width: 60%; font-size: 26px; line-height: 50px; white-space: pre-line !important; user-select: text !important; } .tit { text-align: center; margin: 10px 80px; padding: 10px; border-bottom: solid 1px rgb(134, 124, 124); font-size: 32px; } .right div { margin: 0 100px; letter-spacing: 4px; } .speak { position: fixed; top: 10px; right: 30px; } .li_activ { background: #000; } .activ { color: red; } ` GM_addStyle(css) })();