您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
微信读书 => 微信听书
// ==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(); })();
QingJ © 2025
镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址