Greasy Fork is available in English.

eBooks Assistant

eBooks Assistant for douban.com, weread.qq.com

// ==UserScript==
// @name         eBooks Assistant
// @name:zh-CN   豆瓣读书助手
// @namespace    https://github.com/caspartse/eBooksAssistant
// @version      24.07.2
// @description  eBooks Assistant for douban.com, weread.qq.com
// @description:zh-CN 为豆瓣读书页面添加微信读书、多看阅读、京东读书、当当云阅读、喜马拉雅等直达链接; 为微信读书增加豆瓣评分及链接。
// @icon         https://ebooks-assistant.oss-cn-guangzhou.aliyuncs.com/ebooks_assistant_logo_256.png
// @author       Caspar Tse
// @license      MIT License
// @supportURL   https://github.com/caspartse/eBooksAssistant
// @match        https://book.douban.com/subject/*
// @match        https://weread.qq.com/web/bookDetail/*
// @match        https://weread.qq.com/web/reader/*
// @require      https://cdnjs.cloudflare.com/ajax/libs/jquery/3.7.1/jquery.min.js
// @connect      127.0.0.1
// @connect      api.youdianzishu.com
// @run-at       document-end
// @grant        GM_xmlhttpRequest
// ==/UserScript==

const version = "24.07.2";
// 如果自己部署服务,这里修改成你的服务器地址
const REST_URL = "https://api.youdianzishu.com/v2";

// Base64 icons
const base64_icon_weread = "";
const base64_icon_duokan = "";
const base64_icon_jd = "";
const base64_icon_dangdang = "";
const base64_icon_ximalaya = "";
const base64_icon_douban = "";
const base64_icon_douban_rating = "";

let x_unique_id = Math.random().toString(36).substring(2, 12);
console.log(x_unique_id);

// 信息查询:微信读书
const queryWeread = (isbn, title, subtitle, author, translator, publisher) => {
    const handleResponse = (responseDetail) => {
        const result = JSON.parse(responseDetail.responseText);
        console.log(result);
        if (result.errmsg === "") {
            const { url, price } = result.data;

            let html_template_purchase = `<li><div class="cell price-btn-wrapper"><div class="vendor-name"><img class="eba_vendor_icon" src="${base64_icon_weread}">&nbsp;
            <a target="_blank" href="${url}"><span>微信读书</span></a></div><div class="cell impression_track_mod_buyinfo"><div class="cell price-wrapper">
            <a target="_blank" href="${url}"><span class="buylink-price"> ${price}元 </span></a></div><div class="cell">
            <a target="_blank" href="${url}" class="buy-book-btn e-book-btn"><span>购买电子书</span></a></div></div></div></li>`;

            if ($("#buyinfo .current-version-list").length) {
                $("#buyinfo .current-version-list").prepend(html_template_purchase);
            } else {
                let elm_buyinfo_printed = `<div class="buyinfo-printed" id="buyinfo-printed"><h2><span>当前版本有售</span> &nbsp;·&nbsp;·&nbsp;·&nbsp;·&nbsp;·&nbsp;·</h2><ul class="bs current-version-list"></ul></div>`;
                $("#buyinfo").prepend(elm_buyinfo_printed);
                $("#buyinfo .current-version-list").prepend(html_template_purchase);
            }
        }
    }
    GM_xmlhttpRequest({
        method: "GET",
        url: `${REST_URL}/weread?isbn=${isbn}&title=${title}&subtitle=${subtitle}&author=${author}&translator=${translator}&publisher=${publisher}&version=${version}&r=${Math.random()}`,
        headers: {
            "User-agent": window.navigator.userAgent,
            "X-Referer": window.location.href,
            "X-Unique-ID": x_unique_id
        },
        onload: handleResponse
    });
}

