Clip Studio Reader Downloader

Download books from the browser version of Clip Studio Reader

// ==UserScript==
// @name         Clip Studio Reader Downloader
// @namespace    http://tampermonkey.net/
// @version      1.10
// @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 warningMessagesId = "warning-messages";

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
    COLOPHON: 8 // used to detect if the colophon page is being shown, usually because the reader has reached the end of the book
}

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: {}
    },
    "mbj-bs2.pf.mobilebook.jp": {
        mode: DOWNLOAD_MODE.PAGE_BY_PAGE,
        ids: {
            [ELEMENT.CURRENT_PAGE_COUNTER]: "menu_footer_nombre_progress",
            [ELEMENT.MENU]: "menu_container",
            [ELEMENT.PAGE_SPREAD]: { // more complicated structure is needed
                id: "ct-main-viewer",
                path: "children.1.children.0" // the page spread does not have an ID, so we have to go through the react structure >:(
            },
            // [ELEMENT.PAGE_SLIDER]: "menu_pagination_slider", // not working right, make the user drag it manually
            [ELEMENT.COLOPHON]: "reading_panel_colophon",
        },
        classes: {
            [ELEMENT.SCREEN_CONTROLLER]: "viewer-overlay-container", 
        },
        options:{
            poorSupport: true, // this site has some issues
            pageCounterPercentage: true // this site uses a percentage instead of page numbers, which complicates progress detection
        }
    },
    "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) => {
    if (siteSupport[window.location.hostname]) {
        // id search?
        if (typeof siteSupport[window.location.hostname].ids?.[elementEnum] === "string") {
            console.debug("Searching by id:", siteSupport[window.location.hostname].ids?.[elementEnum]);
            return document.getElementById(siteSupport[window.location.hostname].ids?.[elementEnum]);
        }  
        // id + path search?
        else if (typeof siteSupport[window.location.hostname].ids?.[elementEnum] === "object") {
            console.debug("Searching by id + path:", siteSupport[window.location.hostname].ids?.[elementEnum]);
            const idElement = document.getElementById(siteSupport[window.location.hostname].ids?.[elementEnum].id);
            if (idElement) {
                const pathParts = siteSupport[window.location.hostname].ids?.[elementEnum].path.split(".");
                let currentElement = idElement;
                for (const part of pathParts) {
                    if (part === "children") {
                        currentElement = currentElement.children;
                    } else {
                        currentElement = currentElement[part];
                    }
                    if (!currentElement) {
                        break;
                    }
                }
                return currentElement;
            }
        } 
        // class search?
        else if (typeof siteSupport[window.location.hostname].classes?.[elementEnum] === "string") {
            console.debug("Searching by class:", siteSupport[window.location.hostname].classes?.[elementEnum]);
            return document.getElementsByClassName(siteSupport[window.location.hostname].classes?.[elementEnum])[0];
        // class search with multiple elements of the same class?
        } else if (typeof siteSupport[window.location.hostname].classes?.[elementEnum] === "object") {
            console.debug("Searching by class + index:", siteSupport[window.location.hostname].classes?.[elementEnum]);
            const elements = document.getElementsByClassName(siteSupport[window.location.hostname].classes?.[elementEnum].class);
            return elements[siteSupport[window.location.hostname].classes?.[elementEnum].index];
        }
    }
}

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

function options() {
    return siteSupport[document.location.hostname].options ?? {};
}


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 = (-0.99*direction + 1) * viewportX() / 2;
    screen.dispatchEvent(new PointerEvent("pointerdown", {buttons: 1, clientX: x, clientY: 100, bubbles: true}));
    screen.dispatchEvent(new PointerEvent("mousedown", {buttons: 1, clientX: x, clientY: 100, bubbles: true}));
    screen.dispatchEvent(new PointerEvent("pointerup", {buttons: 0, clientX: x, clientY: 100, bubbles: true}));
    screen.dispatchEvent(new PointerEvent("mouseup", {buttons: 0, clientX: x, clientY: 100, bubbles: true}));
    screen.dispatchEvent(new PointerEvent("click", {buttons: 0, clientX: x, clientY: 100, bubbles: true}));
}

async function waitForPageLoad() {
    if (isLoadingPage() === undefined) {
        await sleep(1000); // with no loader element, we have no idea if the page is loaded, so just wait a second
        return;
    }
    while (isLoadingPage()) {
        await sleep(100);
    }
}

function currentPage() {
    return +getCSRElement(ELEMENT.CURRENT_PAGE_COUNTER).textContent;
}

function totalPageCount() {
    if (options().pageCounterPercentage) return Infinity; // we can't know the total page count if it's a percentage
    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");
}

function isColophonOpen() {
    return getCSRElement(ELEMENT.COLOPHON).style.display !== "none" && getCSRElement(ELEMENT.COLOPHON).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("mousedown", {
        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
    }));
        slider.dispatchEvent(new PointerEvent("mouseup", {
        buttons: 0,
        clientX: viewportX() - 25,
        clientY: viewportY() - 70,
        bubbles: true
    }));
        slider.dispatchEvent(new PointerEvent("click", {
        buttons: 0,
        clientX: viewportX() - 25,
        clientY: viewportY() - 70,
        bubbles: true
    }));
    await sleep(100);
    await waitForPageLoad();
}


// Main
//------------------------------

const MAX_ZERO_PADS = 4; // 9999 pages should be enough for anyone

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 === Infinity? 'an unknown number of' : totalPages} pages in total to save`);
    if (totalPages === Infinity) {
        addWarningMessage("the reader is in percentage mode, so the script cannot reliably track progress nor the total page count. Please check the zip after download");
    }
    const zeroPads = totalPages === Infinity? MAX_ZERO_PADS : 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: ====");

    if (isLoadingPage() === undefined) {
        addWarningMessage("the reader does not have a reliable loading spinner, so this script cannot detect when a page is fully loaded. It will wait 1 second per page instead, which may be too long or too short");
    }

    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" || canvas.style.position === "absolute") {
                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 (totalPages === Infinity && isColophonOpen()) {
            break;
        }
        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 addWarningMessage(message) {
    if (document.getElementById(warningMessagesId) === null) {
        const container = document.createElement("div");
        document.body.appendChild(container).id = warningMessagesId;
        container.style["position"] = "fixed";
        container.style["z-index"] = 900;
        container.style["bottom"] = "250px";
        container.style["right"] = "32px";
        container.style["display"] = "flex";
        container.style["flex-direction"] = "column";
        container.style["align-items"] = "flex-end";
    }
    const container = document.getElementById(warningMessagesId);

    const div = document.createElement("div");
    div.style["position"] = "relative";
    div.style["background"] = "black";
    div.style["border"] = "3px solid purple";
    div.style["padding"] = "8px";
    div.style["color"] = "yellow";
    container.appendChild(div);
    const text = document.createTextNode(message);
    div.appendChild(text);
}

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["z-index"] = 901;
    div.style["bottom"] = "150px";
    div.style["right"] = "32px";
    div.style["background"] = "black";
    div.style["border"] = "3px solid purple";
    div.style["padding"] = "8px";
    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 || totalPageCount() === Infinity) { // if totalPageCount is infinity (likely percentage mode), then current page != 0 can't be used to check if the reader is loaded
        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()) {
        if (options().poorSupport){
            addWarningMessage("the current reader site is poorly supported, some features may not work reliably. Feel free to open an issue on GitHub if you encounter any problems outside of the known warnings");
        }
        (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");
    }
}