Last active
August 10, 2022 12:05
-
-
Save giveme0101/2b269386707c8c17ba5b8e9f52d17fad to your computer and use it in GitHub Desktop.
WeReader 微信读书 => 微信听书
This file contains 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 WeReader | |
// @namespace https://github.com/giveme0101/ | |
// @version 2.1 | |
// @description 微信读书 => 微信听书 | |
// @author Kevin [email protected] | |
// @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