Skip to content

Instantly share code, notes, and snippets.

@giveme0101
Last active August 10, 2022 12:05
Show Gist options
  • Save giveme0101/2b269386707c8c17ba5b8e9f52d17fad to your computer and use it in GitHub Desktop.
Save giveme0101/2b269386707c8c17ba5b8e9f52d17fad to your computer and use it in GitHub Desktop.
WeReader 微信读书 => 微信听书
// ==UserScript==
// @name WeReader
// @namespace https://github.com/giveme0101/
// @version 2.1
// @description 微信读书 => 微信听书
// @author Kevin xiajun94@foxmail.com
// @match https://weread.qq.com/web/reader/*
// @icon0 https://weread.qq.com/favicon.ico
// @icon 
// @run-at document-idle
// @require https://code.jquery.com/jquery-3.1.1.min.js
// @grant none
// ==/UserScript==
// Proxy: https://segmentfault.com/a/1190000015483195
// Object.defineProperty: https://segmentfault.com/a/1190000015427628
window.fuckWeRead = {
fucked : false,
// 标题
title : "",
// 文章内容
buffer : "",
// 每个文字坐标
charMap : {
canvas: [],
span: []
},
// 是否已暂停
pause: true,
// 当前阅读片段索引
segmentIdx: 0,
// 阅读片段信息
segmentInfo: [],
// 鼠标划线位置坐标
selection: {},
audioUrl: "https://tts.baidu.com/text2audio?lan=zh&ie=UTF-8&spd=5.5&text="
};
const inject = () => {
document.addEventListener("DOMNodeRemoved", function(){
if (event.target.className && event.target.className.indexOf('preRenderContainer') != -1){
// console.log('DOMNodeRemoved --> preRenderContainer');
var afterRefresh = $(event.target).get(0).innerText.replaceAll("\n", "");
if (afterRefresh.length != window.fuckWeRead.buffer.length){
window.fuckWeRead.buffer = afterRefresh;
window.fuckWeRead.fucked = !0;
setTimeout(contentChange, 100);
}
}
})
document.querySelector(".readerChapterContent").addEventListener("DOMSubtreeModified",function(){
if (event.target.className && event.target.className.indexOf('chapterTitle') != -1){
window.fuckWeRead.title = $(this).find(".chapterTitle").text().trim();
}
},false);
document.querySelector(".renderTargetContainer").addEventListener("DOMSubtreeModified",function(){
if (event.target.className && event.target.className.indexOf('wr_canvasContainer') != -1){
var $canvasList = $(this).find("canvas");
if ($canvasList.length > 0){
//console.log('DOMSubtreeModified --> wr_canvasContainer');
window.fuckWeRead.charMap.canvas = [];
$canvasList.each(function(idx, $this){
window.fuckWeRead.charMap.canvas.push(getCanvasMap($this));
});
}
}
},false);
document.querySelector("#renderTargetContent").addEventListener("DOMSubtreeModified",function(){
var $spanList = $(this).find("span.wr_absolute");
if ($spanList.length > 0){
// console.log('DOMSubtreeModified --> wr_absolute');
window.fuckWeRead.charMap.span = [];
window.fuckWeRead.charMap.span.push(getSpanMap($spanList));
}
},false);
document.addEventListener("DOMNodeInserted",function(){
// 记录鼠标划线选中位置
if (event.target.className && event.target.className.indexOf('wr_selection') != -1){
// console.log('DOMNodeInserted --> wr_selection');
window.fuckWeRead.selection = {
x: Math.round($(".wr_selection:first").position().left) ,
y: Math.round($(".wr_selection:first").position().top)
};
}
// 添加"从此朗读"按钮
if (event.target.className && event.target.className.indexOf('reader_toolbar_container') != -1){
//console.log('DOMNodeInserted --> reader_toolbar_container');
if ($(this).find(".readStart").length == 0){
var btn = '<button class="toolbarItem readStart"><span style="color:#FFF;">☟</span><span class="toolbarItem_text">从此朗读</span></button>';
$(this).find('.reader_toolbar_itemContainer').append(btn);
$(".readStart").on('click', function(){
readFromHere(window.fuckWeRead.selection.x, window.fuckWeRead.selection.y);
});
}
}
},false);
}
const getCharMap = () => {
return window.fuckWeRead.charMap.canvas.concat(window.fuckWeRead.charMap.span);
}
const readFromHere = (x, y) => {
var wordIdx = findWordIdx(x, y);
if (!wordIdx){
toast("跳转失败!");
return;
}
var segIdx = findCharInSegmentInfo(wordIdx);
if (!segIdx){
toast("跳转失败!");
return;
}
window.fuckWeRead.segmentIdx = segIdx;
$("#playerList .running").attr("src", getUrl());
$("#playerList .player:not(.running)").each(function(){
$(this).attr("src", getUrl()).get(0).load();
});
$("#playerList .running").get(0).play();
window.fuckWeRead.pause = !1;
}
const findCharInSegmentInfo = (idx) => {
var segmentInfo = window.fuckWeRead.segmentInfo;
for (var i = 0, j = segmentInfo.length; i < j; i++){
var seg = segmentInfo[i];
var start = seg.start;
var end = start + seg.length;
if (idx >= start && idx <= end){
return i;
}
}
}
const findWordIdx = (x, y) => {
var map = getCharMap(), idx = 0;
for(var i = 0, j = map.length; i < j; i++){
for(var n = 0, seg = map[i], m = seg.length; n < m; n++){
idx++;
var _x = seg[n].x,
_y = seg[n].y;
if (x == _x && Math.abs(y - _y) < 4){
return idx;
}
}
}
}
const toast = (text) => {
$('<div>').appendTo('body').addClass('toast toast_Show').html('WeReader: ' + text).show(100).delay(1500).fadeOut(1000).queue(function(){
$(this).remove();
});
}
const getCanvasMap = (canvas) => {
var map = [], fontSize = 18;
var context = canvas.getContext("2d");
if (!context.hasOwnProperty("_fillText")){
var _fillText = context._fillText = context.fillText;
context.fillText = function(){
pushMap(map, arguments[0], arguments[1], arguments[2] - fontSize);
context._fillText.apply(this, [...arguments])
}
}
return map;
}
const getSpanMap = (spanList) => {
var textarr = [], map = [];
spanList.each(function() {
var $obj = $(this);
if ($obj.css('transform')) {
var xy = $obj.css("transform").replace(/[^0-9\-,]/g,'').split(',').slice(4,6);
textarr.push({
left: parseInt(xy[0]),
top: parseInt(xy[1]),
text: $obj.text()
});
}
})
_(_.sortBy(textarr, ['top', 'left'])).forEach(function(val) {
pushMap(map, val['text'], val['left'], val['top']);
})
return map;
}
const pushMap = (arr, txt, x, y) => {
if (txt.length > 1){
for (var c of txt){
arr.push({
txt : c,
x: x,
y: y
});
}
} else {
arr.push({
txt : txt,
x: x,
y: y
});
}
}
const getContent = () => {
var segmentIdx = window.fuckWeRead.segmentIdx++;
var segmentInfo = window.fuckWeRead.segmentInfo;
return segmentIdx >= segmentInfo.length ? null : segmentInfo[segmentIdx].content;
}
const getUrl = () => {
var content = getContent();
return content ? window.fuckWeRead.audioUrl + content : null;
}
const isSeparator = (char) => {
return ["。", ";", "…", "?", "!"].indexOf(char) != -1;
}
const renderCover = () => {
var bufferIdx = window.fuckWeRead.segmentIdx;
var segmentInfo = window.fuckWeRead.segmentInfo;
var readIdx = (bufferIdx - 2 < 0) ? 0 : (bufferIdx -2);
var segment = segmentInfo[readIdx];
var start = segment.start;
var end = start + segment.length - 1;
var pos0 = getPos(start);
// 忽略第一字是符号的情况
if (!isHaveText(pos0.txt)){
pos0 = getPos(start + 1);
}
var pos1 = getPos(end);
drawCover(pos0, pos1);
scroll(pos0);
}
const scroll = (pos) => {
$("html,body").animate({ scrollTop: pos.y - 200 }, "slow");
}
const getLine = (txt, left, top, width, height) => {
return '<div class="wr_underline" style="color: red; left: {{left}}px; top: {{top}}px; width: {{width}}px; height: {{height}}px;">{{txt}}</div>'
.replace("{{left}}", left)
.replace("{{top}}", top)
.replace("{{width}}", (width || 0))
.replace("{{height}}", (height || 29))
.replace("{{txt}}", txt);
}
const getLeft = (left, top, width, height) => {
return getLine("➤", left, top, width, height);
}
const getRight = (left, top, width, height) => {
return getLine("】", left, top, width, height);
}
const drawCover = (pos0, pos1) => {
var fontSize = 14,
span = 5,
x1 = pos0.x,
y1 = pos0.y,
x2 = pos1.x,
y2 = pos1.y;
var html = getLeft(x1 - fontSize - span, y1) + getRight(x2 + span, y2);
var $container = $("#progressContainer");
if (!$container.get(0)){
$(".renderTargetContainer").append("<div id = 'progressContainer'></div>");
}
$("#progressContainer").html(html);
}
const getPos = (idx) => {
var _pos = 0, map = getCharMap();
for (var i in map){
var segment = map[i];
if (0 == _pos && idx <= segment.length){
return segment[idx];
}
if (idx <= _pos + segment.length){
return segment[idx - _pos];
}
_pos += segment.length;
}
// 返回文章最后一个字
return map[map.length - 1].slice(-1)[0];
}
const playNextArtical = () => {
var $btn = $(".readerFooter>div").find("button[class=readerFooter_button]")[0];
if ($btn){
var ev = document.createEvent('HTMLEvents');
ev.clientX = ev.clientY = 356;
ev.initEvent('click', false, true);
$btn.dispatchEvent(ev);
}
}
const playNext = (prevIdx) => {
var idx = (prevIdx + 1) % $("#playerList .player").length;
if ($("#playerList .player").eq(idx).attr("src")){
$("#playerList .player").eq(idx).get(0).play();
$("#playerList .player").eq(prevIdx).attr("src", getUrl())
.get(0)
.load();
} else {
window.fuckWeRead.segmentIdx = 0;
playNextArtical();
}
}
const play = () => {
var player = $("#playerList .running").get(0);
if (window.fuckWeRead.pause){
player.play();
window.fuckWeRead.pause = !1;
} else {
player.pause();
window.fuckWeRead.pause = !0;
}
}
const attachEvent = () => {
// https://www.cnblogs.com/zhaodz/p/12031500.html
$("#playerList .player").each(function(_idx, _player){
$(this).on('play', function(){
$(this).addClass('running');
display('暂停');
renderCover();
});
$(this).on('pause', function(){
display('继续');
});
$(this).on('ended', function(){
$(this).removeClass('running');
display('播放');
playNext(_idx);
});
});
$("#fuckPannel").on('click', play);
}
const display = (txt) => {
$("#fuckPannel span").text(txt);
}
const contentChange = () => {
contentInit();
playerInit();
}
const isHaveText = (str) => {
var test = str.replace(/[\ |\~|\`|\!|\@|\#|\$|\%|\^|\&|\*|\(|\)|\-|\_|\+|\=|\||\\|\[|\]|\{|\}|\;|\:|\"|\'|\,|\…|\!|\?|\<|\.|\>|\/|\?/\,/\。/\;/\:/\“/\”/\》/\《/\|/\{/\}/\、/\!/\~/\`]/g,"");
return test.length > 0;
}
const contentInit =() => {
window.fuckWeRead.segmentIdx = 0;
window.fuckWeRead.segmentInfo = [];
var buffer = window.fuckWeRead.buffer,
fixMaxLength = 20,
readMaxLength = 20,
contentLength = buffer.length,
start = 0;
for(;;){
var maxEnd = start + readMaxLength;
maxEnd >= contentLength ? (maxEnd = contentLength) : void(0);
var segment = [], realContent = "", maxContent = buffer.substring(start, maxEnd);
for (var char of maxContent){
segment.push(char);
if (isSeparator(char)){
realContent += segment.join("");
segment = [];
break;
}
}
if (maxEnd == contentLength){
realContent += segment.join("");
}
if (realContent.length < 1){
readMaxLength <<= 1;
continue;
}
if (readMaxLength != fixMaxLength){
readMaxLength = fixMaxLength;
}
if (isHaveText(realContent)){
window.fuckWeRead.segmentInfo.push({
start: start,
length: realContent.length,
content: realContent
});
}
start += realContent.length;
if (start >= contentLength) {
break;
}
}
}
const getNearWord = (y) => {
var map = getCharMap(), idx = 0;
for(var i = 0, j = map.length; i < j; i++){
for(var n = 0, seg = map[i], m = seg.length; n < m; n++){
idx++;
var _y = seg[n].y;
if (_y > y){
return idx;
}
}
}
}
const playerInit = () => {
$("#fuckPannel,#playerList").remove();
var pannel = '<button id="fuckPannel" class="readerControls_item">' +
' <span class style="font-weight: bold; color: #595a5a ;">播放</span>' +
'</button>';
var audio = '<div id = "playerList" style="display:none">'+
' <audio class="player running" controls="controls" src="{{url}}" >' +
' <source class="tts_source" type="audio/mpeg">' +
' </audio>'+
' <audio class="player" controls="controls" src="{{url}}" >' +
' <source class="tts_source" type="audio/mpeg">' +
' </audio>'+
'</div>';
var t = setInterval(function(){
if (window.fuckWeRead.fucked){
clearInterval(t), t = null;
var wordIdx = getNearWord($('html').scrollTop());
var segIdx = findCharInSegmentInfo(wordIdx);
if (segIdx){
toast("已跳转至上次浏览位置,点击播放");
window.fuckWeRead.segmentIdx = segIdx;
}
$(".readerControls_item").eq(0).before(pannel);
$(".app_content").append(audio.replaceAll('{{url}}', function(){
return getUrl();
}));
attachEvent();
!window.fuckWeRead.pause && setTimeout(function(){
$("#playerList .running").get(0).play();
}, 1000);
}
}, 500);
}
(function() {
'use strict';
inject();
})();
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment