豆瓣读书助手

为豆瓣读书页面添加微信读书、多看阅读、京东读书、当当云阅读、喜马拉雅等直达链接; 为微信读书增加豆瓣评分及链接。

  1. // ==UserScript==
  2. // @name eBooks Assistant
  3. // @name:zh-CN 豆瓣读书助手
  4. // @namespace https://github.com/caspartse/eBooksAssistant
  5. // @version 24.07.2
  6. // @description eBooks Assistant for douban.com, weread.qq.com
  7. // @description:zh-CN 为豆瓣读书页面添加微信读书、多看阅读、京东读书、当当云阅读、喜马拉雅等直达链接; 为微信读书增加豆瓣评分及链接。
  8. // @icon https://ebooks-assistant.oss-cn-guangzhou.aliyuncs.com/ebooks_assistant_logo_256.png
  9. // @author Caspar Tse
  10. // @license MIT License
  11. // @supportURL https://github.com/caspartse/eBooksAssistant
  12. // @match https://book.douban.com/subject/*
  13. // @match https://weread.qq.com/web/bookDetail/*
  14. // @match https://weread.qq.com/web/reader/*
  15. // @require https://cdnjs.cloudflare.com/ajax/libs/jquery/3.7.1/jquery.min.js
  16. // @connect 127.0.0.1
  17. // @connect api.youdianzishu.com
  18. // @run-at document-end
  19. // @grant GM_xmlhttpRequest
  20. // ==/UserScript==
  21.  
  22. const version = "24.07.2";
  23. // 如果自己部署服务,这里修改成你的服务器地址
  24. const REST_URL = "https://api.youdianzishu.com/v2";
  25.  
  26. // Base64 icons
  27. const base64_icon_weread = "";
  28. const base64_icon_duokan = "";
  29. const base64_icon_jd = "";
  30. const base64_icon_dangdang = "";
  31. const base64_icon_ximalaya = "";
  32. const base64_icon_douban = "";
  33. const base64_icon_douban_rating = "";
  34.  
  35. let x_unique_id = Math.random().toString(36).substring(2, 12);
  36. console.log(x_unique_id);
  37.  
  38. // 信息查询:微信读书
  39. const queryWeread = (isbn, title, subtitle, author, translator, publisher) => {
  40. const handleResponse = (responseDetail) => {
  41. const result = JSON.parse(responseDetail.responseText);
  42. console.log(result);
  43. if (result.errmsg === "") {
  44. const { url, price } = result.data;
  45.  
  46. 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;
  47. <a target="_blank" href="${url}"><span>微信读书</span></a></div><div class="cell impression_track_mod_buyinfo"><div class="cell price-wrapper">
  48. <a target="_blank" href="${url}"><span class="buylink-price"> ${price}元 </span></a></div><div class="cell">
  49. <a target="_blank" href="${url}" class="buy-book-btn e-book-btn"><span>购买电子书</span></a></div></div></div></li>`;
  50.  
  51. if ($("#buyinfo .current-version-list").length) {
  52. $("#buyinfo .current-version-list").prepend(html_template_purchase);
  53. } else {
  54. 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>`;
  55. $("#buyinfo").prepend(elm_buyinfo_printed);
  56. $("#buyinfo .current-version-list").prepend(html_template_purchase);
  57. }
  58. }
  59. }
  60. GM_xmlhttpRequest({
  61. method: "GET",
  62. url: `${REST_URL}/weread?isbn=${isbn}&title=${title}&subtitle=${subtitle}&author=${author}&translator=${translator}&publisher=${publisher}&version=${version}&r=${Math.random()}`,
  63. headers: {
  64. "User-agent": window.navigator.userAgent,
  65. "X-Referer": window.location.href,
  66. "X-Unique-ID": x_unique_id
  67. },
  68. onload: handleResponse
  69. });
  70. }
  71.  
  72. // 信息查询:多看阅读
  73. const queryDuokan = (isbn, title, subtitle, author, translator, publisher) => {
  74. const handleResponse = (responseDetail) => {
  75. const result = JSON.parse(responseDetail.responseText);
  76. console.log(result);
  77. if (result.errmsg === "") {
  78. const { url, price } = result.data;
  79.  
  80. 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;
  81. <a target="_blank" href="${url}"><span>多看阅读</span></a></div><div class="cell impression_track_mod_buyinfo"><div class="cell price-wrapper">
  82. <a target="_blank" href="${url}"><span class="buylink-price"> ${price}元 </span></a></div><div class="cell">
  83. <a target="_blank" href="${url}" class="buy-book-btn e-book-btn"><span>购买电子书</span></a></div></div></div></li>`;
  84.  
  85. if ($("#buyinfo .current-version-list").length) {
  86. $("#buyinfo .current-version-list").prepend(html_template_purchase);
  87. } else {
  88. 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>`;
  89. $("#buyinfo").prepend(elm_buyinfo_printed);
  90. $("#buyinfo .current-version-list").prepend(html_template_purchase);
  91. }
  92. }
  93. }
  94. GM_xmlhttpRequest({
  95. method: "GET",
  96. url: `${REST_URL}/duokan?isbn=${isbn}&title=${title}&subtitle=${subtitle}&author=${author}&translator=${translator}&publisher=${publisher}&version=${version}&r=${Math.random()}`,
  97. headers: {
  98. "User-agent": window.navigator.userAgent,
  99. "X-Referer": window.location.href,
  100. "X-Unique-ID": x_unique_id
  101. },
  102. onload: handleResponse
  103. });
  104. }
  105.  
  106. // 信息查询:京东读书
  107. const queryJingdong = (isbn, title, subtitle, author, translator, publisher) => {
  108. const handleResponse = (responseDetail) => {
  109. const result = JSON.parse(responseDetail.responseText);
  110. console.log(result);
  111. if (result.errmsg === "") {
  112. const { url, price } = result.data;
  113.  
  114. let html_template_purchase = `<li><div class="cell price-btn-wrapper"><div class="vendor-name"><img class="eba_vendor_icon" src="${base64_icon_jd}">
  115. <a target="_blank" href="${url}"><span>&nbsp;京东读书</span></a></div><div class="cell impression_track_mod_buyinfo"><div class="cell price-wrapper">
  116. <a target="_blank" href="${url}"><span class="buylink-price"> ${price}元 </span></a></div><div class="cell">
  117. <a target="_blank" href="${url}" class="buy-book-btn e-book-btn"><span>购买电子书</span></a></div></div></div></li>`;
  118.  
  119. if ($("#buyinfo .current-version-list").length) {
  120. $("#buyinfo .current-version-list").prepend(html_template_purchase);
  121. } else {
  122. 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>`;
  123. $("#buyinfo").prepend(elm_buyinfo_printed);
  124. $("#buyinfo .current-version-list").prepend(html_template_purchase);
  125. }
  126. }
  127. }
  128. GM_xmlhttpRequest({
  129. method: "GET",
  130. url: `${REST_URL}/jd?isbn=${isbn}&title=${title}&subtitle=${subtitle}&author=${author}&translator=${translator}&publisher=${publisher}&version=${version}&r=${Math.random()}`,
  131. headers: {
  132. "User-agent": window.navigator.userAgent,
  133. "X-Referer": window.location.href,
  134. "X-Unique-ID": x_unique_id
  135. },
  136. onload: handleResponse
  137. });
  138. }
  139.  
  140. // 信息查询:当当云阅读
  141. const queryDangdang = (isbn, title, subtitle, author, translator, publisher) => {
  142. const handleResponse = (responseDetail) => {
  143. const result = JSON.parse(responseDetail.responseText);
  144. console.log(result);
  145. if (result.errmsg === "") {
  146. const { url, price } = result.data;
  147.  
  148. 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;
  149. <a target="_blank" href="${url}"><span>当当阅读</span></a></div><div class="cell impression_track_mod_buyinfo"><div class="cell price-wrapper">
  150. <a target="_blank" href="${url}"><span class="buylink-price"> ${price}元 </span></a></div><div class="cell">
  151. <a target="_blank" href="${url}" class="buy-book-btn e-book-btn"><span>购买电子书</span></a></div></div></div></li>`;
  152.  
  153. if ($("#buyinfo .current-version-list").length) {
  154. $("#buyinfo .current-version-list").prepend(html_template_purchase);
  155. } else {
  156. 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>`;
  157. $("#buyinfo").prepend(elm_buyinfo_printed);
  158. $("#buyinfo .current-version-list").prepend(html_template_purchase);
  159. }
  160. }
  161. }
  162. GM_xmlhttpRequest({
  163. method: "GET",
  164. url: `${REST_URL}/dangdang?isbn=${isbn}&title=${title}&subtitle=${subtitle}&author=${author}&translator=${translator}&publisher=${publisher}&version=${version}&r=${Math.random()}`,
  165. headers: {
  166. "User-agent": window.navigator.userAgent,
  167. "X-Referer": window.location.href,
  168. "X-Unique-ID": x_unique_id
  169. },
  170. onload: handleResponse
  171. });
  172. }
  173.  
  174. // 信息查询:喜马拉雅
  175. const queryXimalaya = (isbn, title, subtitle, author, translator, publisher) => {
  176. const handleResponse = (responseDetail) => {
  177. const result = JSON.parse(responseDetail.responseText);
  178. console.log(result);
  179. if (result.errmsg === "") {
  180. const { url } = result.data;
  181.  
  182. const constructHtmlTemplatePartner = (type) => {
  183. let template = `<div class="online-read-or-audio">
  184. <div class="vendor-info">
  185. <img class="vendor-icon" src="${base64_icon_ximalaya}">
  186. <a class="vendor-name impression_track_mod_buyinfo" target="_blank" href="${url}">
  187. 喜马拉雅
  188. </a>
  189. </div>
  190. <a class="vendor-link" target="_blank" href="${url}">
  191. 去试听
  192. </a>
  193. </div>`;
  194.  
  195. if (type === 'header') {
  196. template = `<div class="online-type" data-ebassistant="audio"><h2>在线试听:</h2>${template}</div>`;
  197. }
  198.  
  199. if (type === 'parent') {
  200. template = `<div class="gray_ad online-partner"><h2>在线试听:</h2>${template}</div>`;
  201. }
  202.  
  203. return template;
  204. }
  205.  
  206. let html_template_partner;
  207. if ($('.online-type[data-ebassistant="audio"]').length) { // 如果有试读听条目
  208. html_template_partner = constructHtmlTemplatePartner();
  209. $('.online-type[data-ebassistant="audio"] h2').after(html_template_partner);
  210. } else if ($('.online-type[data-ebassistant="read"]').length) { // 如果没有试读听条目,但有试读条目
  211. html_template_partner = constructHtmlTemplatePartner('header');
  212. $('.online-type[data-ebassistant="read"]').after(html_template_partner);
  213. } else { // 如果既没有试读听条目,也没有试读条目
  214. if ($('.gray_ad.online-partner').length) { // 如果有 <div class="gray_ad online-partner"> 节点,插入元素
  215. html_template_partner = constructHtmlTemplatePartner('header');
  216. $('.gray_ad.online-partner').after(html_template_partner);
  217. } else { // 如果没有 <div class="gray_ad online-partner"> 节点,创建节点
  218. html_template_partner = constructHtmlTemplatePartner('parent');
  219. $('#buyinfo').append(html_template_partner);
  220. }
  221. }
  222.  
  223. }
  224. }
  225. GM_xmlhttpRequest({
  226. method: "GET",
  227. url: `${REST_URL}/ximalaya?isbn=${isbn}&title=${title}&subtitle=${subtitle}&author=${author}&translator=${translator}&publisher=${publisher}&version=${version}&r=${Math.random()}`,
  228. headers: {
  229. "User-agent": window.navigator.userAgent,
  230. "X-Referer": window.location.href
  231. },
  232. onload: handleResponse
  233. });
  234. }
  235.  
  236. // 同步图书元数据
  237. const syncMetadata = (isbn, metadata) => {
  238. GM_xmlhttpRequest({
  239. method: "POST",
  240. url: `${REST_URL}/sync_metadata?isbn=${isbn}&version=${version}&r=${Math.random()}`,
  241. headers: {
  242. "Content-Type": "application/json",
  243. "User-agent": window.navigator.userAgent,
  244. "X-Referer": window.location.href,
  245. "X-Unique-ID": x_unique_id
  246. },
  247. data: JSON.stringify(metadata),
  248. onload: (responseDetail) => {
  249. const result = JSON.parse(responseDetail.responseText);
  250. console.log(result);
  251. }
  252. });
  253. }
  254.  
  255. // 样式调整:添加新样式
  256. const addNewStyle = () => {
  257. const new_style = `<style type="text/css" media="screen">
  258. /* 豆瓣读书页面 */
  259. .eba_vendor_icon {
  260. text-decoration: none;
  261. display: inline-block;
  262. vertical-align: middle;
  263. width: 15px;
  264. height: 15px;
  265. margin-top: -2px;
  266. border: 0;
  267. border-radius: 50%;
  268. box-shadow: 0 0 1px 0 rgba(0, 0, 0, 0.6);
  269. }
  270.  
  271. /* 微信读书页面 */
  272. .douban_rating {
  273. width: 75px;
  274. height: 15px;
  275. display: inline-block;
  276. background-image: url("${base64_icon_douban_rating}");
  277. background-size: 75px 165px;
  278. background-repeat: no-repeat;
  279. }
  280. .douban_rating_star_0 {
  281. background-position: 0 -150px;
  282. }
  283. .douban_rating_star_1 {
  284. background-position: 0 -135px;
  285. }
  286. .douban_rating_star_2 {
  287. background-position: 0 -120px;
  288. }
  289. .douban_rating_star_3 {
  290. background-position: 0 -105px;
  291. }
  292. .douban_rating_star_4 {
  293. background-position: 0 -90px;
  294. }
  295. .douban_rating_star_5 {
  296. background-position: 0 -75px;
  297. }
  298. .douban_rating_star_6 {
  299. background-position: 0 -60px;
  300. }
  301. .douban_rating_star_7 {
  302. background-position: 0 -45px;
  303. }
  304. .douban_rating_star_8 {
  305. background-position: 0 -30px;
  306. }
  307. .douban_rating_star_9 {
  308. background-position: 0 -15px;
  309. }
  310. .douban_rating_star_10 {
  311. background-position: 0 0;
  312. }
  313. </style>`;
  314. $("html").append(new_style);
  315. }
  316.  
  317. // 豆瓣读书页面主函数
  318. const doubanMain = () => {
  319. try {
  320. const types = ['在线试读', '在线试听'];
  321. const data = ['read', 'audio'];
  322. types.forEach((type, index) => {
  323. $('.online-partner .online-type h2:contains("' + type + '")').parent('.online-type').attr("data-ebassistant", data[index]); // 添加 data-ebassistant 属性
  324. });
  325. } catch(e) {
  326. console.log(e);
  327. }
  328.  
  329. let _doc = document.documentElement.innerHTML;
  330. const regex_linked_data = /<script type="application\/ld\+json">([\s\S]+?)<\/script>/gi;
  331. let linked_data = JSON.parse(regex_linked_data.exec(_doc)[1].trim());
  332. const { isbn, name: title, url } = linked_data;
  333. const author = linked_data.author.map(author => author.name).join(', ');
  334.  
  335. _doc = _doc.replace(/&nbsp;/gi, " ");
  336.  
  337. // 豆瓣评分 rating_score
  338. let rating_score = extractData(_doc, /<strong class="ll rating_num " property="v:average">([\s\S]+?)<\/strong>/gi);
  339. // 出版社 publisher
  340. let publisher = extractData(_doc, /<span class="pl">\s*出版社:?<\/span>\s*:?\s*<a[^>]+>([\s\S]+?)<\/a>/gi);
  341. if (!publisher) {
  342. publisher = extractData(_doc, /<span class="pl">\s*出版社:?<\/span>\s*:?\s*([\s\S]+?)<br\/?>/gi);
  343. }
  344. // 出品方 producer
  345. let producer = extractData(_doc, /<span class="pl">\s*出品方:?<\/span>\s*:?\s*<a[^>]+>([\s\S]+?)<\/a>/gi);
  346. if (!producer) {
  347. producer = extractData(_doc, /<span class="pl">\s*出品方:?<\/span>\s*:?\s*([\s\S]+?)<br\/?>/gi);
  348. }
  349. // 副标题 subtitle
  350. let subtitle = extractData(_doc, /<span class="pl">\s*副标题:?<\/span>\s*:?\s*([\s\S]+?)<br\/?>/gi);
  351. // 原作名 original_title
  352. let original_title = extractData(_doc, /<span class="pl">\s*原作名:?<\/span>\s*:?\s*([\s\S]+?)<br\/?>/gi);
  353. // 译者 translator
  354. let translator = extractData(_doc, /<span class="pl">\s*译者:?<\/span>\s*:?\s*<a[^>]+>([\s\S]+?)<\/a>/gi);
  355. if (!translator) {
  356. translator = extractData(_doc, /<span class="pl">\s*译者:?<\/span>\s*:?\s*([\s\S]+?)<br\/?>/gi);
  357. }
  358. // 出版年 published
  359. let published = extractData(_doc, /<span class="pl">\s*出版年:?<\/span>\s*:?\s*([\s\S]+?)<br\/?>/gi);
  360. // 页数 pages
  361. let pages = extractData(_doc, /<span class="pl">\s*页数:?<\/span>\s*:?\s*([\s\S]+?)<br\/?>/gi);
  362. // 定价 price
  363. let price = extractData(_doc, /<span class="pl">\s*定价:?<\/span>\s*:?\s*([\s\S]+?)<br\/?>/gi);
  364. // 装帧 binding
  365. let binding = extractData(_doc, /<span class="pl">\s*装帧:?<\/span>\s*:?\s*([\s\S]+?)<br\/?>/gi);
  366. // 丛书 series
  367. let series = extractData(_doc, /<span class="pl">\s*丛书:?<\/span>\s*:?\s*<a[^>]+>([\s\S]+?)<\/a>/gi);
  368. if (!series) {
  369. series = extractData(_doc, /<span class="pl">\s*丛书:?<\/span>\s*:?\s*([\s\S]+?)<br\/?>/gi);
  370. }
  371. // 内容简介 description
  372. let description = extractData(_doc, /<meta property="og:description" content="([^"]+?)"/gi);
  373. description = description.replace(/<[^>]+>|\n/g, "");
  374. // 封面图片 cover_url
  375. let cover_url = extractData(_doc, /<meta property="og:image" content="([^"]+?)"/gi);
  376. // 🚀🎉🎊🥳 图书元数据开放接口已上线,可前往 https://forms.gle/91z4wrtQngrbkK1g9 申请使用。
  377. const metadata = {
  378. isbn, rating_score, url, title, author, publisher, producer, subtitle, original_title, translator, published, pages, price, binding, series, description, cover_url
  379. };
  380. console.log(metadata);
  381.  
  382. queryWeread(isbn, title, subtitle, author, translator, publisher);
  383. queryDuokan(isbn, title, subtitle, author, translator, publisher);
  384. queryJingdong(isbn, title, subtitle, author, translator, publisher);
  385. queryDangdang(isbn, title, subtitle, author, translator, publisher);
  386. queryXimalaya(isbn, title, subtitle, author, translator, publisher);
  387. syncMetadata(isbn, metadata);
  388. };
  389. const extractData = (doc, regex) => {
  390. try {
  391. return regex.exec(doc)[1].trim();
  392. } catch(e) {
  393. console.log(e);
  394. return "";
  395. }
  396. };
  397.  
  398. // 微信读书页面主函数
  399. const wereadMain = () => {
  400. let vbookid = "";
  401. const locationHref = window.location.href;
  402. const match = locationHref.match(/(?:bookDetail|reader)\/([0-9a-zA-Z]+)/);
  403.  
  404. if (match && match[1].length <= 24) {
  405. vbookid = match[1];
  406. console.log(vbookid);
  407. } else {
  408. console.log('vbookid not match.');
  409. return;
  410. }
  411.  
  412. const handleResponse = (responseDetail) => {
  413. const result = JSON.parse(responseDetail.responseText);
  414. console.log(result);
  415. if (result.errmsg === "") {
  416. const { url, douban_rating_score, douban_rating_star } = result.data;
  417. const book_ratings_container = $(".book_ratings_container");
  418. const douban_info = `
  419. <div id="eba_douban_rating" class="book_ratings_header" style="margin-bottom:24px;cursor:pointer!important;">
  420. <a style="text-decoration:none!important;color:#1b88ee!important;" target="_blank" href="${url}">
  421. <span style="display:flex;align-items:center;">
  422. <img src="${base64_icon_douban}" style="display:inline-block;height:15px;">
  423. <span style="display:inline-block;height:24px;padding:0 4px;">豆瓣评分&nbsp;${douban_rating_score}&nbsp;</span>
  424. <span class="douban_rating ${douban_rating_star}"></span>
  425. </span>
  426. </a>
  427. </div>`;
  428. $("#eba_douban_rating").remove();
  429. book_ratings_container.prepend(douban_info);
  430. }
  431. };
  432.  
  433. GM_xmlhttpRequest ({
  434. method: "GET",
  435. url: `${REST_URL}/weread/douban_info?vbookid=${vbookid}&version=${version}&r=${Math.random()}`,
  436. headers: {
  437. "User-agent": window.navigator.userAgent,
  438. "X-Referer": window.location.href,
  439. "X-Unique-ID": x_unique_id
  440. },
  441. onload: handleResponse
  442. });
  443. };
  444.  
  445. // 主函数
  446. (() => {
  447. 'use strict';
  448. addNewStyle();
  449. const hostname = window.location.hostname;
  450. if (/book\.douban\.com/.test(hostname)) {
  451. doubanMain();
  452. } else if (/weread\.qq\.com/.test(hostname)) {
  453. setTimeout(() => wereadMain(), 100);
  454. }
  455. })();

QingJ © 2025

镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址