Clip Studio Reader Downloader

Download books from the browser version of Clip Studio Reader

目前為 2024-09-16 提交的版本,檢視 最新版本

您需要先安裝使用者腳本管理器擴展,如 TampermonkeyGreasemonkeyViolentmonkey 之後才能安裝該腳本。

You will need to install an extension such as Tampermonkey to install this script.

您需要先安裝使用者腳本管理器擴充功能,如 TampermonkeyViolentmonkey 後才能安裝該腳本。

您需要先安裝使用者腳本管理器擴充功能,如 TampermonkeyUserscripts 後才能安裝該腳本。

你需要先安裝一款使用者腳本管理器擴展,比如 Tampermonkey,才能安裝此腳本

您需要先安裝使用者腳本管理器擴充功能後才能安裝該腳本。

(我已經安裝了使用者腳本管理器,讓我安裝!)

你需要先安裝一款使用者樣式管理器擴展,比如 Stylus,才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展,比如 Stylus,才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展,比如 Stylus,才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展後才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展後才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展後才能安裝此樣式

(我已經安裝了使用者樣式管理器,讓我安裝!)

// ==UserScript==
// @name         Clip Studio Reader Downloader
// @namespace    http://tampermonkey.net/
// @version      1.8
// @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",
        },
    }
}

// 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");
    }
}