您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
Download pixiv novels in Epub format
当前为
/* eslint-disable no-multi-spaces */ /* eslint-disable no-return-assign */ // ==UserScript== // @name Pixiv novel to Epub // @name:zh-CN Pixiv小说Epub合成器 // @name:en Pixiv novel to Epub // @namespace PY-DNG userscripts // @version 0.1.2 // @description Download pixiv novels in Epub format // @description:zh-CN 以Epub格式下载Pixiv小说 // @description:en Download pixiv novels in Epub format // @author PY-DNG // @license GPL-3.0-or-later // @match *://www.pixiv.net/* // @match *://pixiv.net/* // @connect pximg.net // @require https://update.gf.qytechs.cn/scripts/456034/1303041/Basic%20Functions%20%28For%20userscripts%29.js // @require data:application/javascript,window.setImmediate%20%3D%20window.setImmediate%20%7C%7C%20((f%2C%20...args)%20%3D%3E%20window.setTimeout(()%20%3D%3E%20f(args)%2C%200))%3B // @require https://fastly.jsdelivr.net/npm/[email protected]/dist/jszip.min.js // @require https://fastly.jsdelivr.net/npm/[email protected]/ejs.min.js // @require https://fastly.jsdelivr.net/npm/[email protected]/dist/jepub.min.js // @icon data:image/png;base64,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 // @grant GM_xmlhttpRequest // @grant GM_registerMenuCommand // @run-at document-start // ==/UserScript== // @require https://fastly.jsdelivr.net/npm/[email protected]/setImmediate.min.js // @require https://fastly.jsdelivr.net/npm/[email protected]/dist/jepub.min.js // @require https://fastly.jsdelivr.net/npm/[email protected]/ejs.min.js // @require https://fastly.jsdelivr.net/npm/[email protected]/dist/jszip.min.js /* global LogLevel DoLog Err $ $All $CrE $AEL $$CrE addStyle detectDom destroyEvent copyProp copyProps parseArgs escJsStr replaceText getUrlArgv dl_browser dl_GM AsyncManager */ /* global jEpub, JSZip, ejs */ let PixivAPI = (function() { queueTask.sleep = 200; queueTask.max = 10; return { get, safeGet, utils: { toAbsURL, toSearch, queueTask }, // https://www.pixiv.net/ajax/novel/18673574 novel: id => safeGet(`/ajax/novel/${id}`), // https://www.pixiv.net/ajax/novel/7522350/insert_illusts?id%5B%5D=60139778-1&lang=zh&version=1efff679631a40a674235820806f7431d67065d9 insert_illusts: (novel_id, illust_ids, lang='zh') => { const url = `/ajax/novel/${novel_id}/insert_illusts`; const query = { lang }; if (Array.isArray(illust_ids)) { for (let i = 0; i < illust_ids.length; i++) { const id = illust_ids[i]; query[`id[${i}]`] = id; } } else { query[`id[]`] = illust_ids; } return safeGet(url, query); }, // https://www.pixiv.net/ajax/novel/series/9649276?lang=zh&version=a48f2f681629909b885608393916b81989accf5b // 'version' removed due to unspecified meaning series: (id, lang='zh') => safeGet(`/ajax/novel/series/${id}`, { id, lang }), // https://www.pixiv.net/ajax/novel/series_content/9649276?limit=30&last_order=0&order_by=asc series_content: (id, limit=30, last_order=0, order_by='asc') => safeGet(`/ajax/novel/series_content/${id}`, { limit, last_order, order_by }), }; function safeGet() { return queueTask(() => get.call(this, ...arguments)); } function get(url, params, responseType='json', retry=2) { return new Promise((resolve, reject) => { GM_xmlhttpRequest({ method: 'GET', responseType, headers: { Referer: /^(www\.)?pixiv\.net$/.test(location.host) ? location.href : 'https://www.pixiv.net/' }, url: toAbsURL(url, params), onload: async res => res.status === 200 && (responseType !== 'json' || res.response?.error === false) ? resolve(res.response) : checkRetry(res), onerror: checkRetry }); async function checkRetry(err) { retry-- > 0 ? resolve(await get(url, params, responseType, retry)) : reject(err); } }); } function toAbsURL(pathname, searchOptions) { return new URL(pathname, `https://www.pixiv.net/`).href + (searchOptions ? `?${toSearch(searchOptions)}` : ''); } function toSearch(options) { return new URLSearchParams(options).toString() } function queueTask(task) { init(); return new Promise((resolve, reject) => { queueTask.tasks.push({task, resolve, reject}); checkTask(); }); function checkTask() { setTimeout(() => { if (queueTask.ongoing < queueTask.max && queueTask.tasks.length) { const task = queueTask.tasks.shift(); queueTask.ongoing++; setTimeout( () => task.task().then(v => { queueTask.ongoing--; task.resolve(v); checkTask(); }).catch(e => { queueTask.ongoing--; task.reject(e); checkTask(); }), queueTask.sleep ); } }); } function init() { if (!queueTask.initialized) { const defaults = { max: 3, sleep: 500, }; for (const [p, v] of Object.entries(defaults)) { !queueTask.hasOwnProperty(p) && (queueTask[p] = v); } queueTask.tasks = []; queueTask.ongoing = 0; queueTask.initialized = true; } } } }) (); (async function __MAIN__() { 'use strict'; const CONST = { TextAllLang: { DEFAULT: 'zh-CN', 'zh-CN': { DownloadEpub: '下载当前小说Epub' } }, GFURL: 'https://gf.qytechs.cn/scripts/483999', GFAuthorURL: 'https://gf.qytechs.cn/users/667968' }; // Init language const i18n = Object.keys(CONST.TextAllLang).includes(navigator.language) ? navigator.language : CONST.TextAllLang.DEFAULT; CONST.Text = CONST.TextAllLang[i18n]; // @require fallbacks await Promise.all([ { missing: typeof setImmediate === 'undefined', src: 'https://fastly.jsdelivr.net/npm/[email protected]/setImmediate.min.js' }, { missing: typeof JSZip === 'undefined', src: 'https://fastly.jsdelivr.net/npm/[email protected]/dist/jszip.min.js' }, { missing: typeof ejs === 'undefined', src: 'https://fastly.jsdelivr.net/npm/[email protected]/ejs.min.js' }, { missing: typeof jEpub === 'undefined', src: 'https://fastly.jsdelivr.net/npm/[email protected]/dist/jepub.min.js' } ].filter(script => script.missing).map(src => new Promise((resolve, reject) => document.head.appendChild($$CrE({ tagName: 'script', props: { src }, listeners: [ ['load', resolve], ['error', reject] ] }))))); // User Interface GM_registerMenuCommand(CONST.Text.DownloadEpub, downloadEpub); function downloadEpub() { const pathname = location.pathname; // Novel series // https://www.pixiv.net/novel/series/9649276 /^\/novel\/series\/\d+$/.test(pathname) && downloadSeries(); // Novel // https://www.pixiv.net/novel/show.php?id=18673574 /^\/novel\/show\.php$/.test(pathname) && downloadNovel(); } async function downloadSeries() { const id = location.pathname.split('/').pop(); const epub = new jEpub(); // Get series data const series = (await PixivAPI.series(id)).body; await initEpub(epub, series); // List all novels const promises = []; for (let index = 0; index < series.total; index += 30) { const promise = PixivAPI.series_content(id, 30, index); promises.push(promise); } const list = (await Promise.all(promises)).reduce((l, json) => ((l.push(...json.body.page.seriesContents), l)), []); DoLog(list); const novel_datas = await Promise.all(list.map(async novel => (await PixivAPI.novel(novel.id)).body)); DoLog(novel_datas); // Add chapters one by one // Do not use promise.all, because that will break the order for (const data of novel_datas) { await addChapter(epub, data); } //await Promise.all(novel_datas.map(async data => await addChapter(epub, data))); DoLog('Saving Epub'); saveEpub(epub, series.title + '.epub'); } async function downloadNovel() { const id = getUrlArgv('id'); const json = await PixivAPI.novel(id); const data = json.body; const epub = new jEpub(); await Promise.all([initEpub(epub, data), addChapter(epub, data)]); saveEpub(epub, data.title + '.epub'); } // Compatible with PixivAPI.novel / PixivAPI.series async function initEpub(epub, data) { epub.init({ i18n: 'en', title: data.title, author: data.userName, publisher: '', description: data.description || data.caption, tags: Array.isArray(data.tags) ? data.tags : data.tags.tags.map(tag => tag.tag) }); epub.date(new Date(data.uploadDate || data.lastPublishedContentTimestamp)); epub.notes(`EPUB generated from: <a href="${htmlEncode(location.href)}" title="${htmlEncode(data.extraData.meta.title)}">${htmlEncode(location.href)}</a></br>By <a href="${htmlEncode(CONST.GFURL)}">${htmlEncode(GM_info.script.name)}</a> author <a href="${htmlEncode(CONST.GFAuthorURL)}">${htmlEncode(GM_info.script.author)}</a></br></br>Copyright belongs to the article author. Please comply with relevant legal requirements while reading and distributing this file.`); const coverUrl = data.coverUrl || data.cover.urls.original; const cover = await PixivAPI.safeGet(coverUrl, null, 'blob'); epub.cover(cover); return epub; } async function addChapter(epub, data) { let content = data.content; // Add description and cover image to chapter beginning const cover = await PixivAPI.safeGet(data.coverUrl, null, 'blob'); const coverId = `ChapterCover-${data.id}`; epub.image(cover, coverId); content = data.description + `\n<%= image[${escJsStr(coverId)}] %>\n` + content; // Load images const imagePromises = []; content = content.replace(/\[uploadedimage:([\d\-]+)\]/g, (match_str, id) => { const url = data.textEmbeddedImages[id].urls.original; const promise = PixivAPI.safeGet(url, null, 'blob').then(blob => epub.image(blob, id));//.catch(err => ); imagePromises.push(promise); return `\n<%= image[${id}] %>\n`; }); const illusts = Array.from(new Set( [...content.matchAll(/\[pixivimage:([\d\-]+)\]/g)] )); if (illusts.length) { const illustsJson = await PixivAPI.insert_illusts(data.id, illusts.map(match => match[1])); illusts.forEach(illust => { const id = illust[1]; if (illustsJson.body[id].visible) { const url = illustsJson.body[id].illust.images.original; const promise = PixivAPI.safeGet(url, null, 'blob').then(blob => epub.image(blob, id));//.catch(err => ); imagePromises.push(promise); content = content.replaceAll(illust[0], `\n<%= image[${escJsStr(id)}] %>\n`); } }); } await Promise.all(imagePromises); // Parse '[newpage]' content = content.replaceAll('[newpage]', '\n'); // Parse '[[rb:久世彩葉 > くぜ いろは]]' content = content.replace(/\[\[rb:([^\[\]]+) *> *([^\[\]]+)\]\]/g, (match_str, main, desc) => { return `<ruby>${htmlEncode(main)}<rp>(</rp><rt>${htmlEncode(desc)}</rt><rp>)</rp></ruby>`; }); // Check undealed markers const markers = content.matchAll(/\[+[^\[\]]+\]+/g); markers.length && DoLog(LogLevel.Warning, ['Undealed markers found', markers]); // Up to 2 connected newlines at once content = content.replaceAll(/\n{2,}/g, '\n\n'); // Covert into html content = content.split('\n').map(line => line.trim() ? `<p>${line}</p>` : line).join('\n'); epub.add(data.title, content); } async function saveEpub(epub, filename) { const blob = await epub.generate('blob'); const url = URL.createObjectURL(blob); dl_browser(url, filename); setTimeout(() => URL.revokeObjectURL(url)); } function htmlEncode(text, encodes = '<>\'";&#') { return Array.from(text).map(char => !encodes || encodes.includes(char) ? `&#${char.charCodeAt(0)};` : char).join(''); } })();
QingJ © 2025
镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址