Greasy Fork is available in English.
一个可扩展的通用型小说下载器。
Fra og med
一个可扩展的通用型小说下载器。
在这个 404 ##,由于种种原因,起点、晋江、刺猬猫、SF 等小说网站上的小说经常毫无征兆的消失。即使该小说已经入 V,即使你已经订阅了该小说。
这样的例子数不胜数。随便打开笔趣阁等转载网站,首发于起点,但现在起点上找不到该小说的情形比比皆是。像轻文轻小说这种整个网站都上天了的情况也不是没有。
如果小说消失时被笔趣阁等转载网站转载了,后来的读者尚且能一睹其风采,但如果小说发布的网站不是起点,小说也不够热,根本就没有转载网站转载,那后来者想一睹该小说的风采,就相当困难了。
404 小说文库项的目的是:保存这些质量上乘,但不够热门,没有被其他网站转载,彻底从互联网上消失的作品。
本脚本为 404 小说文库项目的组成部分之一,对于无登录墙的小说网站,如您同意,本脚本将会尝试将当前书籍详情页及目录页(如果存在)存档至互联网档案馆(archive.org),以备日后(被删除后)查看。
存档过程中将会搜集并上报您如下信息:IP 地址、User-Agent、Referer、当前书籍详情页 URL、当前书籍目录页 URL(如果存在)、当前小说下载器脚本版本、当前脚本管理器版本。除上述信息外,不会搜集您任何其他信息。
本软件为油猴脚本,需先在浏览器安装脚本管理器(Greasemonkey、Violentmonkey、Tampermonkey),再安装本脚本。具体可参见:如何安装用户脚本。
本脚本地址:
本脚本执行下载任务时将播放无声音频,以保证脚本后台运行时不被休眠。
如果本脚本支持该小说网站,当打开小说目录页时,网页右上角会出现下载图标,点击该图标即可开始下载。
如果你要下载的小说章节较多,等待时间可能较长,此时请耐心等待。
你通过右下角进度条了解当前下载进度,或者按下 F12,打开网页控制台查看当前下载状态。
下载完成后,本脚本将会自动下载一个 TXT 文档及 EPUB 文件。
TXT 文档请使用记事本或其它阅读软件进行阅读。
EPUB 文件请使用相应阅读器阅读。
A:在反馈之前,请保证您当前运行的脚本版本为最新版,如不是最新版,请更新脚本。
如最新版脚本仍出现错误,请说明具体网址,有无特殊操作以及其他附加说明,并附上调试日志,协助开发者明确出错原因。调试日志为下载生成的 zip 文件中的 debug.log
文件。
如需反馈问题,请至本项目支持页面提交 issue, 对于 greasyfork 评论区的反馈跟进不及时敬请谅解。
A:请提交 issue 并附上以上信息,网站 URL,原创网站或转载网站,有无收费章节,有无如登录墙等额外限制,希望添加的理由等。开发者将视情况,酌情添加。
A:有的。Matrix 空间:#404-novel-project:bgme.me,Telegram 群组:https://t.me/+ZCngCQiJ_xo2NDI1 。
特别提醒:如欲下载支持列表中网站的##章节,请登录相应网站帐户,并确定已购买相应##章节。未登录网站帐户,或未购买的##章节,下载时将直接忽略,无法进行下载。
站点 | 公共章节 | ##章节 | 备注 |
---|---|---|---|
SF 轻小说 | ✅* | ✅** | *不支持对话小说,例:224282。 **VIP 章节仅支持图片版。 |
起点中文网 | ✅ | ✅ | 部分小说 VIP 章节可能出现乱码无法下载。 |
起点女生网 | ✅ | ✅ | |
晋江文学城 | ✅ | ✅ | 晋江文学城 VIP 章节可添加 API Token 以获得更好体验,Token 添加方法参见 Token 填写一节。 |
长佩文学 | ✅ | ✅ | 反爬较严,限制下载速度,每分钟约可下载 6 章,请耐心等待,最好不要多开页面同时下载多本长佩小说。 长佩文学为单页应用,如打开书籍详情页右上角未出现下载图标,请按下 F5 重新加载当前页面。 |
书耽 | ✅ | ✅ | VIP 章节仅支持图片版。 |
海棠文化线上文学城 | ✅ | ✅ | |
次元姬 | ✅ | ✅ | |
米国度 | ✅ | ✅ | |
寒#纪年原创网 | ✅ | ✅ | |
哔哩哔哩漫画 | ✅ | ✅ | |
息壤中文网 | ✅ | ✅ | |
有#小说网 | ✅ | ✅ | |
独阅读 | ✅ | ✅ | |
轻之文库轻小说 | ✅ | ❌ | VIP 章节仅支持 APP 查看 |
纵横中文网 | ✅ | ❌ | |
花语女生网 | ✅ | ❌ | |
17K 小说网 | ✅ | ❌ | |
书海小说网 | ✅ | ❌ | |
塔读文学 | ✅ | ❌ | |
七猫中文网 | ✅ | ❌ | 请先进入作品目录再运行脚本。 |
废文网 | ✅ | ❎ | 部分小说或章节需登录后查看。 |
pixiv | ✅ | ❎ | 单页应用,如打开书籍详情页右上角未出现下载图标,请按下 F5 重新加载当前页面。 |
动漫之家 | ✅ | ❎ | 需下载大量图片,速度较慢,请耐心等待。 需占用大量内存,请保证最终生成文件 4 倍以上内存,即最终下载生成 500MB ZIP 文件,运行时请保证至少 2GB 内存空间。可使用筛选函数,分次下载。 |
Lofter | ✅ | ❎ | 因本脚本会将博文中的图片也一同下载下来,对于图片特别多的博客,下载时请注意内存用量(800MB 限制),根据实际情况使用筛选函数分次下载。 部分博文内含视频内容,为节省内存使用,加快下载速度,本脚本将跳过视频内容。 如您使用广告屏蔽器,可能会影响本脚本在 Lofter 的工作。 |
努努书坊 | ✅ | ❎ | 格式众多,如发现不支持页面敬请反馈。 |
真白萌 | ✅ | ❎ | |
##书库 | ✅ | ❎ | |
#青果 | ✅ | ❎ | |
カクヨム | ✅ | ❎ | |
小説家になろう | ✅ | ❎ | |
ハーメルン | ✅ | ❎ | |
暁 | ✅ | ❎ | |
ファンタジー小説 | ✅ | ❌ | |
Novel Up Plus | ✅ | ❎ |
unsafeWindow
:用于获取自定义筛选函数、自定义保存参数等设置。GM_info
/GM.info
: 获取并输出脚本运行环境。GM_xmlhttpRequest
/GM.xmlHttpRequest
:用于跨域 HTTP 请求。GM_setValue
/GM.setValue
、GM_getValue
/GM.getValue
、GM_deleteValue
/GM.deleteValue
: 用于统计模块,本地统计运行次数。当前部分网站(如晋江文学城)需要手动填写登录 token。
下载抓包精灵(可在 Google Play、#安搜索到,其他软件也可以)并配置好设置,然后登录晋江文学城 android app (其他##均不适用)并随意浏览章节,在形如“https://app.jjwxc.org/androidapi/chapterContent?” 等链接中找到&token=后的字符串(止于下一个&)。
在脚本管理器中新建如下脚本(不要把该脚本代码和其他脚本代码合并,除非你完全理解脚本的意思)并保存:
// ==UserScript==// @name auto inject tokenOptions// @namespace http://tampermonkey.net/// @version 0.1// @description auto inject tokenOptions// @author You// @match *://*/*// @grant none// ==/UserScript==(function () {"use strict";const tokenOptions = {Jjwxc: "填入token,形如客户号+下划线'_'+字母与数字混合的字符串",};window.tokenOptions = tokenOptions;})();
例如:
Jjwxc:"11111111_750afc84c839aaaaafccd841fffd11f1",
[!CAUTION]在设置中启用调试模式以后,日志可能会输出一些包含token的链接,这个设计的初衷是为了更快定位发生的问题。
请不要直接将该日志上传到互联网上,当且仅当从事维护的开发者需要此日志定位问题时再提供(可以通过重新登录之前抓取token设备上的晋江app以使原token失效)。
设置菜单中按击开启。
如欲只下载部分章节,请在点击运行按钮前,按下 F12 打开开发者工具,在 window
下创建 chapterFilter
函数,具体格式如下:
declare enum Status {pending = 0,downloading = 1,failed = 2,finished = 3,aborted = 4,saved = 5,}interface ChapterAdditionalMetadate {lastModified?: number;}declare class Chapter {bookUrl: string;bookname: string;chapterUrl: string;chapterNumber: number;chapterName: string | null;isVIP: boolean;isPaid: boolean | null;sectionName: string | null;sectionNumber: number | null;sectionChapterNumber: number | null;chapterParse: BaseRuleClass["chapterParse"];charset: string;options: object;status: Status;retryTime: number;contentRaw: HTMLElement | null;contentText: string | null;contentHTML: HTMLElement | null;contentImages: attachmentClass[] | null;additionalMetadate: ChapterAdditionalMetadate | null;chapterHtmlFileName: string | number;constructor(bookUrl: string,bookname: string,chapterUrl: string,chapterNumber: number,chapterName: string | null,isVIP: boolean,isPaid: boolean | null,sectionName: string | null,sectionNumber: number | null,sectionChapterNumber: number | null,chapterParse: BaseRuleClass["chapterParse"],charset: string,options: object);init(): Promise<this>;private parse;}declare class attachmentClass {url: string;name: string;mode: "naive" | "TM";headers?: {[index: string]: string;};private defaultHeader;status: Status;retryTime: number;imageBlob: Blob | null | void;constructor(imageUrl: string, name: string, mode: "naive" | "TM");init(): Promise<Blob | null>;private downloadImage;private tmDownloadImage;}interface chapterFilter {(chapter: Chapter): boolean;}
自定义筛选函数示例:
只下载该本小说前 100 章内容:
function chapterFilter(chapter) {return chapter.chapterNumber <= 100;}
只下载第一卷内容:
function chapterFilter(chapter) {return chapter.sectionNumber === 1;}
只下载章节名称中含有“#器”的章节:
function chapterFilter(chapter) {return chapter.chapterName.includes("#器");}
自定义保存参数允许您修改保存文件的样式,章节标题等内容。
使用方法大致同自定义筛选函数,即在 window
下创建 saveOptions
对象,具体格式如下:
declare class saveBook {protected book: Book;mainStyleText: string;tocStyleText: string;constructor(book: Book);saveTxt(): void;saveLog(): void;saveZip(runSaveChapters?: boolean): Promise<void>;addChapter(chapter: Chapter): void;getchapterName(chapter: Chapter): string;genSectionText(sectionName: string): string;genChapterText(chapterName: string, contentText: string): string;genSectionHtmlFile(chapterObj: Chapter): Blob;genChapterHtmlFile(chapterObj: Chapter): Blob;chapterSort(a: Chapter, b: Chapter): 0 | 1 | -1;}interface saveOptions {mainStyleText?: saveBook["mainStyleText"];tocStyleText?: saveBook["tocStyleText"];getchapterName?: saveBook["getchapterName"];genSectionText?: saveBook["genSectionText"];genChapterText?: saveBook["genChapterText"];genSectionHtmlFile?: saveBook["genSectionHtmlFile"];genChapterHtmlFile?: saveBook["genChapterHtmlFile"];chapterSort?: saveBook["chapterSort"];}
自定义保存参数示例:
将章节名称格式修改为 第xx章 xxxx
:
const saveOptions = {getchapterName: (chapter) => {if (chapter.chapterName) {return `第${chapter.chapterNumber.toString()}章 ${chapter.chapterName}`;} else {return `第${chapter.chapterNumber.toString()}章`;}},};window.saveOptions = saveOptions;
更改 ZIP 文档中章节 HTML 文件样式:
const saveOptions = {mainStyleText: `p {text-indent: 4em;display: block;line-height: 1.3em;margin-top: 0.4em;margin-bottom: 0.4em;}`,};window.saveOptions = saveOptions;
txt 文档每个自然段前加两个空格
const saveOptions = {genChapterText: (chapterName, contentText) => {contentText = contentText.split("\n").map((line) => {if (line.trim() === "") {return line;} else {return line.replace(/^/, " ");}}).join("\n");return `## ${chapterName}\n\n${contentText}\n\n`;},};window.saveOptions = saveOptions;
保存章节时倒序排列
const saveOptions = {chapterSort: (a, b) => {if (a.chapterNumber > b.chapterNumber) {return -1;}if (a.chapterNumber === b.chapterNumber) {return 0;}if (a.chapterNumber < b.chapterNumber) {return 1;}return 0;},};window.saveOptions = saveOptions;
使用用户脚本自动注入自定义保存参数:
如您总是想使用某一自定义保存参数,你可以使用如下用户脚本(根据实际需要修改相应数值),自动向页面注入自定义保存参数。
// ==UserScript==// @name auto inject saveOptions// @namespace http://tampermonkey.net/// @version 0.1// @description auto inject saveOptions// @author You// @match *://*/*// @grant none// ==/UserScript==(function () {"use strict";const saveOptions = {getchapterName: (chapter) => {if (chapter.chapterName) {return `第${chapter.chapterNumber.toString()}章 ${chapter.chapterName}`;} else {return `第${chapter.chapterNumber.toString()}章`;}},};window.saveOptions = saveOptions;})();
自定义筛选函数同理也可使用用户脚本自动注入。
interface customFinishCallback {(): void;}
自定义完成回调函数将在下载完成并生成 ZIP 文件后自动执行。
使用自定义完成回调函数可在下载完成后自动完成某些工作,例如:关闭当前窗口。
function customFinishCallback(book: Book) {window.close();}window.customFinishCallback = customFinishCallback;
git clone https://github.com/yingziwu/novel-downloader.git
将项目克隆至本地(访问 github 可能需要使用代理)。yarn install
安装依赖。BaseRuleClass
类,实现 bookParse
、chapterParse
抽象方法,在 router/download.ts
文件中添加相应选择规则,在 header.json
文件 match
字段添加相应的匹配规则。yarn run build
编译生成最终脚本文件 dist/bundle.user.js
。AGPL-3.0
感谢 JetBrains 向本项目提供 WebStorm IDE。