// ==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}">
<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> · · · · · ·</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}">
<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> · · · · · ·</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> 京东读书</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> · · · · · ·</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}">
<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> · · · · · ·</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(/ /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;">豆瓣评分 ${douban_rating_score} </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);
}
})();