// 信息查询:多看阅读
const queryDuokan = (isbn, title, subtitle, author, translator, publisher) => {
    const handleResponse = (responseDetail) => {
        const result = JSON.parse(responseDetail.responseText);
        console.log(result);
        if (result.errmsg === "") {
            const { url, price } = result.data;

            let html_template_purchase = `<li><div class="cell price-btn-wrapper"><div class="vendor-name"><img class="eba_vendor_icon" src="${base64_icon_duokan}">&nbsp;
            <a target="_blank" href="${url}"><span>多看阅读</span></a></div><div class="cell impression_track_mod_buyinfo"><div class="cell price-wrapper">
            <a target="_blank" href="${url}"><span class="buylink-price"> ${price}元 </span></a></div><div class="cell">
            <a target="_blank" href="${url}" class="buy-book-btn e-book-btn"><span>购买电子书</span></a></div></div></div></li>`;

            if ($("#buyinfo .current-version-list").length) {
                $("#buyinfo .current-version-list").prepend(html_template_purchase);
            } else {
                let elm_buyinfo_printed = `<div class="buyinfo-printed" id="buyinfo-printed"><h2><span>当前版本有售</span> &nbsp;·&nbsp;·&nbsp;·&nbsp;·&nbsp;·&nbsp;·</h2><ul class="bs current-version-list"></ul></div>`;
                $("#buyinfo").prepend(elm_buyinfo_printed);
                $("#buyinfo .current-version-list").prepend(html_template_purchase);
            }
        }
    }
    GM_xmlhttpRequest({
        method: "GET",
        url: `${REST_URL}/duokan?isbn=${isbn}&title=${title}&subtitle=${subtitle}&author=${author}&translator=${translator}&publisher=${publisher}&version=${version}&r=${Math.random()}`,
        headers: {
            "User-agent": window.navigator.userAgent,
            "X-Referer": window.location.href,
            "X-Unique-ID": x_unique_id
        },
        onload: handleResponse
    });
}

// 信息查询:京东读书
const queryJingdong = (isbn, title, subtitle, author, translator, publisher) => {
    const handleResponse = (responseDetail) => {
        const result = JSON.parse(responseDetail.responseText);
        console.log(result);
        if (result.errmsg === "") {
            const { url, price } = result.data;

            let html_template_purchase = `<li><div class="cell price-btn-wrapper"><div class="vendor-name"><img class="eba_vendor_icon" src="${base64_icon_jd}">
            <a target="_blank" href="${url}"><span>&nbsp;京东读书</span></a></div><div class="cell impression_track_mod_buyinfo"><div class="cell price-wrapper">
            <a target="_blank" href="${url}"><span class="buylink-price"> ${price}元 </span></a></div><div class="cell">
            <a target="_blank" href="${url}" class="buy-book-btn e-book-btn"><span>购买电子书</span></a></div></div></div></li>`;

            if ($("#buyinfo .current-version-list").length) {
                $("#buyinfo .current-version-list").prepend(html_template_purchase);
            } else {
                let elm_buyinfo_printed = `<div class="buyinfo-printed" id="buyinfo-printed"><h2><span>当前版本有售</span> &nbsp;·&nbsp;·&nbsp;·&nbsp;·&nbsp;·&nbsp;·</h2><ul class="bs current-version-list"></ul></div>`;
                $("#buyinfo").prepend(elm_buyinfo_printed);
                $("#buyinfo .current-version-list").prepend(html_template_purchase);
            }
        }
    }
    GM_xmlhttpRequest({
        method: "GET",
        url: `${REST_URL}/jd?isbn=${isbn}&title=${title}&subtitle=${subtitle}&author=${author}&translator=${translator}&publisher=${publisher}&version=${version}&r=${Math.random()}`,
        headers: {
            "User-agent": window.navigator.userAgent,
            "X-Referer": window.location.href,
            "X-Unique-ID": x_unique_id
        },
        onload: handleResponse
    });
}

