Greasy Fork is available in English.

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