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