// 信息查询:当当云阅读
const queryDangdang = (isbn, title, subtitle, author, translator, publisher) => {
    const handleResponse = (responseDetail) => {
        const result = JSON.parse(responseDetail.responseText);
        console.log(result);
        if (result.errmsg === "") {
            const { url, price } = result.data;

            let html_template_purchase = `<li><div class="cell price-btn-wrapper"><div class="vendor-name"><img class="eba_vendor_icon" src="${base64_icon_dangdang}">&nbsp;
            <a target="_blank" href="${url}"><span>当当阅读</span></a></div><div class="cell impression_track_mod_buyinfo"><div class="cell price-wrapper">
            <a target="_blank" href="${url}"><span class="buylink-price"> ${price}元 </span></a></div><div class="cell">
            <a target="_blank" href="${url}" class="buy-book-btn e-book-btn"><span>购买电子书</span></a></div></div></div></li>`;

            if ($("#buyinfo .current-version-list").length) {
                $("#buyinfo .current-version-list").prepend(html_template_purchase);
            } else {
                let elm_buyinfo_printed = `<div class="buyinfo-printed" id="buyinfo-printed"><h2><span>当前版本有售</span> &nbsp;·&nbsp;·&nbsp;·&nbsp;·&nbsp;·&nbsp;·</h2><ul class="bs current-version-list"></ul></div>`;
                $("#buyinfo").prepend(elm_buyinfo_printed);
                $("#buyinfo .current-version-list").prepend(html_template_purchase);
            }
        }
    }
    GM_xmlhttpRequest({
        method: "GET",
        url: `${REST_URL}/dangdang?isbn=${isbn}&title=${title}&subtitle=${subtitle}&author=${author}&translator=${translator}&publisher=${publisher}&version=${version}&r=${Math.random()}`,
        headers: {
            "User-agent": window.navigator.userAgent,
            "X-Referer": window.location.href,
            "X-Unique-ID": x_unique_id
        },
        onload: handleResponse
    });
}

// 信息查询:喜马拉雅
const queryXimalaya = (isbn, title, subtitle, author, translator, publisher) => {
    const handleResponse = (responseDetail) => {
        const result = JSON.parse(responseDetail.responseText);
        console.log(result);
        if (result.errmsg === "") {
            const { url } = result.data;

            const constructHtmlTemplatePartner = (type) => {
                let template = `<div class="online-read-or-audio">
                    <div class="vendor-info">
                        <img class="vendor-icon" src="${base64_icon_ximalaya}">
                        <a class="vendor-name impression_track_mod_buyinfo" target="_blank" href="${url}">
                            喜马拉雅
                        </a>
                    </div>
                    <a class="vendor-link" target="_blank" href="${url}">
                        去试听
                    </a>
                </div>`;

                if (type === 'header') {
                    template = `<div class="online-type" data-ebassistant="audio"><h2>在线试听:</h2>${template}</div>`;
                }

                if (type === 'parent') {
                    template = `<div class="gray_ad online-partner"><h2>在线试听:</h2>${template}</div>`;
                }

                return template;
            }

            let html_template_partner;
            if ($('.online-type[data-ebassistant="audio"]').length) { // 如果有试读听条目
                html_template_partner = constructHtmlTemplatePartner();
                $('.online-type[data-ebassistant="audio"] h2').after(html_template_partner);
            } else if ($('.online-type[data-ebassistant="read"]').length) { // 如果没有试读听条目,但有试读条目
                html_template_partner = constructHtmlTemplatePartner('header');
                $('.online-type[data-ebassistant="read"]').after(html_template_partner);
            } else { // 如果既没有试读听条目,也没有试读条目
                if ($('.gray_ad.online-partner').length) { // 如果有 <div class="gray_ad online-partner"> 节点,插入元素
                    html_template_partner = constructHtmlTemplatePartner('header');
                    $('.gray_ad.online-partner').after(html_template_partner);
                } else { // 如果没有 <div class="gray_ad online-partner"> 节点,创建节点
                    html_template_partner = constructHtmlTemplatePartner('parent');
                    $('#buyinfo').append(html_template_partner);
                }
            }

        }
    }
    GM_xmlhttpRequest({
        method: "GET",
        url: `${REST_URL}/ximalaya?isbn=${isbn}&title=${title}&subtitle=${subtitle}&author=${author}&translator=${translator}&publisher=${publisher}&version=${version}&r=${Math.random()}`,
        headers: {
            "User-agent": window.navigator.userAgent,
            "X-Referer": window.location.href
        },
        onload: handleResponse
    });
}

// 同步图书元数据
const syncMetadata = (isbn, metadata) => {
    GM_xmlhttpRequest({
        method: "POST",
        url: `${REST_URL}/sync_metadata?isbn=${isbn}&version=${version}&r=${Math.random()}`,
        headers: {
            "Content-Type": "application/json",
            "User-agent": window.navigator.userAgent,
            "X-Referer": window.location.href,
            "X-Unique-ID": x_unique_id
        },
        data: JSON.stringify(metadata),
        onload: (responseDetail) => {
            const result = JSON.parse(responseDetail.responseText);
            console.log(result);
        }
    });
}

