七猫全文在线免费读

Qimao Novel Reading, No Need for a Client, Novels Available for Download

// ==UserScript==
// @name              七猫全文在线免费读
// @namespace         https://github.com/ibxff
// @version           2.0.+n.3
// @description       七猫小说全文免费网页读 不用客户端 可下载小说
// @description:zh-cn 七猫小说全文免费网页读 不用客户端 可下载小说
// @description:en    Qimao Novel Reading, No Need for a Client, Novels Available for Download
// @author            ibxff
// @match             https://api-bc.wtzw.com/*
// @match             https://www.qimao.com/*
// @require           https://greasyfork.org/scripts/479459-cryptojs/code/CryptoJS-.js?version=1277994
// @require           https://cdnjs.cloudflare.com/ajax/libs/crypto-js/3.1.9-1/crypto-js.js
// @require           https://cdnjs.cloudflare.com/ajax/libs/FileSaver.js/2.0.5/FileSaver.min.js
// @icon              
// @license           MIT
// @grant             GM_xmlhttpRequest
// @grant             GM_addStyle
// ==/UserScript==



(function() {


    var rotateLeft = function(lValue, iShiftBits) {
    return(lValue << iShiftBits) | (lValue >>> (32 - iShiftBits));
    }

var addUnsigned = function(lX, lY) {
    var lX4, lY4, lX8, lY8, lResult;
    lX8 = (lX & 0x80000000);
    lY8 = (lY & 0x80000000);
    lX4 = (lX & 0x40000000);
    lY4 = (lY & 0x40000000);
    lResult = (lX & 0x3FFFFFFF) + (lY & 0x3FFFFFFF);
    if(lX4 & lY4) return(lResult ^ 0x80000000 ^ lX8 ^ lY8);
    if(lX4 | lY4) {
        if(lResult & 0x40000000) return(lResult ^ 0xC0000000 ^ lX8 ^ lY8);
        else return(lResult ^ 0x40000000 ^ lX8 ^ lY8);
    } else {
        return(lResult ^ lX8 ^ lY8);
    }
}

var F = function(x, y, z) {
    return(x & y) | ((~x) & z);
}

var G = function(x, y, z) {
    return(x & z) | (y & (~z));
}

var H = function(x, y, z) {
    return(x ^ y ^ z);
}

var I = function(x, y, z) {
    return(y ^ (x | (~z)));
}

var FF = function(a, b, c, d, x, s, ac) {
    a = addUnsigned(a, addUnsigned(addUnsigned(F(b, c, d), x), ac));
    return addUnsigned(rotateLeft(a, s), b);
};

var GG = function(a, b, c, d, x, s, ac) {
    a = addUnsigned(a, addUnsigned(addUnsigned(G(b, c, d), x), ac));
    return addUnsigned(rotateLeft(a, s), b);
};

var HH = function(a, b, c, d, x, s, ac) {
    a = addUnsigned(a, addUnsigned(addUnsigned(H(b, c, d), x), ac));
    return addUnsigned(rotateLeft(a, s), b);
};

var II = function(a, b, c, d, x, s, ac) {
    a = addUnsigned(a, addUnsigned(addUnsigned(I(b, c, d), x), ac));
    return addUnsigned(rotateLeft(a, s), b);
};

var convertToWordArray = function(string) {
    var lWordCount;
    var lMessageLength = string.length;
    var lNumberOfWordsTempOne = lMessageLength + 8;
    var lNumberOfWordsTempTwo = (lNumberOfWordsTempOne - (lNumberOfWordsTempOne % 64)) / 64;
    var lNumberOfWords = (lNumberOfWordsTempTwo + 1) * 16;
    var lWordArray = Array(lNumberOfWords - 1);
    var lBytePosition = 0;
    var lByteCount = 0;
    while(lByteCount < lMessageLength) {
        lWordCount = (lByteCount - (lByteCount % 4)) / 4;
        lBytePosition = (lByteCount % 4) * 8;
        lWordArray[lWordCount] = (lWordArray[lWordCount] | (string.charCodeAt(lByteCount) << lBytePosition));
        lByteCount++;
    }
    lWordCount = (lByteCount - (lByteCount % 4)) / 4;
    lBytePosition = (lByteCount % 4) * 8;
    lWordArray[lWordCount] = lWordArray[lWordCount] | (0x80 << lBytePosition);
    lWordArray[lNumberOfWords - 2] = lMessageLength << 3;
    lWordArray[lNumberOfWords - 1] = lMessageLength >>> 29;
    return lWordArray;
};

var wordToHex = function(lValue) {
    var WordToHexValue = "",
        WordToHexValueTemp = "",
        lByte, lCount;
    for(lCount = 0; lCount <= 3; lCount++) {
        lByte = (lValue >>> (lCount * 8)) & 255;
        WordToHexValueTemp = "0" + lByte.toString(16);
        WordToHexValue = WordToHexValue + WordToHexValueTemp.substr(WordToHexValueTemp.length - 2, 2);
    }
    return WordToHexValue;
};

var uTF8Encode = function(string) {
    string = string.replace(/\x0d\x0a/g, "\x0a");
    var output = "";
    for(var n = 0; n < string.length; n++) {
        var c = string.charCodeAt(n);
        if(c < 128) {
            output += String.fromCharCode(c);
        } else if((c > 127) && (c < 2048)) {
            output += String.fromCharCode((c >> 6) | 192);
            output += String.fromCharCode((c & 63) | 128);
        } else {
            output += String.fromCharCode((c >> 12) | 224);
            output += String.fromCharCode(((c >> 6) & 63) | 128);
            output += String.fromCharCode((c & 63) | 128);
        }
    }
    return output;
};

function md5(string) {
    var x = Array();
    var k, AA, BB, CC, DD, a, b, c, d;
    var S11 = 7,
        S12 = 12,
        S13 = 17,
        S14 = 22;
    var S21 = 5,
        S22 = 9,
        S23 = 14,
        S24 = 20;
    var S31 = 4,
        S32 = 11,
        S33 = 16,
        S34 = 23;
    var S41 = 6,
        S42 = 10,
        S43 = 15,
        S44 = 21;
    string = uTF8Encode(string);
    x = convertToWordArray(string);
    a = 0x67452301;
    b = 0xEFCDAB89;
    c = 0x98BADCFE;
    d = 0x10325476;
    for(k = 0; k < x.length; k += 16) {
        AA = a;
        BB = b;
        CC = c;
        DD = d;
        a = FF(a, b, c, d, x[k + 0], S11, 0xD76AA478);
        d = FF(d, a, b, c, x[k + 1], S12, 0xE8C7B756);
        c = FF(c, d, a, b, x[k + 2], S13, 0x242070DB);
        b = FF(b, c, d, a, x[k + 3], S14, 0xC1BDCEEE);
        a = FF(a, b, c, d, x[k + 4], S11, 0xF57C0FAF);
        d = FF(d, a, b, c, x[k + 5], S12, 0x4787C62A);
        c = FF(c, d, a, b, x[k + 6], S13, 0xA8304613);
        b = FF(b, c, d, a, x[k + 7], S14, 0xFD469501);
        a = FF(a, b, c, d, x[k + 8], S11, 0x698098D8);
        d = FF(d, a, b, c, x[k + 9], S12, 0x8B44F7AF);
        c = FF(c, d, a, b, x[k + 10], S13, 0xFFFF5BB1);
        b = FF(b, c, d, a, x[k + 11], S14, 0x895CD7BE);
        a = FF(a, b, c, d, x[k + 12], S11, 0x6B901122);
        d = FF(d, a, b, c, x[k + 13], S12, 0xFD987193);
        c = FF(c, d, a, b, x[k + 14], S13, 0xA679438E);
        b = FF(b, c, d, a, x[k + 15], S14, 0x49B40821);
        a = GG(a, b, c, d, x[k + 1], S21, 0xF61E2562);
        d = GG(d, a, b, c, x[k + 6], S22, 0xC040B340);
        c = GG(c, d, a, b, x[k + 11], S23, 0x265E5A51);
        b = GG(b, c, d, a, x[k + 0], S24, 0xE9B6C7AA);
        a = GG(a, b, c, d, x[k + 5], S21, 0xD62F105D);
        d = GG(d, a, b, c, x[k + 10], S22, 0x2441453);
        c = GG(c, d, a, b, x[k + 15], S23, 0xD8A1E681);
        b = GG(b, c, d, a, x[k + 4], S24, 0xE7D3FBC8);
        a = GG(a, b, c, d, x[k + 9], S21, 0x21E1CDE6);
        d = GG(d, a, b, c, x[k + 14], S22, 0xC33707D6);
        c = GG(c, d, a, b, x[k + 3], S23, 0xF4D50D87);
        b = GG(b, c, d, a, x[k + 8], S24, 0x455A14ED);
        a = GG(a, b, c, d, x[k + 13], S21, 0xA9E3E905);
        d = GG(d, a, b, c, x[k + 2], S22, 0xFCEFA3F8);
        c = GG(c, d, a, b, x[k + 7], S23, 0x676F02D9);
        b = GG(b, c, d, a, x[k + 12], S24, 0x8D2A4C8A);
        a = HH(a, b, c, d, x[k + 5], S31, 0xFFFA3942);
        d = HH(d, a, b, c, x[k + 8], S32, 0x8771F681);
        c = HH(c, d, a, b, x[k + 11], S33, 0x6D9D6122);
        b = HH(b, c, d, a, x[k + 14], S34, 0xFDE5380C);
        a = HH(a, b, c, d, x[k + 1], S31, 0xA4BEEA44);
        d = HH(d, a, b, c, x[k + 4], S32, 0x4BDECFA9);
        c = HH(c, d, a, b, x[k + 7], S33, 0xF6BB4B60);
        b = HH(b, c, d, a, x[k + 10], S34, 0xBEBFBC70);
        a = HH(a, b, c, d, x[k + 13], S31, 0x289B7EC6);
        d = HH(d, a, b, c, x[k + 0], S32, 0xEAA127FA);
        c = HH(c, d, a, b, x[k + 3], S33, 0xD4EF3085);
        b = HH(b, c, d, a, x[k + 6], S34, 0x4881D05);
        a = HH(a, b, c, d, x[k + 9], S31, 0xD9D4D039);
        d = HH(d, a, b, c, x[k + 12], S32, 0xE6DB99E5);
        c = HH(c, d, a, b, x[k + 15], S33, 0x1FA27CF8);
        b = HH(b, c, d, a, x[k + 2], S34, 0xC4AC5665);
        a = II(a, b, c, d, x[k + 0], S41, 0xF4292244);
        d = II(d, a, b, c, x[k + 7], S42, 0x432AFF97);
        c = II(c, d, a, b, x[k + 14], S43, 0xAB9423A7);
        b = II(b, c, d, a, x[k + 5], S44, 0xFC93A039);
        a = II(a, b, c, d, x[k + 12], S41, 0x655B59C3);
        d = II(d, a, b, c, x[k + 3], S42, 0x8F0CCC92);
        c = II(c, d, a, b, x[k + 10], S43, 0xFFEFF47D);
        b = II(b, c, d, a, x[k + 1], S44, 0x85845DD1);
        a = II(a, b, c, d, x[k + 8], S41, 0x6FA87E4F);
        d = II(d, a, b, c, x[k + 15], S42, 0xFE2CE6E0);
        c = II(c, d, a, b, x[k + 6], S43, 0xA3014314);
        b = II(b, c, d, a, x[k + 13], S44, 0x4E0811A1);
        a = II(a, b, c, d, x[k + 4], S41, 0xF7537E82);
        d = II(d, a, b, c, x[k + 11], S42, 0xBD3AF235);
        c = II(c, d, a, b, x[k + 2], S43, 0x2AD7D2BB);
        b = II(b, c, d, a, x[k + 9], S44, 0xEB86D391);
        a = addUnsigned(a, AA);
        b = addUnsigned(b, BB);
        c = addUnsigned(c, CC);
        d = addUnsigned(d, DD);
    }
    var tempValue = wordToHex(a) + wordToHex(b) + wordToHex(c) + wordToHex(d);
    return tempValue.toLowerCase();
}






async function decode(response) {
    let $ = response
    let txt = CryptoJS.enc.Base64.parse($.data.content).toString()
    let iv = txt.slice(0,32)
    let _content = await decrypt(txt.slice(32),iv).trim()
        .replace(/\n/g,'<br>')
    return await _content;
}

const decrypt = function (data,iv) {
    let key = CryptoJS.enc.Hex.parse('32343263636238323330643730396531')
    iv = CryptoJS.enc.Hex.parse(iv)
    let HexStr = CryptoJS.enc.Hex.parse(data)
    let Base64Str = CryptoJS.enc.Base64.stringify(HexStr)
    let decrypted = CryptoJS.AES.decrypt(Base64Str, key, {
    iv: iv,
    mode: CryptoJS.mode.CBC,
    padding: CryptoJS.pad.Pkcs7
    })
    return decrypted.toString(CryptoJS.enc.Utf8)
}

async function novel(murl,func){

    // const matchb=murl.match(/\/(\d+)-(\d+)/)
    const matchb=murl.match(/https:\/\/www\.qimao\.com\/shuku\/([0-9_]+)-([0-9]+)\//)

    const toparams = obj => Object.keys(obj).map(key => `${encodeURIComponent(key)}=${encodeURIComponent(obj[key])}`).join('&');

    const sign_key = 'd3dGiJc651gSQ8w1';

    const params = {
        id: matchb[1],
        chapterId: matchb[2]
    };
    const paramSign = md5(Object.keys(params).sort().reduce((pre, n) => pre + n + '=' + params[n], '') + sign_key);
    params['sign'] = paramSign;

    const headers={
        "app-version": "51110",
        "platform": "android",
        "reg": "0",
        "AUTHORIZATION": "",
        "application-id": "com.****.reader",
        "net-env": "1",
        "channel": "unknown",
        "qm-params": "",
    }

    const headersSign = md5(Object.keys(headers).sort().reduce((pre, n) => pre + n + '=' + headers[n], '') + sign_key);
    headers['sign'] = headersSign;


    const url = "https://api-ks.wtzw.com/api/v1/chapter/content?" + toparams(params)


    console.log(url,headers)
    return new Promise((resolve, reject) => {
        GM_xmlhttpRequest({
            method: "GET",
            url: url,
            headers: headers,
            onload: async function(response) {
                //console.log(response.responseText)
                const data = JSON.parse(response.responseText);
                const get = (await decode(data)).replace(/<br>/g, '\n');
                await func(get)
                resolve()
                return
            },
            ontimeout:async function(){
                return await func(murl,func=func)
            },
            onerror : async function(response){
                func('err')
                throw new Error(JSON.stringify(response))
                reject()
            }
        });
    })

}



const murl=window.location.href

const mode =
    ///^https:\/\/www\.qimao\.com\/shuku\/\d+-\d+\//.test(murl) ? 'reader' :
    /https:\/\/www\.qimao\.com\/shuku\/([0-9_]+)-([0-9]+)\//.test(murl) ? 'reader' :
    /https:\/\/www\.qimao\.com\/shuku\/([0-9_]+)\//.test(murl) ? 'page' : null;
    // /^https:\/\/www\.qimao\.com\/shuku\/\d+\//.test(murl) ? 'page' : null;


switch(mode){

    case 'reader':
        GM_addStyle(`
            .reader-fixed-left li.go-back[data-v-b5fc2672]{
                border-radius:5px!important;
                background-color:#f1e5af36!important;

            }
            .reader-layout-theme[data-theme=default] .reader-fixed-left li.reader-guide[data-v-b5fc2672]{
                margin-top:0px!important;
                background-color:#f1e5af36!important;
            }
            .reader-header-con{
                display: none!important;
            }
            #__layout div div.wrapper.reader.reader-layout-theme div.reader-header div.reader-header-con{
                display: none!important;
            }

            .qm-fixed-right-item click reader-phone{
                display: none!important;
            }
            .s-tit{
                display: none!important;
            }
            .chapter-tips > *{
                color: #66666647!important;
                text-align: left!important;
            }
            .chapter-title{
                text-align: left!important;
            }
            .chapter-detail-wrap-info{
                border-bottom: 1px solid #ddd!important;
            }
            .qm-fixed-right-link{
                width: fit-content!important;
            }
            .qm-fixed-right.type-3 .qm-fixed-right-link[data-v-754474f8]:before{
                width:8px!important;
            }
            .qm-fixed-right[data-v-754474f8]{
                left:51%!important;
            }
            .i-arrow {
                border-radius: 5px;
            }
            .reader-setting , .book-catalog{
                border-radius: 10px!important;
                outline: 2px solid #0000000a;
            }
            .reader-login-code{
                display: none!important;
            }
            .show-part[data-v-10da8d56]:after{
                display: none!important;
            }
            .chapter-detail-article p{
                font-size: 18px;
                text-indent: 2em;
                line-height: 1.8;
                padding: 13px 0;
                user-select: text;
            }
            .article{
                user-select: text;
            }
            body{
                user-select: text;
            }
        `)
        novel(murl,func=(get)=>{
            // console.log(get)
            if(get==='err')return
            document.getElementsByClassName('chapter-detail-article')[0].innerHTML=
                    '<p>'+get.replace(/\n/g, "</p><p>")+'</p>';

        })
    break;
    case 'page':
        //console.log('debug')
        async function startdownload(){
            GM_addStyle(`
                .tab-inner{
                    display: none!important
                }
            `)
            document.querySelector("#__layout > div > div.wrapper > div > div > div > div.book-detail-body > div > div.qm-tab.type-2.normal > ul > li:nth-child(1) > div").click()
            await new Promise(resolve => setTimeout(resolve, 120));
            var content=
                '使用七猫全文在线免费读下载\n'+
                ['title','tags-wrap','sub-title','statistics-wrap','update-info','book-introduction-item']
                    .map(e => document.getElementsByClassName(e)[0].textContent+'\n')
                    .join()
                    .replace(/\n\s*(?!0\b)/g, '')
                    .replace(/,/g, '\n');
            for (const e of ['title','tags-wrap','sub-title','statistics-wrap','update-info','book-introduction-item']) {
                document.getElementsByClassName(e)[0].classList.add('succeed');
                await new Promise(resolve => setTimeout(resolve, 120));
            }
            document.querySelector("#__layout > div > div.wrapper > div > div > div > div.book-detail-body > div > div.qm-tab.type-2.normal > ul > li:nth-child(2) > div").click()

            await new Promise(resolve => setTimeout(resolve, 120));

            const elements = document.getElementsByClassName('qm-book-catalog-list-content')[0].children;


            let a;
            let span;
            for (var i = 0; i < elements.length; i++) {
                a=elements[i].querySelector('a')
                span=a.querySelector('.txt')
                span.classList.add('processing')
                content+='\n\n'+span.innerText+'\n'
                await novel(a.href,func=(c)=>{
                    span.classList.remove('processing')
                    if(c==='err'){
                        span.classList.add('err')
                        content+='\n章节获取错误\n'
                    }
                    span.classList.add('succeed')
                    content+=c
                });
            }

            const blob = new Blob([content], { type: "text/plain;charset=utf-8" });
            saveAs(blob, document.getElementsByClassName('title')[0].querySelector('span').innerText+".txt");
        }

        GM_addStyle(`
            #__layout > div > div.wrapper > div > div > div > div.book-detail-header > div > div.book-information.clearfix.left > div.wrap-txt > div.btns-wrap.clearfix > div > div.qm-popper-title > span{
                display :none!important;
            }
            .err {
                background-color: pink;

            }

            .succeed {
                background-color: #D2F9D1;
                border-radius: 5px;
            }

            .processing {
                background-color: navajowhite;
                border-radius: 5px;
            }
            .qm-book-catalog-list-content li{
                line-height : normal!important;
                padding : 18px;
            }
            .qm-book-catalog-list-content li .txt{
                width: auto!important;
            }

            .qm-book-catalog-list-content li .vip-icon{
                float: right!important;
                margin-top: 0px!important;
            }

        `)
        const menu=document.querySelector("#__layout > div > div.wrapper > div > div > div > div.book-detail-header > div > div.book-information.clearfix.left > div.wrap-txt > div.btns-wrap.clearfix")
        const copy=document.querySelector("#__layout > div > div.wrapper > div > div > div > div.book-detail-header > div > div.book-information.clearfix.left > div.wrap-txt > div.btns-wrap.clearfix > a.qm-btn.item.inline-block.default.large.radius")
            // ?document.querySelector("#__layout > div > div.wrapper > div > div > div > div.book-detail-header > div > div.book-information.clearfix.left > div.wrap-txt > div.btns-wrap.clearfix > a.qm-btn.item.inline-block.default.large.radius") :
            // document.querySelector("#__layout > div > div.wrapper > div > div > div > div.book-detail-header > div > div.book-information.clearfix.left > div.wrap-txt > div.btns-wrap.clearfix > a")
        const btn=copy.cloneNode(true);
        menu.insertBefore(btn, copy.nextSibling)
        btn.innerText='下载全本'
        btn.addEventListener('click',startdownload)



    break;

}


})()