Goda漫画下载

打开章节页面,一键下载全部章节

このスクリプトの質問や評価の投稿はこちら通報はこちらへお寄せください。
/* eslint-disable no-multi-spaces */

// ==UserScript==
// @name               Goda漫画下载
// @namespace          http://tampermonkey.net/
// @version            0.1
// @description        打开章节页面,一键下载全部章节
// @author             PY-DNG
// @license            MIT
// @match              http*://cn.godamanga.com/chapterlist/*
// @require            https://greasyfork.org/scripts/456034/code/script.js
// @connect            godamanga.com
// @connect            godamanga.online
// @icon               
// @grant              GM_registerMenuCommand
// @grant              GM_xmlhttpRequest
// @grant              GM_download
// ==/UserScript==

(function __MAIN__() {
    'use strict';

	// Constances
	const CONST = {
		TextAllLang: {
			DEFAULT: 'zh-CN',
			'zh-CN': {
				DownloadAllChapters: '下载全部章节',
				FolderName: 'Goda漫画下载',
				DownloadFullName: '{FolderName}/{MangaName}/{ChapterName}/{ImageName}',
				ChapterFolderName: '{Number} - {Name}',
				ImageFileName: '{Number}.{Ext}'
			}
		}
	};

	// Init language
	const i18n = !Object.keys(CONST.TextAllLang).includes(navigator.language) ? navigator.language : CONST.TextAllLang.DEFAULT;
	CONST.Text = CONST.TextAllLang[i18n];

	main();
	function main() {
		GMXHRHook();
		GM_registerMenuCommand(CONST.Text.DownloadAllChapters, downloadAll);
	}

	function downloadAll() {
		const mangaName = $('header nav.ct-breadcrumbs a[href*="{URL}"] span'.replace('{URL}', $('.chapter_list_title').pathname)).innerText;
		const chapter_links = [...$All('ul.main.version-chaps a')].map(a => a.href).reverse();
		const len = chapter_links.length.toString().length;
		chapter_links.forEach((url, i) => downloadChapter(url, mangaName, fillNum(i+1, len)));
	}

	function downloadChapter(url, mangaName, chapterNo) {
		getDocument(url, function(oDom) {
			const chapterName = $(oDom, 'header nav.ct-breadcrumbs .last-item').innerText;
			const urls = [...$All(arguments[0], 'img[src*="/scomic/"]')].map(img => img.src);
			const len = urls.length.toString().length;
			urls.forEach((img_url, i) => downloadImage(img_url, mangaName, chapterName, chapterNo, fillNum(i+1, len)));
		});
	}

	function downloadImage(url, mangaName, title, chapterNo, imageNo, retry=3, err=null) {
		if (retry <= 0) {
			DoLog(LogLevel.Error, ['downloadImage: GM_xmlhttpRequest error(max retry reached)', err], true);
			return;
		}
		GM_xmlhttpRequest({
			url: url,
			responseType: 'blob',
			timeout: 20000,
			ontimeout: err => downloadImage(url, title, chapterNo, imageNo, retry-1, err),
			onerror: err => downloadImage(url, title, chapterNo, imageNo, retry-1, err),
			onload: (response) => {
				const blob = response.response;
				const dataUrl = URL.createObjectURL(blob);
				const ext = getExtname(blob.type.split(';')[0]) || 'unkown_filetype.jpg';
				const Text = CONST.Text;
				GM_download({
					name: replaceText(Text.DownloadFullName, {
						'{FolderName}': Text.FolderName,
						'{MangaName}': mangaName,
						'{ChapterName}': replaceText(Text.ChapterFolderName, {'{Number}' : chapterNo, '{Name}': title}),
						'{ImageName}': replaceText(Text.ImageFileName, {'{Number}': imageNo, '{Ext}': ext}),
					}),
					url: dataUrl
				})
			}
		});

		function getExtname(...args) {
			const map = {
				'image/png': 'png',
				'image/jpg': 'jpg',
				'image/gif': 'gif',
				'image/bmp': 'bmp',
				'image/jpeg': 'jpeg',
				'image/webp': 'webp',
				'image/tiff': 'tiff',
				'image/vnd.microsoft.icon': 'ico',
			};
			return map[args.find(a => map[a])];
		}
	}

	function fillNum(num, len) {
		const str = num.toString();
		return '0'.repeat(len-str.length) + str;
	}

	// Download and parse a url page into a html document(dom).
    // when xhr onload: callback.apply([dom, args])
    function getDocument(url, callback, args=[]) {
        GM_xmlhttpRequest({
            method       : 'GET',
            url          : url,
            responseType : 'blob',
			timeout      : 15 * 1000,
            onload       : function(response) {
                const htmlblob = response.response;
				parseDocument(htmlblob, callback, args);
            },
			onerror      : reqerror,
			ontimeout    : reqerror
        });

		function reqerror(e) {
			DoLog(LogLevel.Error, 'getDocument: Request Error');
			DoLog(LogLevel.Error, e);
			throw new Error('getDocument: Request Error')
		}
    }

	function parseDocument(htmlblob, callback, args=[]) {
		const reader = new FileReader();
		reader.onload = function(e) {
			const htmlText = reader.result;
			const dom = new DOMParser().parseFromString(htmlText, 'text/html');
			args = [dom].concat(args);
			callback.apply(null, args);
			//callback(dom, htmlText);
		}
		const charset = document.characterSet;
		reader.readAsText(htmlblob, charset);
	}

	// GM_XHR HOOK: The number of running GM_XHRs in a time must under maxXHR
	// Returns the abort function to stop the request anyway(no matter it's still waiting, or requesting)
	// (If the request is invalid, such as url === '', will return false and will NOT make this request)
	// If the abort function called on a request that is not running(still waiting or finished), there will be NO onabort event
	// Requires: function delItem(){...} & function uniqueIDMaker(){...}
	function GMXHRHook(maxXHR=5) {
		const GM_XHR = GM_xmlhttpRequest;
		const getID = uniqueIDMaker();
		let todoList = [], ongoingList = [];
		GM_xmlhttpRequest = safeGMxhr;

		function safeGMxhr() {
			// Get an id for this request, arrange a request object for it.
			const id = getID();
			const request = {id: id, args: arguments, aborter: null};

			// Deal onload function first
			dealEndingEvents(request);

			/* DO NOT DO THIS! KEEP ITS ORIGINAL PROPERTIES!
			// Stop invalid requests
			if (!validCheck(request)) {
				return false;
			}
			*/

			// Judge if we could start the request now or later?
			todoList.push(request);
			checkXHR();
			return makeAbortFunc(id);

			// Decrease activeXHRCount while GM_XHR onload;
			function dealEndingEvents(request) {
				const e = request.args[0];

				// onload event
				const oriOnload = e.onload;
				e.onload = function() {
					reqFinish(request.id);
					checkXHR();
					oriOnload ? oriOnload.apply(null, arguments) : function() {};
				}

				// onerror event
				const oriOnerror = e.onerror;
				e.onerror = function() {
					reqFinish(request.id);
					checkXHR();
					oriOnerror ? oriOnerror.apply(null, arguments) : function() {};
				}

				// ontimeout event
				const oriOntimeout = e.ontimeout;
				e.ontimeout = function() {
					reqFinish(request.id);
					checkXHR();
					oriOntimeout ? oriOntimeout.apply(null, arguments) : function() {};
				}

				// onabort event
				const oriOnabort = e.onabort;
				e.onabort = function() {
					reqFinish(request.id);
					checkXHR();
					oriOnabort ? oriOnabort.apply(null, arguments) : function() {};
				}
			}

			// Check if the request is invalid
			function validCheck(request) {
				const e = request.args[0];

				if (!e.url) {
					return false;
				}

				return true;
			}

			// Call a XHR from todoList and push the request object to ongoingList if called
			function checkXHR() {
				if (ongoingList.length >= maxXHR) {return false;};
				if (todoList.length === 0) {return false;};
				const req = todoList.shift();
				const reqArgs = req.args;
				const aborter = GM_XHR.apply(null, reqArgs);
				req.aborter = aborter;
				ongoingList.push(req);
				return req;
			}

			// Make a function that aborts a certain request
			function makeAbortFunc(id) {
				return function() {
					let i;

					// Check if the request haven't been called
					for (i = 0; i < todoList.length; i++) {
						const req = todoList[i];
						if (req.id === id) {
							// found this request: haven't been called
							delItem(todoList, i);
							return true;
						}
					}

					// Check if the request is running now
					for (i = 0; i < ongoingList.length; i++) {
						const req = todoList[i];
						if (req.id === id) {
							// found this request: running now
							req.aborter();
							reqFinish(id);
							checkXHR();
						}
					}

					// Oh no, this request is already finished...
					return false;
				}
			}

			// Remove a certain request from ongoingList
			function reqFinish(id) {
				let i;
				for (i = 0; i < ongoingList.length; i++) {
					const req = ongoingList[i];
					if (req.id === id) {
						ongoingList = delItem(ongoingList, i);
						return true;
					}
				}
				return false;
			}
		}
	}

	// Makes a function that returns a unique ID number each time
	function uniqueIDMaker() {
		let id = 0;
		return makeID;
		function makeID() {
			id++;
			return id;
		}
	}

	// Del a item from an array using its index. Returns the array but can NOT modify the original array directly!!
	function delItem(arr, delIndex) {
		arr = arr.slice(0, delIndex).concat(arr.slice(delIndex+1));
		return arr;
	}
})();