// 样式调整:添加新样式
const addNewStyle = () => {
    const new_style = `<style type="text/css" media="screen">
    /* 豆瓣读书页面 */
    .eba_vendor_icon {
        text-decoration: none;
        display: inline-block;
        vertical-align: middle;
        width: 15px;
        height: 15px;
        margin-top: -2px;
        border: 0;
        border-radius: 50%;
        box-shadow: 0 0 1px 0 rgba(0, 0, 0, 0.6);
    }

    /* 微信读书页面 */
    .douban_rating {
        width: 75px;
        height: 15px;
        display: inline-block;
        background-image: url("${base64_icon_douban_rating}");
        background-size: 75px 165px;
        background-repeat: no-repeat;
    }
    .douban_rating_star_0 {
        background-position: 0 -150px;
    }
    .douban_rating_star_1 {
        background-position: 0 -135px;
    }
    .douban_rating_star_2 {
        background-position: 0 -120px;
    }
    .douban_rating_star_3 {
        background-position: 0 -105px;
    }
    .douban_rating_star_4 {
        background-position: 0 -90px;
    }
    .douban_rating_star_5 {
        background-position: 0 -75px;
    }
    .douban_rating_star_6 {
        background-position: 0 -60px;
    }
    .douban_rating_star_7 {
        background-position: 0 -45px;
    }
    .douban_rating_star_8 {
        background-position: 0 -30px;
    }
    .douban_rating_star_9 {
        background-position: 0 -15px;
    }
    .douban_rating_star_10 {
        background-position: 0 0;
    }
    </style>`;
    $("html").append(new_style);
}

// 豆瓣读书页面主函数
const doubanMain = () => {
    try {
        const types = ['在线试读', '在线试听'];
        const data = ['read', 'audio'];
        types.forEach((type, index) => {
            $('.online-partner .online-type h2:contains("' + type + '")').parent('.online-type').attr("data-ebassistant", data[index]); // 添加 data-ebassistant 属性
        });
    } catch(e) {
        console.log(e);
    }

    let _doc = document.documentElement.innerHTML;
    const regex_linked_data = /<script type="application\/ld\+json">([\s\S]+?)<\/script>/gi;
    let linked_data = JSON.parse(regex_linked_data.exec(_doc)[1].trim());
    const { isbn, name: title, url } = linked_data;
    const author = linked_data.author.map(author => author.name).join(', ');

    _doc = _doc.replace(/&nbsp;/gi, " ");

    // 豆瓣评分 rating_score
    let rating_score = extractData(_doc, /<strong class="ll rating_num " property="v:average">([\s\S]+?)<\/strong>/gi);
    // 出版社 publisher
    let publisher = extractData(_doc, /<span class="pl">\s*出版社:?<\/span>\s*:?\s*<a[^>]+>([\s\S]+?)<\/a>/gi);
    if (!publisher) {
        publisher = extractData(_doc, /<span class="pl">\s*出版社:?<\/span>\s*:?\s*([\s\S]+?)<br\/?>/gi);
    }
    // 出品方 producer
    let producer = extractData(_doc, /<span class="pl">\s*出品方:?<\/span>\s*:?\s*<a[^>]+>([\s\S]+?)<\/a>/gi);
    if (!producer) {
        producer = extractData(_doc, /<span class="pl">\s*出品方:?<\/span>\s*:?\s*([\s\S]+?)<br\/?>/gi);
    }
    // 副标题 subtitle
    let subtitle = extractData(_doc, /<span class="pl">\s*副标题:?<\/span>\s*:?\s*([\s\S]+?)<br\/?>/gi);
    // 原作名 original_title
    let original_title = extractData(_doc, /<span class="pl">\s*原作名:?<\/span>\s*:?\s*([\s\S]+?)<br\/?>/gi);
    // 译者 translator
    let translator = extractData(_doc, /<span class="pl">\s*译者:?<\/span>\s*:?\s*<a[^>]+>([\s\S]+?)<\/a>/gi);
    if (!translator) {
        translator = extractData(_doc, /<span class="pl">\s*译者:?<\/span>\s*:?\s*([\s\S]+?)<br\/?>/gi);
    }
    // 出版年 published
    let published = extractData(_doc, /<span class="pl">\s*出版年:?<\/span>\s*:?\s*([\s\S]+?)<br\/?>/gi);
    // 页数 pages
    let pages = extractData(_doc, /<span class="pl">\s*页数:?<\/span>\s*:?\s*([\s\S]+?)<br\/?>/gi);
    // 定价 price
    let price = extractData(_doc, /<span class="pl">\s*定价:?<\/span>\s*:?\s*([\s\S]+?)<br\/?>/gi);
    // 装帧 binding
    let binding = extractData(_doc, /<span class="pl">\s*装帧:?<\/span>\s*:?\s*([\s\S]+?)<br\/?>/gi);
    // 丛书 series
    let series = extractData(_doc, /<span class="pl">\s*丛书:?<\/span>\s*:?\s*<a[^>]+>([\s\S]+?)<\/a>/gi);
    if (!series) {
        series = extractData(_doc, /<span class="pl">\s*丛书:?<\/span>\s*:?\s*([\s\S]+?)<br\/?>/gi);
    }
    // 内容简介 description
    let description = extractData(_doc, /<meta property="og:description" content="([^"]+?)"/gi);
    description = description.replace(/<[^>]+>|\n/g, "");
    // 封面图片 cover_url
    let cover_url = extractData(_doc, /<meta property="og:image" content="([^"]+?)"/gi);
    // 🚀🎉🎊🥳 图书元数据开放接口已上线,可前往 https://forms.gle/91z4wrtQngrbkK1g9 申请使用。
    const metadata = {
        isbn, rating_score, url, title, author, publisher, producer, subtitle, original_title, translator, published, pages, price, binding, series, description, cover_url
    };
    console.log(metadata);

    queryWeread(isbn, title, subtitle, author, translator, publisher);
    queryDuokan(isbn, title, subtitle, author, translator, publisher);
    queryJingdong(isbn, title, subtitle, author, translator, publisher);
    queryDangdang(isbn, title, subtitle, author, translator, publisher);
    queryXimalaya(isbn, title, subtitle, author, translator, publisher);
    syncMetadata(isbn, metadata);
};
const extractData = (doc, regex) => {
    try {
        return regex.exec(doc)[1].trim();
    } catch(e) {
        console.log(e);
        return "";
    }
};

