WeReader

微信读书 => 微信听书

You will need to install an extension such as Tampermonkey, Greasemonkey or Violentmonkey to install this script.

You will need to install an extension such as Tampermonkey or Violentmonkey to install this script.

You will need to install an extension such as Tampermonkey or Violentmonkey to install this script.

You will need to install an extension such as Tampermonkey or Userscripts to install this script.

You will need to install an extension such as Tampermonkey to install this script.

You will need to install a user script manager extension to install this script.

(I already have a user script manager, let me install it!)

You will need to install an extension such as Stylus to install this style.

You will need to install an extension such as Stylus to install this style.

You will need to install an extension such as Stylus to install this style.

You will need to install a user style manager extension to install this style.

You will need to install a user style manager extension to install this style.

You will need to install a user style manager extension to install this style.

(I already have a user style manager, let me install it!)

// ==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();
})();