Clip Studio Reader Downloader

Download books from the browser version of Clip Studio Reader

  1. // ==UserScript==
  2. // @name Clip Studio Reader Downloader
  3. // @namespace http://tampermonkey.net/
  4. // @version 1.9
  5. // @description Download books from the browser version of Clip Studio Reader
  6. // @author mrcoconuat
  7. // @supportURL https://github.com/MrCocoNuat/clip-studio-reader-downloader/issues
  8. // @match *://*/*
  9. // @require https://unpkg.com/jszip@3.10.1/dist/jszip.js
  10. // @require https://unpkg.com/file-saver@2.0.5/dist/FileSaver.js
  11. // @icon https://www.google.com/s2/favicons?sz=64&domain=mobilebook.jp
  12. // @license MIT
  13. // @grant none
  14. // ==/UserScript==
  15.  
  16. 'use strict';
  17.  
  18. // Site-blind Clip Studio Reader integration support:
  19. //------------------------------
  20.  
  21. const downloadButtonId = "download-button";
  22. const errorMessageId = "error-message";
  23.  
  24. const ELEMENT = {
  25. SCREEN_CONTROLLER: 0, // used to flip pages
  26. CURRENT_PAGE_COUNTER: 1, // duh
  27. TOTAL_PAGE_COUNTER: 2, // duh
  28. LOADER_SPINNER: 3, // used to detect if the reader is loading a page
  29. MENU: 4, // used to detect if the menu must be raised since it contains the page scroller
  30. PAGE_SPREAD: 5, // contains the actual pages, and is checked to see if the reader as a whole has loaded
  31. PAGE_SLIDER: 7 // duh
  32. }
  33.  
  34. const DOWNLOAD_MODE = {
  35. PAGE_BY_PAGE: 0, // each page must be descrambled and rendered individually. Usually these are paid books
  36. DIRECT: 1 // links to each page are accessible right from the start. Usually these are free samples or outright free books
  37. }
  38.  
  39. // Data distribution for site-specific integrations:
  40. //------------------------------
  41.  
  42. const siteSupport = {
  43. "mbj-bs.pf.mobilebook.jp": {
  44. mode: DOWNLOAD_MODE.PAGE_BY_PAGE,
  45. ids: {
  46. [ELEMENT.SCREEN_CONTROLLER]: "screen_surface",
  47. [ELEMENT.CURRENT_PAGE_COUNTER]: "paging_slider_nombre_current",
  48. [ELEMENT.TOTAL_PAGE_COUNTER]: "paging_slider_nombre_total",
  49. [ELEMENT.LOADER_SPINNER]: "loading_spinner_layer",
  50. [ELEMENT.MENU]: "menu_layer",
  51. [ELEMENT.PAGE_SPREAD]: "spread_a",
  52. [ELEMENT.PAGE_SLIDER]: "paging_slider"
  53. },
  54. // sometimes the necessary element does not have an id, which sucks
  55. classes: {}
  56. },
  57. "api.distribution.mediadotech.com": {
  58. mode: DOWNLOAD_MODE.PAGE_BY_PAGE,
  59. ids: {
  60. [ELEMENT.SCREEN_CONTROLLER]: "screen_control_pad",
  61. [ELEMENT.CURRENT_PAGE_COUNTER]: "menu_nombre_current",
  62. [ELEMENT.TOTAL_PAGE_COUNTER]: "menu_nombre_total",
  63. [ELEMENT.LOADER_SPINNER]: "screen_loading_spinner_layer",
  64. [ELEMENT.MENU]: "menu_container",
  65. [ELEMENT.PAGE_SPREAD]: "screen_layer"
  66. },
  67. },
  68. "comic-viewer.iowl.jp": {
  69. mode: DOWNLOAD_MODE.PAGE_BY_PAGE,
  70. ids: {
  71. [ELEMENT.SCREEN_CONTROLLER]: "screen_control_pad",
  72. [ELEMENT.CURRENT_PAGE_COUNTER]: "menu_nombre_current",
  73. [ELEMENT.TOTAL_PAGE_COUNTER]: "menu_nombre_total",
  74. [ELEMENT.LOADER_SPINNER]: "screen_loading_spinner_layer",
  75. [ELEMENT.MENU]: "menu_footer",
  76. [ELEMENT.PAGE_SPREAD]: "screen_layer",
  77. },
  78. },
  79. "comic.pixiv.net": {
  80. // readers can have extra conditions tacked on
  81. condition: () => document.location.pathname.substring(0, 7) === "/viewer",
  82. mode: DOWNLOAD_MODE.DIRECT,
  83. href: (pageNumber) => document.getElementById(`page-${pageNumber}`)?.style["background-image"].match(/url\("(.*)"\)/)?.[1], // TODO: these urls are not all loaded at start. So clicking it too fast will lose some ending pages
  84. classes: {
  85. [ELEMENT.PAGE_SPREAD]: "h-screen",
  86. },
  87. },
  88. "bs.comicdc.jp": {
  89. mode: DOWNLOAD_MODE.PAGE_BY_PAGE,
  90. ids: {
  91. [ELEMENT.SCREEN_CONTROLLER]: "screen_control_pad",
  92. [ELEMENT.CURRENT_PAGE_COUNTER]: "menu_nombre_current",
  93. [ELEMENT.TOTAL_PAGE_COUNTER]: "menu_nombre_total",
  94. [ELEMENT.LOADER_SPINNER]: "screen_loading_spinner_layer",
  95. [ELEMENT.MENU]: "menu_container",
  96. [ELEMENT.PAGE_SPREAD]: "screen_layer"
  97. },
  98. }
  99. }
  100.  
  101. // try by ID if given, then by class
  102. const getCSRElement = (elementEnum) => {
  103. return document.getElementById(siteSupport[window.location.hostname].ids?.[elementEnum])
  104. ?? document.getElementsByClassName(siteSupport[window.location.hostname].classes?.[elementEnum])[0];
  105. }
  106.  
  107. // returns a list of direct image links
  108. function getCSRPageHrefs() {
  109. // start from page 0, and go until null
  110. const result = [];
  111. for (let i = 0; /*blank*/; i++) {
  112. const href = siteSupport[window.location.hostname].href(i);
  113. if (href === undefined) {
  114. break;
  115. }
  116. result.push(href);
  117. }
  118. return result;
  119. }
  120.  
  121. function siteIsSupported() {
  122. return !!siteSupport[document.location.hostname] && (siteSupport[document.location.hostname].condition?.() ?? true);
  123. }
  124.  
  125. function downloadMode() {
  126. return siteSupport[document.location.hostname].mode;
  127. }
  128.  
  129.  
  130. init();
  131.  
  132.  
  133. // SVG Handling
  134. // Convert SVG to image (JPEG, PNG, etc.) in the browser
  135. // Thanks, Thom Kiesewetter
  136. // https://stackoverflow.com/a/58142441
  137. //------------------------------
  138.  
  139. const downloadSvg = `<svg width="800px" height="800px" viewBox="0 0 24 24" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink">
  140. <defs>
  141. <path id="download-a" d="M4.29289322,1.70710678 C3.90236893,1.31658249 3.90236893,0.683417511 4.29289322,0.292893219 C4.68341751,-0.0976310729 5.31658249,-0.0976310729 5.70710678,0.292893219 L7.70710678,2.29289322 C8.09763107,2.68341751 8.09763107,3.31658249 7.70710678,3.70710678 C7.31658249,4.09763107 6.68341751,4.09763107 6.29289322,3.70710678 L4.29289322,1.70710678 Z M0,8 L16,8 L16,10 L0,10 L0,8 Z"/>
  142. <path id="download-c" d="M11,9.58578644 L13.2928932,7.29289322 C13.6834175,6.90236893 14.3165825,6.90236893 14.7071068,7.29289322 C15.0976311,7.68341751 15.0976311,8.31658249 14.7071068,8.70710678 L10.7071068,12.7071068 C10.3165825,13.0976311 9.68341751,13.0976311 9.29289322,12.7071068 L5.29289322,8.70710678 C4.90236893,8.31658249 4.90236893,7.68341751 5.29289322,7.29289322 C5.68341751,6.90236893 6.31658249,6.90236893 6.70710678,7.29289322 L9,9.58578644 L9,0.998529185 C9,0.447056744 9.44771525,-7.95978809e-15 10,-7.99360578e-15 C10.5522847,-8.02742346e-15 11,0.447056744 11,0.998529185 L11,9.58578644 Z M18,16 L18,10 C18,9.44771525 18.4477153,9 19,9 C19.5522847,9 20,9.44771525 20,10 L20,17 C20,17.5522847 19.5522847,18 19,18 L1,18 C0.44771525,18 0,17.5522847 0,17 L0,10 C0,9.44771525 0.44771525,9 1,9 C1.55228475,9 2,9.44771525 2,10 L2,16 L18,16 Z"/>
  143. </defs>
  144. <g fill="none" fill-rule="evenodd" transform="translate(2 3)">
  145. <g transform="translate(2 6)">
  146. <mask id="download-b" fill="#ffffff">
  147. <use xlink:href="#download-a"/>
  148. </mask>
  149. <use fill="#D8D8D8" fill-rule="nonzero" xlink:href="#download-a"/>
  150. <g fill="#FFA0A0" mask="url(#download-b)">
  151. <rect width="24" height="24" transform="translate(-4 -9)"/>
  152. </g>
  153. </g>
  154. <mask id="download-d" fill="#ffffff">
  155. <use xlink:href="#download-c"/>
  156. </mask>
  157. <use fill="#000000" fill-rule="nonzero" xlink:href="#download-c"/>
  158. <g fill="#7600FF" mask="url(#download-d)">
  159. <rect width="24" height="24" transform="translate(-2 -3)"/>
  160. </g>
  161. </g>
  162. </svg>`
  163.  
  164. function svgToPng(svg, callback) {
  165. const url = getSvgUrl(svg);
  166. svgUrlToPng(url, (imgData) => {
  167. callback(imgData);
  168. URL.revokeObjectURL(url);
  169. });
  170. }
  171.  
  172. function getSvgUrl(svg) {
  173. return URL.createObjectURL(new Blob([svg], {type: 'image/svg+xml'}));
  174. }
  175.  
  176. function svgUrlToPng(svgUrl, callback) {
  177. const svgImage = document.createElement('img');
  178. // can't be display none, but also don't take up space
  179. svgImage.style.position = "fixed";
  180. svgImage.style.visibility = "hidden";
  181. document.body.appendChild(svgImage);
  182. svgImage.onload = function () {
  183. const canvas = document.createElement('canvas');
  184. canvas.width = svgImage.clientWidth;
  185. canvas.height = svgImage.clientHeight;
  186. const canvasCtx = canvas.getContext('2d');
  187. canvasCtx.drawImage(svgImage, 0, 0);
  188. const imgData = canvas.toDataURL('image/png');
  189. callback(imgData);
  190. // document.body.removeChild(imgPreview);
  191. };
  192. svgImage.src = svgUrl;
  193. }
  194.  
  195.  
  196. // Utility
  197. //------------------------------
  198.  
  199. function error(str) {
  200. console.error(`[CSRD]: ${str}`);
  201. }
  202.  
  203. function log(str) {
  204. console.log(`[CSRD]: ${str}`);
  205. }
  206.  
  207. function info(str) {
  208. console.info(`[CSRD]: ${str}`);
  209. }
  210.  
  211. function debug(str) {
  212. console.debug(`[CSRD]: ${str}`);
  213. }
  214.  
  215. async function sleep(ms) {
  216. await new Promise(r => setTimeout(r, ms)); // give the browser a break - idle wait
  217. }
  218.  
  219. // How to get the browser viewport dimensions?
  220. // Thanks, ryanve
  221. // https://stackoverflow.com/a/8876069
  222. function viewportX() {
  223. return Math.max(document.documentElement.clientWidth || 0, window.innerWidth || 0);
  224. }
  225.  
  226. function viewportY() {
  227. return Math.max(document.documentElement.clientHeight || 0, window.innerHeight || 0);
  228. }
  229.  
  230. function digitCount(num) {
  231. return num.toString().length;
  232. }
  233.  
  234. function dataUrlToData(base64Url) {
  235. return base64Url.substr(base64Url.indexOf(',') + 1);
  236. }
  237.  
  238.  
  239. // Reader Utility
  240. //------------------------------
  241.  
  242. // positive flips forward (increase page number), negative flips backwards (decrease page number), 0 is to open menu
  243. function flipPage(direction) {
  244. const screen = getCSRElement(ELEMENT.SCREEN_CONTROLLER);
  245. // click on the left, middle, or right of the screen depending on arg
  246. const x = (-direction + 1) * viewportX() / 2;
  247. screen.dispatchEvent(new PointerEvent("pointerdown", {buttons: 1, clientX: x, clientY: 100, bubbles: true}));
  248. screen.dispatchEvent(new PointerEvent("pointerup", {buttons: 0, clientX: x, clientY: 100, bubbles: true}));
  249. }
  250.  
  251. async function waitForPageLoad() {
  252. while (isLoadingPage()) {
  253. await sleep(100);
  254. }
  255. }
  256.  
  257. function currentPage() {
  258. return +getCSRElement(ELEMENT.CURRENT_PAGE_COUNTER).textContent;
  259. }
  260.  
  261. function totalPageCount() {
  262. return +getCSRElement(ELEMENT.TOTAL_PAGE_COUNTER).textContent;
  263. }
  264.  
  265. function isLoadingPage() {
  266. return getCSRElement(ELEMENT.LOADER_SPINNER).classList.contains("onstage");
  267. }
  268.  
  269. function isMenuOpen() {
  270. return getCSRElement(ELEMENT.MENU).style.display !== "none" && getCSRElement(ELEMENT.MENU).classList.contains("onstage");
  271. }
  272.  
  273. async function flipToFirstPage() {
  274. const slider = getCSRElement(ELEMENT.PAGE_SLIDER);
  275. if (slider === undefined && currentPage() !== 1) {
  276. throw Error("This reader's automatic page slider is not supported, please move to page 1 manually and click the download button again");
  277. }
  278. if (currentPage() === 1) {
  279. info(`already on first page`);
  280. return;
  281. }
  282.  
  283. info(`flipping to first page`);
  284.  
  285. if (!isMenuOpen()) {
  286. debug("opening menu to load scroller");
  287. flipPage(0); // open menu
  288. await sleep(200);
  289. }
  290.  
  291. // click the very right of the page slider - page 1
  292. slider.dispatchEvent(new PointerEvent("pointerdown", {
  293. buttons: 1,
  294. clientX: viewportX() - 25,
  295. clientY: viewportY() - 70,
  296. bubbles: true
  297. }));
  298. slider.dispatchEvent(new PointerEvent("pointerup", {
  299. buttons: 0,
  300. clientX: viewportX() - 25,
  301. clientY: viewportY() - 70,
  302. bubbles: true
  303. }));
  304. await sleep(100);
  305. await waitForPageLoad();
  306. }
  307.  
  308.  
  309. // Main
  310. //------------------------------
  311.  
  312. async function generatePageByPageZip() {
  313. const jsZip = new JSZip();
  314.  
  315. let totalPages = totalPageCount();
  316. if (totalPages === 0) {
  317. throw Error("The total number of pages reported by the reader is 0, the page slider seems to have been opened incorrectly");
  318. }
  319.  
  320. info(`there are ${totalPages} pages in total to save`);
  321. const zeroPads = digitCount(totalPages);
  322.  
  323. let pageNumber = 1;
  324. let lastReportedReaderPageNumber; // this helps distinguish a 1 page spread on 2 canvases, vs the reader always rendering 1 page only because the viewport is too narrow
  325. // any time the spread is not 2 canvases, this will no longer match pageNumber. But that is ok, we prioritize continuity of pageNumber instead (e.g. page 01, 02, 04 is bad!)
  326. let expectedPagesInSpread = 2; // a 2 page spread is normal. But we have to handle when the viewport is narrow enough that only 1 is shown
  327.  
  328. log("==== downloading pages: ====");
  329.  
  330. while (true) {
  331. const element_spread = getCSRElement(ELEMENT.PAGE_SPREAD);
  332. // Right to left means reverse the array
  333. for (const canvas of [...element_spread.children].toReversed()) {
  334. if (canvas.style.visibility === "hidden" || canvas.style.display === "none") {
  335. debug(`page before ${pageNumber} is a hidden or junk page, skipping it`);
  336. //possibly the unseen half of a 1 page spread on 2 canvases. Need to check after this canvas spread is completed and the page is flipped, what to do
  337. expectedPagesInSpread--;
  338. continue;
  339. }
  340.  
  341. info(`saving page ${pageNumber}`);
  342. jsZip.file(`${pageNumber.toString().padStart(zeroPads, "0")}.png`, dataUrlToData(canvas.toDataURL()), {base64: true});
  343. pageNumber++;
  344. }
  345.  
  346. if (pageNumber > totalPages) {
  347. break;
  348. }
  349.  
  350. // Need to check after this canvas spread is completed and the page is flipped, what to do
  351. lastReportedReaderPageNumber = currentPage();
  352. await sleep(100);
  353. flipPage(1);
  354. await sleep(100);
  355. await waitForPageLoad();
  356.  
  357. debug(`currentPage() - lastReportedReaderPageNumber = ${currentPage()} - ${lastReportedReaderPageNumber}`);
  358. if (currentPage() - lastReportedReaderPageNumber > expectedPagesInSpread) {
  359. // there are less pages than we thought! - the reader reports that currentPage() - lastDownloadedPageNumber passed (usually 2)
  360. // but we only saw expectedPages and incremented pageNumber by that much (usually 1 when this branch is entered)! decrement totalPages so we don't run over the totalPages with pageNumber
  361. totalPages -= (currentPage() - lastReportedReaderPageNumber) - expectedPagesInSpread;
  362. debug(`cutting ${(currentPage() - lastReportedReaderPageNumber) - expectedPagesInSpread} from total pages, now ${totalPages}`)
  363. }
  364. expectedPagesInSpread = 2; // reset this
  365.  
  366. await sleep(100);
  367. }
  368.  
  369. return jsZip;
  370. }
  371.  
  372. async function generateDirectZip() {
  373. const jsZip = new JSZip();
  374. const pageHrefs = getCSRPageHrefs();
  375. const totalPages = pageHrefs.length;
  376. info(`there are ${totalPages} pages in total to save`);
  377. const zeroPads = digitCount(totalPages);
  378. log("==== downloading pages: ====");
  379.  
  380. for (const [i, href] of pageHrefs.entries()) {
  381. jsZip.file(`${i.toString().padStart(zeroPads, "0")}.png`, await (await fetch(href)).blob());
  382. }
  383.  
  384. return jsZip;
  385. }
  386.  
  387. function injectErrorMessage(message) {
  388. if (document.getElementById(errorMessageId) !== null) {
  389. document.getElementById(errorMessageId).remove();
  390. }
  391.  
  392. const div = document.createElement("div");
  393. div.id = errorMessageId;
  394. div.style["position"] = "fixed";
  395. div.style["border-radius"] = "5%";
  396. div.style["z-index"] = 900;
  397. div.style["bottom"] = "150px";
  398. div.style["right"] = "32px";
  399. div.style["background"] = "black";
  400. div.style["border"] = "3px solid purple";
  401. div.style["padding"] = "8px";
  402. div.style["cursor"] = "pointer";
  403. div.style["color"] = "red";
  404. document.body.appendChild(div);
  405. const text = document.createTextNode(message);
  406. div.appendChild(text);
  407. }
  408.  
  409. async function downloadBookAsZip() {
  410. let jsZip;
  411. try {
  412. switch (downloadMode()) {
  413. case DOWNLOAD_MODE.PAGE_BY_PAGE:
  414. if (!isMenuOpen()) {
  415. info("opening menu to load scroller");
  416. flipPage(0); // open menu
  417. await sleep(200);
  418. }
  419. await flipToFirstPage();
  420.  
  421. jsZip = await generatePageByPageZip();
  422. break;
  423. case DOWNLOAD_MODE.DIRECT:
  424. jsZip = await generateDirectZip();
  425. break;
  426. }
  427. } catch (exception) {
  428. injectErrorMessage(exception.message);
  429. throw exception;
  430. }
  431.  
  432. log("generating zip file, rename it however you like - this script cannot figure out the book's name");
  433. await jsZip.generateAsync({type: "blob"}).then(blob => saveAs(blob, "Clip_Studio_Reader_Downloader_RENAME_ME.zip"));
  434. log("==== all done! Enjoy your book ^_^ ====");
  435.  
  436. }
  437.  
  438. function injectDownloadButton() {
  439. log("reader is loaded, download button injected");
  440. const parent = document.body;
  441. const div = document.createElement("div");
  442. parent.appendChild(div);
  443. div.id = downloadButtonId;
  444. // not all sites use tailwind
  445. div.style["position"] = "fixed";
  446. div.style["border-radius"] = "50%";
  447. div.style["z-index"] = 900;
  448. div.style["bottom"] = "32px";
  449. div.style["right"] = "32px";
  450. div.style["background"] = "black";
  451. div.style["border"] = "3px solid purple";
  452. div.style["padding"] = "8px";
  453. div.style["cursor"] = "pointer";
  454. div.addEventListener("pointerdown", downloadBookAsZip);
  455. svgToPng(downloadSvg, (imgData) => {
  456. const image = document.createElement('img');
  457. image.style.height = "64px";
  458. div.appendChild(image);
  459. image.src = imgData;
  460. });
  461. }
  462.  
  463. function checkReaderLoad(observer, timeoutId) {
  464. if (currentPage() !== 0) {
  465. observer.disconnect();
  466. // stop the 30 second timeout
  467. clearTimeout(timeoutId);
  468. injectDownloadButton();
  469. }
  470. }
  471.  
  472. // Userscript to wait for page to load before executing code techniques?
  473. // Thanks, goweon
  474. // https://stackoverflow.com/a/47406751
  475. function checkPageLoad(observer) {
  476. if (getCSRElement(ELEMENT.PAGE_SPREAD)) {
  477. observer.disconnect();
  478. log("==== Clip Studio Reader Downloader ====");
  479. log("https://github.com/MrCocoNuat/clip-studio-reader-downloader");
  480. log("waiting up to 30 seconds for reader to load");
  481.  
  482. switch (downloadMode()) {
  483. case DOWNLOAD_MODE.PAGE_BY_PAGE:
  484. let readerObserver;
  485. const timeoutId = setTimeout(async () => {
  486. error("ERR: reader load timeout. the reader seems to have started incorrectly or the reader may have taken too long to load - do you need to reopen the book?");
  487. readerObserver.disconnect();
  488. }, 30000);
  489. readerObserver = new MutationObserver((changes, innerObserver) => checkReaderLoad(innerObserver, timeoutId));
  490. readerObserver.observe(getCSRElement(ELEMENT.CURRENT_PAGE_COUNTER), {childList: true, subtree: true});
  491. break;
  492. case DOWNLOAD_MODE.DIRECT:
  493. injectDownloadButton(); // not much to wait for
  494. break;
  495. }
  496. }
  497. }
  498.  
  499.  
  500. function init() {
  501. if (siteIsSupported()) {
  502. (new MutationObserver((changes, observer) => checkPageLoad(observer))).observe(document, {
  503. childList: true,
  504. subtree: true
  505. });
  506. } else {
  507. log("No instance of Clip Studio Reader found on the current page");
  508. }
  509. }