// 微信读书页面主函数
const wereadMain = () => {
    let vbookid = "";
    const locationHref = window.location.href;
    const match = locationHref.match(/(?:bookDetail|reader)\/([0-9a-zA-Z]+)/);

    if (match && match[1].length <= 24) {
        vbookid = match[1];
        console.log(vbookid);
    } else {
        console.log('vbookid not match.');
        return;
    }

    const handleResponse = (responseDetail) => {
        const result = JSON.parse(responseDetail.responseText);
        console.log(result);
        if (result.errmsg === "") {
            const { url, douban_rating_score, douban_rating_star } = result.data;
            const book_ratings_container = $(".book_ratings_container");
            const douban_info = `
                <div id="eba_douban_rating" class="book_ratings_header" style="margin-bottom:24px;cursor:pointer!important;">
                    <a style="text-decoration:none!important;color:#1b88ee!important;" target="_blank" href="${url}">
                    <span style="display:flex;align-items:center;">
                        <img src="${base64_icon_douban}" style="display:inline-block;height:15px;">
                        <span style="display:inline-block;height:24px;padding:0 4px;">豆瓣评分&nbsp;${douban_rating_score}&nbsp;</span>
                        <span class="douban_rating ${douban_rating_star}"></span>
                    </span>
                    </a>
                </div>`;
            $("#eba_douban_rating").remove();
            book_ratings_container.prepend(douban_info);
        }
    };

    GM_xmlhttpRequest ({
        method: "GET",
        url: `${REST_URL}/weread/douban_info?vbookid=${vbookid}&version=${version}&r=${Math.random()}`,
        headers: {
            "User-agent": window.navigator.userAgent,
            "X-Referer": window.location.href,
            "X-Unique-ID": x_unique_id
        },
        onload: handleResponse
    });
};

// 主函数
(() => {
    'use strict';
    addNewStyle();
    const hostname = window.location.hostname;
    if (/book\.douban\.com/.test(hostname)) {
        doubanMain();
    } else if (/weread\.qq\.com/.test(hostname)) {
        setTimeout(() => wereadMain(), 100);
    }
})();