Last active
June 30, 2022 10:31
-
-
Save btquanto/390cd3779c50c62ebf3598ee0cccea67 to your computer and use it in GitHub Desktop.
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
// ==UserScript== | |
// @name Novel Reader | |
// @namespace http://tampermonkey.net/ | |
// @version 0.1 | |
// @description try to take over the world! | |
// @author You | |
// @match https://metruyenchu.com/truyen/*/chuong* | |
// @match https://truyen.tangthuvien.vn/doc-truyen/*/chuong* | |
// @match https://www.novelpub.com/novel/* | |
// @match https://www.lightnovelpub.com/novel/* | |
// @match https://jpmtl.com/books/*/* | |
// @icon https://theitfox.com/public/favicon.jpg | |
// @grant none | |
// ==/UserScript== | |
(function () { | |
"use strict"; | |
class PromiseQueue { | |
constructor() { | |
this.tasks = []; | |
} | |
schedule(func) { | |
this.tasks.push(func); | |
if (this.tasks.length == 1) { | |
this.run(); | |
} | |
} | |
clear() { | |
this.tasks = []; | |
} | |
async run() { | |
const func = this.tasks[0]; | |
if (func !== undefined) { | |
try { | |
await func(); | |
} catch (ex) { | |
console.error(ex); | |
} | |
this.tasks.shift(); | |
await this.run(); | |
} | |
} | |
} | |
const VIEW_HTML = ` | |
<div | |
style=" | |
display: flex; | |
flex-flow: column; | |
background: #ebebeb; | |
padding: 8px; | |
border-radius: 8px; | |
position: fixed; | |
right: 54px; | |
top: 54px; | |
z-index: 9999; | |
-webkit-box-shadow: 1px 1px 4px 1px rgb(0 0 0 / 20%); | |
-moz-box-shadow: 1px 1px 4px 1px rgb(0 0 0 / 20%); | |
box-shadow: 1px 1px 4px 1px rgb(0 0 0 / 20%); | |
" | |
> | |
<a id="btn-toggle-show" href="#" style="margin-left: auto;"> | |
<img style="width: 20px" src=""/> | |
</a> | |
<div id="speech-controls-container" style="margin-top: 4px"> | |
<div style="display: flex; align-items: center"> | |
<label for="voice" style="width: 90px">Voice</label> | |
<select id="voice" style="width: 164px"></select> | |
</div> | |
<div style="display: flex; align-items: center; margin-top: 8px"> | |
<label for="language" style="width: 90px">Language</label> | |
<select id="language" style="width: 164px"></select> | |
</div> | |
<div style="display: flex; align-items: center; margin-top: 4px; width: 270px"> | |
<label for="speed" style="width: 90px">Speed</label> | |
<input | |
id="speed" | |
type="range" | |
value="1" | |
min="0.5" | |
max="2" | |
step="0.1" | |
style="width: 164px" | |
/> | |
</div> | |
<div style="display: flex; align-items: center; margin-top: 8px"> | |
<label for="pitch" style="width: 90px">Pitch</label> | |
<input | |
id="pitch" | |
type="range" | |
value="1" | |
min="0.1" | |
max="2" | |
step="0.1" | |
style="width: 164px" | |
/> | |
</div> | |
<div style="display: flex; justify-content: center; margin-top: 8px"> | |
<div id="playBtn" style="display: flex; align-items: center"> | |
<img | |
style="width: 28px; height: 28px" | |
src="" | |
/> | |
<img | |
style="width: 28px; height: 28px; display: none" | |
src="" | |
/> | |
</div> | |
<div id="stopBtn" style="margin-left: 12px"> | |
<img | |
style="width: 28px; height: 28px" | |
src="" | |
/> | |
</div> | |
<div id="nextBtn" style="margin-left: 12px"> | |
<img | |
style="width: 28px; height: 28px" | |
src="" | |
/> | |
</div> | |
</div> | |
</div> | |
</div> | |
`; | |
const LANGUAGES = { "en-US": "English", "vi-VN": "Vietnamese" }; | |
const DEFAULT_LANGUAGE = "vi-VN"; | |
const GROUP_SIZE = 2; | |
const SpeechSynthesis = window.speechSynthesis; | |
const queue = new PromiseQueue(); | |
let controlsContainer = null; | |
let voiceSelect = null; | |
let languageSelect = null; | |
let speedLabel = null; | |
let pitchLabel = null; | |
let playBtn = null; | |
let language = null; | |
let voices = []; | |
let voice = null; | |
let speed = 1; | |
let pitch = 1; | |
let utterance = null; | |
let state = "stopped"; | |
function changeVoice(e) { | |
localStorage.__voice = e.target.value; | |
voice = voices.filter((v) => v.name == e.target.value)[0]; | |
} | |
function changeLanguage(e) { | |
language = e.target.value; | |
localStorage.__language = e.target.value; | |
voices = speechSynthesis.getVoices().filter((v) => v.lang.match(language)); | |
voice = voices[0]; | |
voiceSelect.innerHTML = voices.map((v) => `<option value="${v.name}" ${v.name == voice.name ? "selected" : ""}>${v.name}</option>`).join("\n"); | |
} | |
function changeSpeed(e) { | |
speed = parseFloat(e.target.value); | |
localStorage.__speed = speed; | |
speedLabel.innerText = `Speed (${speed}x)`; | |
} | |
function changePitch(e) { | |
pitch = parseFloat(e.target.value); | |
localStorage.__pitch = pitch; | |
pitchLabel.innerText = `Pitch (${pitch}x)`; | |
} | |
function getTexts() { | |
const groupSize = GROUP_SIZE; | |
if (location.host.match("metruyenchu.com")) { | |
return document | |
.querySelector("#js-read__content") | |
.innerText.split("\n") | |
.filter((s) => s.length > 0 && s != "— QUẢNG CÁO —") | |
.reduce((prev, item, idx) => { | |
let arr = prev[Math.floor(idx / groupSize)] || []; | |
prev[Math.floor(idx / groupSize)] = arr; | |
arr.push(item); | |
return prev; | |
}, []) | |
.map((group) => group.join("\n")); | |
} | |
if (location.host.match("tangthuvien")) { | |
return document | |
.querySelector(".chapter-c-content .box-chap") | |
.innerText.split("\n") | |
.filter((s) => s.length > 0) | |
.reduce((prev, item, idx) => { | |
let arr = prev[Math.floor(idx / groupSize)] || []; | |
prev[Math.floor(idx / groupSize)] = arr; | |
arr.push(item); | |
return prev; | |
}, []) | |
.map((group) => group.join("\n")); | |
} | |
if (location.host.match("novelpub.com")) { | |
return [...document.querySelectorAll("div#chapter-container > p")] | |
.map((p) => p.textContent) | |
.filter( | |
(s) => | |
s != "The latest_episodes are on_the novelpub.com website." && | |
s != "Visit novelpub.com for a better_user experience" && | |
s != "You can_find the rest of this_content on the novelpub.com platform." && | |
s != "Try the novelpub.com platform_for the most advanced_reading experience." | |
) | |
.reduce((prev, item, idx) => { | |
let arr = prev[Math.floor(idx / groupSize)] || []; | |
prev[Math.floor(idx / groupSize)] = arr; | |
arr.push(item); | |
return prev; | |
}, []) | |
.map((group) => group.join("\n")); | |
} | |
if (location.host.match("jpmtl.com")) { | |
return [...document.querySelectorAll("div.chapter-content__content > div.cp-content > p")] | |
.map((p) => p.textContent) | |
.reduce((prev, item, idx) => { | |
let arr = prev[Math.floor(idx / groupSize)] || []; | |
prev[Math.floor(idx / groupSize)] = arr; | |
arr.push(item); | |
return prev; | |
}, []) | |
.map((group) => group.join("\n")); | |
} | |
return []; | |
} | |
function onEnd() { | |
if (location.host.match("metruyenchu.com")) { | |
document.querySelector("#nextChapter").click(); | |
} | |
if (location.host.match("tangthuvien")) { | |
document.querySelector(".bot-next_chap.bot-control").click(); | |
} | |
if (location.host.match("novelpub.com")) { | |
document.querySelector("a.nextchap").click(); | |
} | |
if (location.host.match("jpmtl.com")) { | |
document.querySelector("a.chapter-wrapper__nav:last-child").click(); | |
} | |
} | |
function nextButton() { | |
SpeechSynthesis.cancel(); | |
} | |
function playButton() { | |
if (!utterance) { | |
stopButton(); | |
} | |
if (state == "playing") { | |
playBtn.querySelector("img:nth-child(1)").style.display = ""; | |
playBtn.querySelector("img:nth-child(2)").style.display = "none"; | |
state = "paused"; | |
localStorage.__reading = 0; | |
SpeechSynthesis.pause(); | |
} else if (state == "stopped") { | |
playBtn.querySelector("img:nth-child(1)").style.display = "none"; | |
playBtn.querySelector("img:nth-child(2)").style.display = ""; | |
state = "playing"; | |
localStorage.__reading = 1; | |
const texts = getTexts(); | |
console.log(texts); | |
const length = texts.length; | |
texts.forEach((text, idx) => { | |
queue.schedule(() => { | |
let tries = 0; | |
function speakText(resolve, reject, fallback=false) { | |
utterance = new SpeechSynthesisUtterance(text); | |
if(!fallback) utterance.voice = voice; | |
utterance.lang = language; | |
utterance.rate = speed; | |
utterance.pitch = pitch; | |
utterance.addEventListener("end", (event) => { | |
resolve(); | |
if (idx == length - 1) { | |
onEnd(); | |
} | |
}); | |
utterance.addEventListener("error", (event) => { | |
console.error(event, `Error: ${text}`); | |
if (tries <= 10) { | |
tries++; | |
setTimeout(() => { | |
speakText(resolve, reject, tries >= 3); | |
}, 1000); | |
} else { | |
reject(); | |
if (idx == length - 1) { | |
onEnd(); | |
} | |
} | |
}); | |
SpeechSynthesis.speak(utterance); | |
} | |
return new Promise(speakText); | |
}); | |
}); | |
} else if (state == "paused") { | |
playBtn.querySelector("img:nth-child(1)").style.display = "none"; | |
playBtn.querySelector("img:nth-child(2)").style.display = ""; | |
state = "playing"; | |
localStorage.__reading = 1; | |
SpeechSynthesis.resume(); | |
} | |
} | |
function stopButton() { | |
playBtn.querySelector("img:nth-child(1)").style.display = ""; | |
playBtn.querySelector("img:nth-child(2)").style.display = "none"; | |
state = "stopped"; | |
localStorage.__reading = 0; | |
queue.clear(); | |
SpeechSynthesis.cancel(); | |
} | |
function toggleShowButton() { | |
if (!controlsContainer.style.display) { | |
controlsContainer.style.display = "none"; | |
} else { | |
controlsContainer.style.display = ""; | |
} | |
} | |
function setup() { | |
utterance = null; | |
SpeechSynthesis.cancel(); | |
speed = parseFloat(localStorage.__speed); | |
if (isNaN(speed)) { | |
speed = 1.0; | |
localStorage.__speed = speed; | |
} | |
pitch = parseFloat(localStorage.__pitch); | |
if (isNaN(pitch)) { | |
pitch = 1.0; | |
localStorage.__pitch = pitch; | |
} | |
language = localStorage.__language; | |
if (!language) { | |
language = DEFAULT_LANGUAGE; | |
} | |
voices = speechSynthesis.getVoices().filter((v) => v.lang.match(language)); | |
voice = voices.filter((v) => v.name == localStorage.__voice)[0]; | |
if (!voice) { | |
voice = voices[0]; | |
} | |
const container = document.createElement("div"); | |
container.innerHTML = VIEW_HTML; | |
const div = container.querySelector("div:first-child").cloneNode(true); | |
document.body.appendChild(div); | |
div.querySelector("#speed").value = speed; | |
div.querySelector("#pitch").value = pitch; | |
voiceSelect = div.querySelector("#voice"); | |
voiceSelect.innerHTML = voices.map((v) => `<option value="${v.name}" ${v.name == voice.name ? "selected" : ""}>${v.name}</option>`).join("\n"); | |
voiceSelect.onchange = changeVoice; | |
languageSelect = div.querySelector("#language"); | |
languageSelect.innerHTML = Object.entries(LANGUAGES) | |
.map(([key, value]) => `<option value="${key}" ${key == language ? "selected" : ""}>${value}</option>`) | |
.join("\n"); | |
languageSelect.onchange = changeLanguage; | |
div.querySelector("#speed").onchange = changeSpeed; | |
div.querySelector("#pitch").onchange = changePitch; | |
speedLabel = div.querySelector("label[for=speed]"); | |
speedLabel.innerText = `Speed (${speed}x)`; | |
pitchLabel = div.querySelector("label[for=pitch]"); | |
pitchLabel.innerText = `Pitch (${pitch}x)`; | |
playBtn = div.querySelector("div#playBtn"); | |
playBtn.onclick = playButton; | |
div.querySelector("div#stopBtn").onclick = stopButton; | |
div.querySelector("div#nextBtn").onclick = nextButton; | |
div.querySelector("#btn-toggle-show").onclick = toggleShowButton; | |
controlsContainer = div.querySelector("#speech-controls-container"); | |
if (parseInt(localStorage.__reading) && !SpeechSynthesis.speaking) { | |
playBtn.click(); | |
} | |
} | |
setTimeout(setup, 1000); | |
})(); |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment