// ==UserScript==
// @name Clip Studio Reader Downloader
// @namespace http://tampermonkey.net/
// @version 1.9
// @description Download books from the browser version of Clip Studio Reader
// @author mrcoconuat
// @supportURL https://github.com/MrCocoNuat/clip-studio-reader-downloader/issues
// @match *://*/*
// @require https://unpkg.com/[email protected]/dist/jszip.js
// @require https://unpkg.com/[email protected]/dist/FileSaver.js
// @icon https://www.google.com/s2/favicons?sz=64&domain=mobilebook.jp
// @license MIT
// @grant none
// ==/UserScript==
'use strict';
// Site-blind Clip Studio Reader integration support:
//------------------------------
const downloadButtonId = "download-button";
const errorMessageId = "error-message";
const ELEMENT = {
SCREEN_CONTROLLER: 0, // used to flip pages
CURRENT_PAGE_COUNTER: 1, // duh
TOTAL_PAGE_COUNTER: 2, // duh
LOADER_SPINNER: 3, // used to detect if the reader is loading a page
MENU: 4, // used to detect if the menu must be raised since it contains the page scroller
PAGE_SPREAD: 5, // contains the actual pages, and is checked to see if the reader as a whole has loaded
PAGE_SLIDER: 7 // duh
}
const DOWNLOAD_MODE = {
PAGE_BY_PAGE: 0, // each page must be descrambled and rendered individually. Usually these are paid books
DIRECT: 1 // links to each page are accessible right from the start. Usually these are free samples or outright free books
}
// Data distribution for site-specific integrations:
//------------------------------
const siteSupport = {
"mbj-bs.pf.mobilebook.jp": {
mode: DOWNLOAD_MODE.PAGE_BY_PAGE,
ids: {
[ELEMENT.SCREEN_CONTROLLER]: "screen_surface",
[ELEMENT.CURRENT_PAGE_COUNTER]: "paging_slider_nombre_current",
[ELEMENT.TOTAL_PAGE_COUNTER]: "paging_slider_nombre_total",
[ELEMENT.LOADER_SPINNER]: "loading_spinner_layer",
[ELEMENT.MENU]: "menu_layer",
[ELEMENT.PAGE_SPREAD]: "spread_a",
[ELEMENT.PAGE_SLIDER]: "paging_slider"
},
// sometimes the necessary element does not have an id, which sucks
classes: {}
},
"api.distribution.mediadotech.com": {
mode: DOWNLOAD_MODE.PAGE_BY_PAGE,
ids: {
[ELEMENT.SCREEN_CONTROLLER]: "screen_control_pad",
[ELEMENT.CURRENT_PAGE_COUNTER]: "menu_nombre_current",
[ELEMENT.TOTAL_PAGE_COUNTER]: "menu_nombre_total",
[ELEMENT.LOADER_SPINNER]: "screen_loading_spinner_layer",
[ELEMENT.MENU]: "menu_container",
[ELEMENT.PAGE_SPREAD]: "screen_layer"
},
},
"comic-viewer.iowl.jp": {
mode: DOWNLOAD_MODE.PAGE_BY_PAGE,
ids: {
[ELEMENT.SCREEN_CONTROLLER]: "screen_control_pad",
[ELEMENT.CURRENT_PAGE_COUNTER]: "menu_nombre_current",
[ELEMENT.TOTAL_PAGE_COUNTER]: "menu_nombre_total",
[ELEMENT.LOADER_SPINNER]: "screen_loading_spinner_layer",
[ELEMENT.MENU]: "menu_footer",
[ELEMENT.PAGE_SPREAD]: "screen_layer",
},
},
"comic.pixiv.net": {
// readers can have extra conditions tacked on
condition: () => document.location.pathname.substring(0, 7) === "/viewer",
mode: DOWNLOAD_MODE.DIRECT,
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
classes: {
[ELEMENT.PAGE_SPREAD]: "h-screen",
},
},
"bs.comicdc.jp": {
mode: DOWNLOAD_MODE.PAGE_BY_PAGE,
ids: {
[ELEMENT.SCREEN_CONTROLLER]: "screen_control_pad",
[ELEMENT.CURRENT_PAGE_COUNTER]: "menu_nombre_current",
[ELEMENT.TOTAL_PAGE_COUNTER]: "menu_nombre_total",
[ELEMENT.LOADER_SPINNER]: "screen_loading_spinner_layer",
[ELEMENT.MENU]: "menu_container",
[ELEMENT.PAGE_SPREAD]: "screen_layer"
},
}
}
// try by ID if given, then by class
const getCSRElement = (elementEnum) => {
return document.getElementById(siteSupport[window.location.hostname].ids?.[elementEnum])
?? document.getElementsByClassName(siteSupport[window.location.hostname].classes?.[elementEnum])[0];
}
// returns a list of direct image links
function getCSRPageHrefs() {
// start from page 0, and go until null
const result = [];
for (let i = 0; /*blank*/; i++) {
const href = siteSupport[window.location.hostname].href(i);
if (href === undefined) {
break;
}
result.push(href);
}
return result;
}
function siteIsSupported() {
return !!siteSupport[document.location.hostname] && (siteSupport[document.location.hostname].condition?.() ?? true);
}
function downloadMode() {
return siteSupport[document.location.hostname].mode;
}
init();
// SVG Handling
// Convert SVG to image (JPEG, PNG, etc.) in the browser
// Thanks, Thom Kiesewetter
// https://stackoverflow.com/a/58142441
//------------------------------
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">
<defs>
<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"/>
<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"/>
</defs>
<g fill="none" fill-rule="evenodd" transform="translate(2 3)">
<g transform="translate(2 6)">
<mask id="download-b" fill="#ffffff">
<use xlink:href="#download-a"/>
</mask>
<use fill="#D8D8D8" fill-rule="nonzero" xlink:href="#download-a"/>
<g fill="#FFA0A0" mask="url(#download-b)">
<rect width="24" height="24" transform="translate(-4 -9)"/>
</g>
</g>
<mask id="download-d" fill="#ffffff">
<use xlink:href="#download-c"/>
</mask>
<use fill="#000000" fill-rule="nonzero" xlink:href="#download-c"/>
<g fill="#7600FF" mask="url(#download-d)">
<rect width="24" height="24" transform="translate(-2 -3)"/>
</g>
</g>
</svg>`
function svgToPng(svg, callback) {
const url = getSvgUrl(svg);
svgUrlToPng(url, (imgData) => {
callback(imgData);
URL.revokeObjectURL(url);
});
}
function getSvgUrl(svg) {
return URL.createObjectURL(new Blob([svg], {type: 'image/svg+xml'}));
}
function svgUrlToPng(svgUrl, callback) {
const svgImage = document.createElement('img');
// can't be display none, but also don't take up space
svgImage.style.position = "fixed";
svgImage.style.visibility = "hidden";
document.body.appendChild(svgImage);
svgImage.onload = function () {
const canvas = document.createElement('canvas');
canvas.width = svgImage.clientWidth;
canvas.height = svgImage.clientHeight;
const canvasCtx = canvas.getContext('2d');
canvasCtx.drawImage(svgImage, 0, 0);
const imgData = canvas.toDataURL('image/png');
callback(imgData);
// document.body.removeChild(imgPreview);
};
svgImage.src = svgUrl;
}
// Utility
//------------------------------
function error(str) {
console.error(`[CSRD]: ${str}`);
}
function log(str) {
console.log(`[CSRD]: ${str}`);
}
function info(str) {
console.info(`[CSRD]: ${str}`);
}
function debug(str) {
console.debug(`[CSRD]: ${str}`);
}
async function sleep(ms) {
await new Promise(r => setTimeout(r, ms)); // give the browser a break - idle wait
}
// How to get the browser viewport dimensions?
// Thanks, ryanve
// https://stackoverflow.com/a/8876069
function viewportX() {
return Math.max(document.documentElement.clientWidth || 0, window.innerWidth || 0);
}
function viewportY() {
return Math.max(document.documentElement.clientHeight || 0, window.innerHeight || 0);
}
function digitCount(num) {
return num.toString().length;
}
function dataUrlToData(base64Url) {
return base64Url.substr(base64Url.indexOf(',') + 1);
}
// Reader Utility
//------------------------------
// positive flips forward (increase page number), negative flips backwards (decrease page number), 0 is to open menu
function flipPage(direction) {
const screen = getCSRElement(ELEMENT.SCREEN_CONTROLLER);
// click on the left, middle, or right of the screen depending on arg
const x = (-direction + 1) * viewportX() / 2;
screen.dispatchEvent(new PointerEvent("pointerdown", {buttons: 1, clientX: x, clientY: 100, bubbles: true}));
screen.dispatchEvent(new PointerEvent("pointerup", {buttons: 0, clientX: x, clientY: 100, bubbles: true}));
}
async function waitForPageLoad() {
while (isLoadingPage()) {
await sleep(100);
}
}
function currentPage() {
return +getCSRElement(ELEMENT.CURRENT_PAGE_COUNTER).textContent;
}
function totalPageCount() {
return +getCSRElement(ELEMENT.TOTAL_PAGE_COUNTER).textContent;
}
function isLoadingPage() {
return getCSRElement(ELEMENT.LOADER_SPINNER).classList.contains("onstage");
}
function isMenuOpen() {
return getCSRElement(ELEMENT.MENU).style.display !== "none" && getCSRElement(ELEMENT.MENU).classList.contains("onstage");
}
async function flipToFirstPage() {
const slider = getCSRElement(ELEMENT.PAGE_SLIDER);
if (slider === undefined && currentPage() !== 1) {
throw Error("This reader's automatic page slider is not supported, please move to page 1 manually and click the download button again");
}
if (currentPage() === 1) {
info(`already on first page`);
return;
}
info(`flipping to first page`);
if (!isMenuOpen()) {
debug("opening menu to load scroller");
flipPage(0); // open menu
await sleep(200);
}
// click the very right of the page slider - page 1
slider.dispatchEvent(new PointerEvent("pointerdown", {
buttons: 1,
clientX: viewportX() - 25,
clientY: viewportY() - 70,
bubbles: true
}));
slider.dispatchEvent(new PointerEvent("pointerup", {
buttons: 0,
clientX: viewportX() - 25,
clientY: viewportY() - 70,
bubbles: true
}));
await sleep(100);
await waitForPageLoad();
}
// Main
//------------------------------
async function generatePageByPageZip() {
const jsZip = new JSZip();
let totalPages = totalPageCount();
if (totalPages === 0) {
throw Error("The total number of pages reported by the reader is 0, the page slider seems to have been opened incorrectly");
}
info(`there are ${totalPages} pages in total to save`);
const zeroPads = digitCount(totalPages);
let pageNumber = 1;
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
// 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!)
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
log("==== downloading pages: ====");
while (true) {
const element_spread = getCSRElement(ELEMENT.PAGE_SPREAD);
// Right to left means reverse the array
for (const canvas of [...element_spread.children].toReversed()) {
if (canvas.style.visibility === "hidden" || canvas.style.display === "none") {
debug(`page before ${pageNumber} is a hidden or junk page, skipping it`);
//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
expectedPagesInSpread--;
continue;
}
info(`saving page ${pageNumber}`);
jsZip.file(`${pageNumber.toString().padStart(zeroPads, "0")}.png`, dataUrlToData(canvas.toDataURL()), {base64: true});
pageNumber++;
}
if (pageNumber > totalPages) {
break;
}
// Need to check after this canvas spread is completed and the page is flipped, what to do
lastReportedReaderPageNumber = currentPage();
await sleep(100);
flipPage(1);
await sleep(100);
await waitForPageLoad();
debug(`currentPage() - lastReportedReaderPageNumber = ${currentPage()} - ${lastReportedReaderPageNumber}`);
if (currentPage() - lastReportedReaderPageNumber > expectedPagesInSpread) {
// there are less pages than we thought! - the reader reports that currentPage() - lastDownloadedPageNumber passed (usually 2)
// 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
totalPages -= (currentPage() - lastReportedReaderPageNumber) - expectedPagesInSpread;
debug(`cutting ${(currentPage() - lastReportedReaderPageNumber) - expectedPagesInSpread} from total pages, now ${totalPages}`)
}
expectedPagesInSpread = 2; // reset this
await sleep(100);
}
return jsZip;
}
async function generateDirectZip() {
const jsZip = new JSZip();
const pageHrefs = getCSRPageHrefs();
const totalPages = pageHrefs.length;
info(`there are ${totalPages} pages in total to save`);
const zeroPads = digitCount(totalPages);
log("==== downloading pages: ====");
for (const [i, href] of pageHrefs.entries()) {
jsZip.file(`${i.toString().padStart(zeroPads, "0")}.png`, await (await fetch(href)).blob());
}
return jsZip;
}
function injectErrorMessage(message) {
if (document.getElementById(errorMessageId) !== null) {
document.getElementById(errorMessageId).remove();
}
const div = document.createElement("div");
div.id = errorMessageId;
div.style["position"] = "fixed";
div.style["border-radius"] = "5%";
div.style["z-index"] = 900;
div.style["bottom"] = "150px";
div.style["right"] = "32px";
div.style["background"] = "black";
div.style["border"] = "3px solid purple";
div.style["padding"] = "8px";
div.style["cursor"] = "pointer";
div.style["color"] = "red";
document.body.appendChild(div);
const text = document.createTextNode(message);
div.appendChild(text);
}
async function downloadBookAsZip() {
let jsZip;
try {
switch (downloadMode()) {
case DOWNLOAD_MODE.PAGE_BY_PAGE:
if (!isMenuOpen()) {
info("opening menu to load scroller");
flipPage(0); // open menu
await sleep(200);
}
await flipToFirstPage();
jsZip = await generatePageByPageZip();
break;
case DOWNLOAD_MODE.DIRECT:
jsZip = await generateDirectZip();
break;
}
} catch (exception) {
injectErrorMessage(exception.message);
throw exception;
}
log("generating zip file, rename it however you like - this script cannot figure out the book's name");
await jsZip.generateAsync({type: "blob"}).then(blob => saveAs(blob, "Clip_Studio_Reader_Downloader_RENAME_ME.zip"));
log("==== all done! Enjoy your book ^_^ ====");
}
function injectDownloadButton() {
log("reader is loaded, download button injected");
const parent = document.body;
const div = document.createElement("div");
parent.appendChild(div);
div.id = downloadButtonId;
// not all sites use tailwind
div.style["position"] = "fixed";
div.style["border-radius"] = "50%";
div.style["z-index"] = 900;
div.style["bottom"] = "32px";
div.style["right"] = "32px";
div.style["background"] = "black";
div.style["border"] = "3px solid purple";
div.style["padding"] = "8px";
div.style["cursor"] = "pointer";
div.addEventListener("pointerdown", downloadBookAsZip);
svgToPng(downloadSvg, (imgData) => {
const image = document.createElement('img');
image.style.height = "64px";
div.appendChild(image);
image.src = imgData;
});
}
function checkReaderLoad(observer, timeoutId) {
if (currentPage() !== 0) {
observer.disconnect();
// stop the 30 second timeout
clearTimeout(timeoutId);
injectDownloadButton();
}
}
// Userscript to wait for page to load before executing code techniques?
// Thanks, goweon
// https://stackoverflow.com/a/47406751
function checkPageLoad(observer) {
if (getCSRElement(ELEMENT.PAGE_SPREAD)) {
observer.disconnect();
log("==== Clip Studio Reader Downloader ====");
log("https://github.com/MrCocoNuat/clip-studio-reader-downloader");
log("waiting up to 30 seconds for reader to load");
switch (downloadMode()) {
case DOWNLOAD_MODE.PAGE_BY_PAGE:
let readerObserver;
const timeoutId = setTimeout(async () => {
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?");
readerObserver.disconnect();
}, 30000);
readerObserver = new MutationObserver((changes, innerObserver) => checkReaderLoad(innerObserver, timeoutId));
readerObserver.observe(getCSRElement(ELEMENT.CURRENT_PAGE_COUNTER), {childList: true, subtree: true});
break;
case DOWNLOAD_MODE.DIRECT:
injectDownloadButton(); // not much to wait for
break;
}
}
}
function init() {
if (siteIsSupported()) {
(new MutationObserver((changes, observer) => checkPageLoad(observer))).observe(document, {
childList: true,
subtree: true
});
} else {
log("No instance of Clip Studio Reader found on the current page");
}
}