eBooks Assistant

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

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