tmo-dl

Script to download manga from TuMangaOnline

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Greasemonkey 油猴子Violentmonkey 暴力猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Violentmonkey 暴力猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Userscripts ,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展后才能安装此脚本。

(我已经安装了用户脚本管理器,让我安装!)

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

(我已经安装了用户样式管理器,让我安装!)

// ==UserScript==
// @name         tmo-dl
// @namespace    https://gist.github.com/hiroshil
// @version      1.0
// @description  Script to download manga from TuMangaOnline
// @license      MIT
// @author       hiroshil
// @match        https://visortmo.com/library/manga/*/*
// @icon         https://www.google.com/s2/favicons?sz=64&domain=visortmo.com
// ==/UserScript==
/* jshint esversion: 8 */

//https://stackoverflow.com/questions/951021/what-is-the-javascript-version-of-sleep

function waitForElm(selector) {
    return new Promise(resolve => {
        if ($(selector).length) {
            return resolve($(selector));
        }
        const observer = new MutationObserver(mutations => {
            if ($(selector).length) {
                resolve($(selector));
                observer.disconnect();
            }
        });
        observer.observe(document.body, {
            childList: true,
            subtree: true
        });
    });
}
function waitForXHRRequest(url,method,responseType = undefined, getUrl = false) {
	return new Promise( (resolve, reject) => {
		var res = new XMLHttpRequest();
		res.responseType = responseType;
		res.open(method, url, true);
		res.send();
		res.onload = () => {
			if (res.status >= 200 && res.status < 400) { // If response is all good...
                if (getUrl) {
                    resolve(res.responseURL);
                }
				resolve(res.response);
			} else {
				reject(false);
			}
		}
	});

}
var sleepSetTimeout_ctrl;
function sleep(ms) {
    clearInterval(sleepSetTimeout_ctrl);
    return new Promise(resolve => sleepSetTimeout_ctrl = setTimeout(resolve, ms));
}
async function downloadChapter(url,filename, callback){
    const finalURL = await waitForXHRRequest(url, "GET", undefined, true);
    const regex = /viewer\/(.*?)(?:\/|$)/;
    const match = finalURL.match(regex);

    if (match) {
        const chapId = match[1];
        const res = await waitForXHRRequest(finalURL.split(chapId)[0] + chapId + "/cascade", "GET");
        if (res){
            const imgs = $(res).find("#main-container .viewer-img");
            const zipFileWriter = new zip.BlobWriter();
            const zipWriter = new zip.ZipWriter(zipFileWriter);
            for (let i = 0; i < imgs.length; i++) {
                const src = $(imgs[i]).attr("data-src");
                var blob;
                do {
                    blob = await waitForXHRRequest(src,'GET','blob');
                    await sleep(200);
                }
                while (!blob);
                if (blob.type.includes("image")){
                    const blobReader = new zip.BlobReader(blob);
                    await zipWriter.add(String(i) + "_" + src.split("/").pop(), blobReader);
                }
            }
            await zipWriter.close();
            const zipFileBlob = await zipFileWriter.getData();
            const anchor = document.createElement("a");
            const clickEvent = new MouseEvent("click");
            anchor.href = window.URL.createObjectURL(zipFileBlob);
            anchor.download = filename;
            anchor.dispatchEvent(clickEvent);
        }
        else {
            alert("Error!! Please try again!");
        }
    } else {
        alert("Error!! Please try again!");
    }
    callback();
}
function setupButton(jNode) {
      jNode.each(function() {
          const chapName = $(this).prev().find("div > div.col-10.text-truncate > a").text().trim();
          var viewBtn = $( "#" + $(this).attr("id") + " > div > ul > li > div > div.col-2.col-sm-1.text-right").first();
          var chapURL = viewBtn.find("a").attr('href');
          viewBtn.prev().remove();
          var dwnlBtn = viewBtn.clone();
          dwnlBtn.find("a").removeAttr("href").removeClass("btn-sm").addClass("btn-lg").find("span").removeAttr('class').addClass("fas fa-arrow-alt-circle-down dl-btn");
          dwnlBtn.click(function() {
              $(this).attr('disabled','disabled');
              $(this).find(".dl-btn").removeClass("fas").addClass("far");
              const output = $(".element-title.my-2").contents().first().text().trim() + " - " + chapName + ".zip";
              console.log(output);
              downloadChapter(chapURL, output, () => { $(".dl-btn").closest("[disabled]").removeAttr('disabled').find("span").removeClass("far").addClass("fas"); });
          });
          viewBtn.after(dwnlBtn);
      });
}
async function main(){
    var xhr = await waitForXHRRequest("https://raw.githubusercontent.com/gildas-lormeau/zip.js/v2.7.17/dist/zip.js","GET");
    eval(xhr);
    const jNode = await waitForElm('.upload-link > div');
    setupButton(jNode);
}
main();