DownloadAllContent

Lightweight web scraping script. Fetch and download main textual content from the current page, provide special support for novels

Install this script?
Author's suggested script

You may also like Pagetual.

Install this script
  1. // ==UserScript==
  2. // @name DownloadAllContent
  3. // @name:zh-CN 怠惰小说下载器
  4. // @name:zh-TW 怠惰小説下載器
  5. // @name:ja 怠惰者小説ダウンロードツール
  6. // @namespace hoothin
  7. // @version 2.8.3.15
  8. // @description Lightweight web scraping script. Fetch and download main textual content from the current page, provide special support for novels
  9. // @description:zh-CN 通用网站内容爬虫抓取工具,可批量抓取任意站点的小说、论坛内容等并保存为TXT文档
  10. // @description:zh-TW 通用網站內容爬蟲抓取工具,可批量抓取任意站點的小說、論壇內容等並保存為TXT文檔
  11. // @description:ja 軽量なWebスクレイピングスクリプト。ユニバーサルサイトコンテンツクロールツール、クロール、フォーラム内容など
  12. // @author hoothin
  13. // @match http://*/*
  14. // @match https://*/*
  15. // @match ftp://*/*
  16. // @grant GM_xmlhttpRequest
  17. // @grant GM_registerMenuCommand
  18. // @grant GM_setValue
  19. // @grant GM_getValue
  20. // @grant GM_openInTab
  21. // @grant GM_setClipboard
  22. // @grant GM_addStyle
  23. // @grant unsafeWindow
  24. // @license MIT License
  25. // @compatible chrome
  26. // @compatible firefox
  27. // @compatible opera 未测试
  28. // @compatible safari 未测试
  29. // @contributionURL https://ko-fi.com/hoothin
  30. // @contributionAmount 1
  31. // ==/UserScript==
  32.  
  33. if (window.top != window.self) {
  34. try {
  35. if (window.self.innerWidth < 250 || window.self.innerHeight < 250) {
  36. return;
  37. }
  38. } catch(e) {
  39. return;
  40. }
  41. }
  42.  
  43. (function (global, factory) {
  44. if (typeof define === "function" && define.amd) {
  45. define([], factory);
  46. } else if (typeof exports !== "undefined") {
  47. factory();
  48. } else {
  49. var mod = {
  50. exports: {}
  51. };
  52. factory();
  53. global.FileSaver = mod.exports;
  54. }
  55. })(this, function () {
  56. "use strict";
  57.  
  58. /*
  59. * FileSaver.js
  60. * A saveAs() FileSaver implementation.
  61. *
  62. * By Eli Grey, http://eligrey.com
  63. *
  64. * License : https://github.com/eligrey/FileSaver.js/blob/master/LICENSE.md (MIT)
  65. * source : http://purl.eligrey.com/github/FileSaver.js
  66. */
  67. var _global = typeof window === 'object' && window.window === window ? window : typeof self === 'object' && self.self === self ? self : typeof global === 'object' && global.global === global ? global : void 0;
  68.  
  69. function bom(blob, opts) {
  70. if (typeof opts === 'undefined') opts = {
  71. autoBom: false
  72. };else if (typeof opts !== 'object') {
  73. console.warn('Deprecated: Expected third argument to be a object');
  74. opts = {
  75. autoBom: !opts
  76. };
  77. }
  78.  
  79. if (opts.autoBom && /^\s*(?:text\/\S*|application\/xml|\S*\/\S*\+xml)\s*;.*charset\s*=\s*utf-8/i.test(blob.type)) {
  80. return new Blob([String.fromCharCode(0xFEFF), blob], {
  81. type: blob.type
  82. });
  83. }
  84.  
  85. return blob;
  86. }
  87.  
  88. function download(url, name, opts) {
  89. var xhr = new XMLHttpRequest();
  90. xhr.open('GET', url);
  91. xhr.responseType = 'blob';
  92.  
  93. xhr.onload = function () {
  94. saveAs(xhr.response, name, opts);
  95. };
  96.  
  97. xhr.onerror = function () {
  98. console.error('could not download file');
  99. };
  100.  
  101. xhr.send();
  102. }
  103.  
  104. function corsEnabled(url) {
  105. var xhr = new XMLHttpRequest();
  106.  
  107. xhr.open('HEAD', url, false);
  108.  
  109. try {
  110. xhr.send();
  111. } catch (e) {}
  112.  
  113. return xhr.status >= 200 && xhr.status <= 299;
  114. }
  115.  
  116.  
  117. function click(node) {
  118. try {
  119. node.dispatchEvent(new MouseEvent('click'));
  120. } catch (e) {
  121. var evt = document.createEvent('MouseEvents');
  122. evt.initMouseEvent('click', true, true, window, 0, 0, 0, 80, 20, false, false, false, false, 0, null);
  123. node.dispatchEvent(evt);
  124. }
  125. }
  126.  
  127.  
  128. var isMacOSWebView = _global.navigator && /Macintosh/.test(navigator.userAgent) && /AppleWebKit/.test(navigator.userAgent) && !/Safari/.test(navigator.userAgent);
  129. var saveAs = _global.saveAs || (
  130. typeof window !== 'object' || window !== _global ? function saveAs() {}
  131.  
  132. : 'download' in HTMLAnchorElement.prototype && !isMacOSWebView ? function saveAs(blob, name, opts) {
  133. var URL = _global.URL || _global.webkitURL;
  134. var a = document.createElement('a');
  135. name = name || blob.name || 'download';
  136. a.download = name;
  137. a.rel = 'noopener';
  138.  
  139. if (typeof blob === 'string') {
  140. a.href = blob;
  141.  
  142. if (a.origin !== location.origin) {
  143. corsEnabled(a.href) ? download(blob, name, opts) : click(a, a.target = '_blank');
  144. } else {
  145. click(a);
  146. }
  147. } else {
  148. a.href = URL.createObjectURL(blob);
  149. setTimeout(function () {
  150. URL.revokeObjectURL(a.href);
  151. }, 4E4);
  152.  
  153. setTimeout(function () {
  154. click(a);
  155. }, 0);
  156. }
  157. }
  158. : 'msSaveOrOpenBlob' in navigator ? function saveAs(blob, name, opts) {
  159. name = name || blob.name || 'download';
  160.  
  161. if (typeof blob === 'string') {
  162. if (corsEnabled(blob)) {
  163. download(blob, name, opts);
  164. } else {
  165. var a = document.createElement('a');
  166. a.href = blob;
  167. a.target = '_blank';
  168. setTimeout(function () {
  169. click(a);
  170. });
  171. }
  172. } else {
  173. navigator.msSaveOrOpenBlob(bom(blob, opts), name);
  174. }
  175. }
  176. : function saveAs(blob, name, opts, popup) {
  177. popup = popup || open('', '_blank');
  178.  
  179. if (popup) {
  180. popup.document.title = popup.document.body.innerText = 'downloading...';
  181. }
  182.  
  183. if (typeof blob === 'string') return download(blob, name, opts);
  184. var force = blob.type === 'application/octet-stream';
  185.  
  186. var isSafari = /constructor/i.test(_global.HTMLElement) || _global.safari;
  187.  
  188. var isChromeIOS = /CriOS\/[\d]+/.test(navigator.userAgent);
  189.  
  190. if ((isChromeIOS || force && isSafari || isMacOSWebView) && typeof FileReader !== 'undefined') {
  191. var reader = new FileReader();
  192.  
  193. reader.onloadend = function () {
  194. var url = reader.result;
  195. url = isChromeIOS ? url : url.replace(/^data:[^;]*;/, 'data:attachment/file;');
  196. if (popup) popup.location.href = url;else location = url;
  197. popup = null;
  198. };
  199.  
  200. reader.readAsDataURL(blob);
  201. } else {
  202. var URL = _global.URL || _global.webkitURL;
  203. var url = URL.createObjectURL(blob);
  204. if (popup) popup.location = url;else location.href = url;
  205. popup = null;
  206.  
  207. setTimeout(function () {
  208. URL.revokeObjectURL(url);
  209. }, 4E4);
  210. }
  211. });
  212. _global.saveAs = saveAs.saveAs = saveAs;
  213.  
  214. if (typeof module !== 'undefined') {
  215. module.exports = saveAs;
  216. }
  217. });
  218.  
  219. (function() {
  220. 'use strict';
  221. var indexReg=/^(\w.*)?PART\b|^Prologue|^(\w.*)?Chapter\s*[\-_]?\d+|分卷|^序$|^序\s*[·言章]|^前\s*言|^附\s*[录錄]|^引\s*[言子]|^摘\s*要|^[楔契]\s*子|^后\s*记|^後\s*記|^附\s*言|^结\s*语|^結\s*語|^尾\s*[声聲]|^最終話|^最终话|^番\s*外|^\d+[\s\.、,,)\-_::][^\d#\.]|^(\d|\s|\.)*[第(]?\s*[\d〇零一二两三四五六七八九十百千万萬-]+\s*[、)章节節回卷折篇幕集话話]/i;
  222. var innerNextPage=/^\s*(下一[页頁张張]|next\s*page|次のページ)/i;
  223. var lang=navigator.appName=="Netscape"?navigator.language:navigator.userLanguage;
  224. var i18n={};
  225. var rCats=[];
  226. var processFunc, nextPageFunc;
  227. const AsyncFunction=Object.getPrototypeOf(async function(){}).constructor;
  228. var win=(typeof unsafeWindow=='undefined'?window:unsafeWindow);
  229. switch (lang){
  230. case "zh-CN":
  231. case "zh-SG":
  232. i18n={
  233. fetch:"开始下载小说",
  234. info:"来源:#t#\n本文是使用怠惰小说下载器(DownloadAllContent)下载的",
  235. error:"该段内容获取失败",
  236. downloading:"已下载完成 %s 段,剩余 %s 段<br>正在下载 %s",
  237. complete:"已全部下载完成,共 %s 段",
  238. del:"设置文本干扰码的CSS选择器",
  239. custom:"自定规则下载",
  240. customInfo:"输入网址或者章节CSS选择器",
  241. reSort:"按标题名重新排序章节",
  242. reSortUrl:"按网址重新排序章节",
  243. setting:"选项参数设置",
  244. searchRule:"搜索网站规则",
  245. abort:"跳过此章",
  246. save:"保存当前",
  247. saveAsMd:"存为 Markdown",
  248. saveAsJSON: "存为 JSON",
  249. downThreadNum:"设置同时下载的线程数,负数为单线程下载间隔",
  250. enableTouch:"在移动端按→↓←↑的方向滑动屏幕画正方形立即开始下载",
  251. customTitle:"自定义章节标题,输入内页文字对应选择器",
  252. saveUrl: "储存 URL",
  253. disableAutoStartSave: "禁用自动保存",
  254. maxDlPerMin:"每分钟最大下载数",
  255. reSortDefault:"默认按页面中位置排序章节",
  256. reverseOrder:"反转章节排序",
  257. saveBtn:"保存设置",
  258. saveOk:"保存成功",
  259. nextPage:"嗅探章节内分页",
  260. nextPageReg:"自定义分页正则",
  261. retainImage:"保留正文中图片的网址",
  262. minTxtLength:"当检测到的正文字数小于此数,则尝试重新抓取",
  263. showFilterList:"下载前显示章节筛选排序窗口",
  264. ok:"确定",
  265. close:"关闭",
  266. dacSortByPos:"按页内位置排序",
  267. dacSortByUrl:"按网址排序",
  268. dacSortByName:"按章节名排序",
  269. reverse:"反选",
  270. dacUseIframe:"使用 iframe 后台加载内容(慢速)",
  271. dacSaveAsZip:"下载为 zip",
  272. dacSetCustomRule:"修改规则",
  273. dacAddUrl:"添加章节",
  274. prefix:"给章节名称添加前缀",
  275. dacStartDownload:"下载选中",
  276. downloadShortcut:"下载章节快捷键",
  277. downloadSingleShortcut:"下载单页快捷键",
  278. downloadCustomShortcut:"自定义下载快捷键"
  279. };
  280. break;
  281. case "zh":
  282. case "zh-TW":
  283. case "zh-HK":
  284. i18n={
  285. fetch:"開始下載小說",
  286. info:"來源:#t#\n本文是使用怠惰小說下載器(DownloadAllContent)下載的",
  287. error:"該段內容獲取失敗",
  288. downloading:"已下載完成 %s 段,剩餘 %s 段<br>正在下載 %s",
  289. complete:"已全部下載完成,共 %s 段",
  290. del:"設置文本干擾碼的CSS選擇器",
  291. custom:"自訂規則下載",
  292. customInfo:"輸入網址或者章節CSS選擇器",
  293. reSort:"按標題名重新排序章節",
  294. reSortUrl:"按網址重新排序章節",
  295. setting:"選項參數設定",
  296. searchRule:"搜尋網站規則",
  297. abort:"跳過此章",
  298. save:"保存當前",
  299. saveAsMd:"存爲 Markdown",
  300. saveAsJSON: "存爲 JSON",
  301. downThreadNum:"設置同時下載的綫程數,負數為單線程下載間隔",
  302. enableTouch:"在行動端按→↓←↑的方向滑動螢幕畫方立即開始下載",
  303. customTitle:"自訂章節標題,輸入內頁文字對應選擇器",
  304. saveUrl: "儲存 URL",
  305. disableAutoStartSave: "禁用自動保存",
  306. maxDlPerMin:"每分鐘最大下載數",
  307. reSortDefault:"預設依頁面中位置排序章節",
  308. reverseOrder:"反轉章節排序",
  309. saveBtn:"儲存設定",
  310. saveOk:"儲存成功",
  311. nextPage:"嗅探章節內分頁",
  312. nextPageReg:"自訂分頁正規",
  313. retainImage:"保留內文圖片的網址",
  314. minTxtLength:"當偵測到的正文字數小於此數,則嘗試重新抓取",
  315. showFilterList:"下載前顯示章節篩選排序視窗",
  316. ok:"確定",
  317. close:"關閉",
  318. dacSortByPos:"依頁內位置排序",
  319. dacSortByUrl:"依網址排序",
  320. dacSortByName:"依章節名排序",
  321. reverse:"反選",
  322. dacUseIframe:"使用 iframe 背景載入內容(慢速)",
  323. dacSaveAsZip:"下載為 zip",
  324. dacSetCustomRule:"修改規則",
  325. dacAddUrl:"新增章節",
  326. prefix:"為章節名稱加上前綴",
  327. dacStartDownload:"下載選取",
  328. downloadShortcut:"下載章節快速鍵",
  329. downloadSingleShortcut:"下載單頁快速鍵",
  330. downloadCustomShortcut:"自設下載快速鍵"
  331. };
  332. break;
  333. case "ar":
  334. case "ar-AE":
  335. case "ar-BH":
  336. case "ar-DZ":
  337. case "ar-EG":
  338. case "ar-IQ":
  339. case "ar-JO":
  340. case "ar-KW":
  341. case "ar-LB":
  342. case "ar-LY":
  343. case "ar-MA":
  344. case "ar-OM":
  345. case "ar-QA":
  346. case "ar-SA":
  347. case "ar-SY":
  348. case "ar-TN":
  349. case "ar-YE":
  350. i18n={
  351. encode: true,
  352. fetch: "%D8%AA%D8%AD%D9%85%D9%8A%D9%84",
  353. info: "%D8%A7%D9%84%D9%85%D8%B5%D8%AF%D8%B1:%20#t#%0A%D8%AA%D9%85%20%D8%AA%D9%86%D8%B2%D9%8A%D9%84%20%D8%A7%D9%84%D9%80%20TXT%20%D8%A8%D9%88%D8%A7%D8%B3%D8%B7%D8%A9%20'DownloadAllContent'",
  354. error: "%D9%81%D8%B4%D9%84%20%D9%81%D9%8A%20%D8%AA%D8%AD%D9%85%D9%8A%D9%84%20%D8%A7%D9%84%D9%81%D8%B5%D9%84%20%D8%A7%D9%84%D8%AD%D8%A7%D9%84%D9%8A",
  355. downloading: "......%25s%20%D8%AA%D8%AD%D9%85%D9%8A%D9%84%3Cbr%3E%D8%B5%D9%81%D8%AD%D8%A7%D8%AA%20%D9%85%D8%AA%D8%A8%D9%82%D9%8A%D8%A9%20%25s%20%D8%B5%D9%81%D8%AD%D8%A7%D8%AA%20%D8%AA%D9%85%20%D8%AA%D8%AD%D9%85%D9%8A%D9%84%D9%87%D8%A7%D8%8C%20%D9%87%D9%86%D8%A7%D9%83%20%25s",
  356. complete: "%D8%B5%D9%81%D8%AD%D8%A7%D8%AA%20%D9%81%D9%8A%20%D8%A7%D9%84%D9%85%D8%AC%D9%85%D9%88%D8%B9%20%25s%20%D8%A7%D9%83%D8%AA%D9%85%D9%84!%20%D8%AD%D8%B5%D9%84%D8%AA%20%D8%B9%D9%84%D9%89",
  357. del: "%D9%84%D8%AA%D8%AC%D8%A7%D9%87%D9%84%20CSS%20%D8%AA%D8%B9%D9%8A%D9%8A%D9%86%20%D9%85%D8%AD%D8%AF%D8%AF%D8%A7%D8%AA",
  358. custom: "%D8%AA%D8%AD%D9%85%D9%8A%D9%84%20%D9%85%D8%AE%D8%B5%D8%B5",
  359. customInfo: "%D9%84%D8%B1%D9%88%D8%A7%D8%A8%D8%B7%20%D8%A7%D9%84%D9%81%D8%B5%D9%88%D9%84%20sss%20%D8%A5%D8%AF%D8%AE%D8%A7%D9%84%20%D8%A7%D9%84%D8%B1%D9%88%D8%A7%D8%A8%D8%B7%20%D8%A3%D9%88%20%D9%85%D8%AD%D8%AF%D8%AF%D8%A7%D8%AA",
  360. reSort: "%D8%A5%D8%B9%D8%A7%D8%AF%D8%A9%20%D8%A7%D9%84%D8%AA%D8%B1%D8%AA%D9%8A%D8%A8%20%D8%AD%D8%B3%D8%A8%20%D8%A7%D9%84%D8%B9%D9%86%D9%88%D8%A7%D9%86",
  361. reSortUrl: "%D8%A5%D8%B9%D8%A7%D8%AF%D8%A9%20%D8%A7%D9%84%D8%AA%D8%B1%D8%AA%D9%8A%D8%A8%20%D8%AD%D8%B3%D8%A8%20%D8%A7%D9%84%D8%B1%D9%88%D8%A7%D8%A8%D8%B7",
  362. setting: "%D9%81%D8%AA%D8%AD%20%D8%A7%D9%84%D8%A5%D8%B9%D8%AF%D8%A7%D8%AF%D8%A7%D8%AA",
  363. searchRule: "%D9%82%D8%A7%D8%B9%D8%AF%D8%A9%20%D8%A7%D9%84%D8%A8%D8%AD%D8%AB",
  364. abort: "%D8%A5%D9%8A%D9%82%D8%A7%D9%81",
  365. save: "%D8%AD%D9%81%D8%B8",
  366. saveAsMd: "Markdown%20%D8%AD%D9%81%D8%B8%20%D9%83%D9%80",
  367. saveAsJSON: "JSON%20%D8%AD%D9%81%D8%B8%20%D9%83%D9%80",
  368. downThreadNum: "%D8%AA%D8%B9%D9%8A%D9%8A%D9%86%20%D8%B9%D8%AF%D8%AF%20%D8%A7%D9%84%D8%AE%D9%8A%D9%88%D8%B7%20%D9%84%D9%84%D8%AA%D8%AD%D9%85%D9%8A%D9%84",
  369. enableTouch: "On%20the%20mobile%20device,%20slide%20the%20screen%20in%20the%20direction%20of%20%E2%86%92%E2%86%93%E2%86%90%E2%86%91%20to%20draw%20a%20square%20will%20start%20downloading%20immediately",
  370. customTitle: "%D8%AA%D8%AE%D8%B5%D9%8A%D8%B5%20%D8%B9%D9%86%D9%88%D8%A7%D9%86%20%D8%A7%D9%84%D9%81%D8%B5%D9%84%D8%8C%20%D8%A5%D8%AF%D8%AE%D8%A7%D9%84%20%D8%A7%D9%84%D9%85%D8%AD%D8%AF%D8%AF%20%D9%81%D9%8A%20%D8%A7%D9%84%D8%B5%D9%81%D8%AD%D8%A9%20%D8%A7%D9%84%D8%AF%D8%A7%D8%AE%D9%84%D9%8A%D8%A9",
  371. saveUrl: "%D8%AD%D9%81%D8%B8%20URL",
  372. disableAutoStartSave: "%D8%AA%D8%B9%D8%B7%D9%8A%D9%84%20%D8%A7%D9%84%D8%AD%D9%81%D8%B8%20%D8%A7%D9%84%D8%AA%D9%84%D9%82%D8%A7%D8%A6%D9%8A",
  373. maxDlPerMin: "%D8%A7%D9%84%D8%AD%D8%AF%20%D8%A7%D9%84%D8%A3%D9%82%D8%B5%D9%89%20%D9%84%D8%B9%D8%AF%D8%AF%20%D8%A7%D9%84%D8%AA%D9%86%D8%B2%D9%8A%D9%84%D8%A7%D8%AA%20%D9%81%D9%8A%20%D8%A7%D9%84%D8%AF%D9%82%D9%8A%D9%82%D8%A9",
  374. reSortDefault: "%D8%A7%D9%84%D8%AA%D8%B1%D8%AA%D9%8A%D8%A8%20%D8%A7%D9%84%D8%A7%D9%81%D8%AA%D8%B1%D8%A7%D8%B6%D9%8A%20%D8%AD%D8%B3%D8%A8%20%D8%A7%D9%84%D9%85%D9%88%D9%82%D8%B9%20%D9%81%D9%8A%20%D8%A7%D9%84%D8%B5%D9%81%D8%AD%D8%A9",
  375. reverseOrder: "%D8%B9%D9%83%D8%B3%20%D8%AA%D8%B1%D8%AA%D9%8A%D8%A8%20%D8%A7%D9%84%D9%81%D8%B5%D9%88%D9%84",
  376. saveBtn: "%D8%AD%D9%81%D8%B8%20%D8%A7%D9%84%D8%A5%D8%B9%D8%AF%D8%A7%D8%AF%D8%A7%D8%AA",
  377. saveOk: "%D8%AA%D9%85%20%D8%A7%D9%84%D8%AD%D9%81%D8%B8",
  378. nextPage: "%D8%A7%D9%84%D8%AA%D8%AD%D9%82%D9%82%20%D9%85%D9%86%20%D8%A7%D9%84%D8%B5%D9%81%D8%AD%D8%A9%20%D8%A7%D9%84%D8%AA%D8%A7%D9%84%D9%8A%D8%A9%20%D9%81%D9%8A%20%D8%A7%D9%84%D9%81%D8%B5%D9%84",
  379. nextPageReg: "%D9%85%D8%AE%D8%B5%D8%B5%20%D9%84%D9%84%D8%B5%D9%81%D8%AD%D8%A9%20%D8%A7%D9%84%D8%AA%D8%A7%D9%84%D9%8A%D8%A9%20RegExp",
  380. retainImage: "%D8%A7%D9%84%D8%A7%D8%AD%D8%AA%D9%81%D8%A7%D8%B8%20%D8%A8%D8%B9%D9%86%D9%88%D8%A7%D9%86%20%D8%A7%D9%84%D8%B5%D9%88%D8%B1%D8%A9%20%D8%A5%D8%B0%D8%A7%20%D9%83%D8%A7%D9%86%D8%AA%20%D9%87%D9%86%D8%A7%D9%83%20%D8%B5%D9%88%D8%B1%20%D9%81%D9%8A%20%D8%A7%D9%84%D9%86%D8%B5",
  381. minTxtLength: "%D8%A7%D9%84%D9%85%D8%AD%D8%A7%D9%88%D9%84%D8%A9%20%D9%85%D8%B1%D8%A9%20%D8%A3%D8%AE%D8%B1%D9%89%20%D8%B9%D9%86%D8%AF%D9%85%D8%A7%20%D9%8A%D9%83%D9%88%D9%86%20%D8%B7%D9%88%D9%84%20%D8%A7%D9%84%D9%85%D8%AD%D8%AA%D9%88%D9%89%20%D8%A3%D9%82%D9%84%20%D9%85%D9%86%20%D9%87%D8%B0%D8%A7",
  382. showFilterList: "%D8%B9%D8%B1%D8%B6%20%D9%86%D8%A7%D9%81%D8%B0%D8%A9%20%D8%A7%D9%84%D8%AA%D8%B5%D9%81%D9%8A%D8%A9%20%D9%88%D8%A7%D9%84%D8%AA%D8%B1%D8%AA%D9%8A%D8%A8%20%D9%82%D8%A8%D9%84%20%D8%A7%D9%84%D8%AA%D8%AD%D9%85%D9%8A%D9%84",
  383. ok: "%D9%85%D9%88%D8%A7%D9%81%D9%82",
  384. close: "%D8%A5%D8%BA%D9%84%D8%A7%D9%82",
  385. dacSortByPos: "%D8%A7%D9%84%D8%AA%D8%B1%D8%AA%D9%8A%D8%A8%20%D8%AD%D8%B3%D8%A8%20%D8%A7%D9%84%D9%85%D9%88%D9%82%D8%B9",
  386. dacSortByUrl: "%D8%A7%D9%84%D8%AA%D8%B1%D8%AA%D9%8A%D8%A8%20%D8%AD%D8%B3%D8%A8%20%D8%A7%D9%84%D8%B1%D8%A7%D8%A8%D8%B7",
  387. dacSortByName: "%D8%A7%D9%84%D8%AA%D8%B1%D8%AA%D9%8A%D8%A8%20%D8%AD%D8%B3%D8%A8%20%D8%A7%D9%84%D8%A7%D8%B3%D9%85",
  388. reverse: "%D8%B9%D9%83%D8%B3%20%D8%A7%D9%84%D8%A7%D8%AE%D8%AA%D9%8A%D8%A7%D8%B1",
  389. dacUseIframe: "%D9%84%D8%AA%D8%AD%D9%85%D9%8A%D9%84%20%D8%A7%D9%84%D9%85%D8%AD%D8%AA%D9%88%D9%89%20(%D8%A8%D8%B7%D9%8A%D8%A1)%20iframe%20%D8%A7%D8%B3%D8%AA%D8%AE%D8%AF%D8%A7%D9%85",
  390. dacSaveAsZip: "zip%20%D8%AD%D9%81%D8%B8%20%D9%83%D9%80",
  391. dacSetCustomRule: "%D8%AA%D8%B9%D8%AF%D9%8A%D9%84%20%D8%A7%D9%84%D9%82%D9%88%D8%A7%D8%B9%D8%AF",
  392. dacAddUrl: "%D8%A5%D8%B6%D8%A7%D9%81%D8%A9%20%D9%81%D8%B5%D9%84",
  393. prefix:"Prefix%20of%20chapter%20name",
  394. dacStartDownload: "%D8%AA%D8%AD%D9%85%D9%8A%D9%84%20%D8%A7%D9%84%D9%85%D8%AD%D8%AF%D8%AF",
  395. downloadShortcut: "%D8%AA%D8%AD%D9%85%D9%8A%D9%84%20%D8%A7%D9%84%D9%81%D8%B5%D9%84",
  396. downloadSingleShortcut: "%D8%AA%D8%AD%D9%85%D9%8A%D9%84%20%D8%B5%D9%81%D8%AD%D8%A9%20%D9%88%D8%A7%D8%AD%D8%AF%D8%A9",
  397. downloadCustomShortcut: "%D8%AA%D8%AD%D9%85%D9%8A%D9%84%20%D9%85%D8%AE%D8%B5%D8%B5"
  398. };
  399. break;
  400. default:
  401. i18n={
  402. fetch:"Download",
  403. info:"Source: #t#\nThe TXT is downloaded by 'DownloadAllContent'",
  404. error:"Failed in downloading current chapter",
  405. downloading:"%s pages are downloaded, there are still %s pages left<br>Downloading %s ......",
  406. complete:"Completed! Get %s pages in total",
  407. del:"Set css selectors for ignore",
  408. custom:"Custom to download",
  409. customInfo:"Input urls OR sss selectors for chapter links",
  410. reSort:"ReSort by title",
  411. reSortUrl:"Resort by URLs",
  412. setting:"Open Setting",
  413. searchRule:"Search rule",
  414. abort:"Abort",
  415. save:"Save",
  416. saveAsMd:"Save as Markdown",
  417. saveAsJSON: "Save as JSON",
  418. downThreadNum:"Set threadNum for download, negative means interval of single thread",
  419. enableTouch:"On the mobile device, slide the screen in the direction of →↓←↑ to draw a square will start downloading immediately",
  420. customTitle: "Customize the chapter title, enter the selector on inner page",
  421. saveUrl: "Save URL",
  422. disableAutoStartSave: "Disable auto save",
  423. maxDlPerMin:"Maximum number of downloads per minute",
  424. reSortDefault: "Default sort by position in the page",
  425. reverseOrder:"Reverse chapter ordering",
  426. saveBtn:"Save Setting",
  427. saveOk:"Save Over",
  428. nextPage:"Check next page in chapter",
  429. nextPageReg:"Custom RegExp of next page",
  430. retainImage:"Keep the URL of image if there are images in the text",
  431. minTxtLength:"Try to crawl again when the length of content is less than this",
  432. showFilterList: "Show chapter filtering and sorting window before downloading",
  433. ok:"OK",
  434. close:"Close",
  435. dacSortByPos:"Sort by position",
  436. dacSortByUrl:"Sort by URL",
  437. dacSortByName:"Sort by name",
  438. reverse:"Reverse selection",
  439. dacUseIframe: "Use iframe to load content (slow)",
  440. dacSaveAsZip: "Save as zip",
  441. dacSetCustomRule:"Modify rules",
  442. dacAddUrl:"Add Chapter",
  443. prefix:"Prefix of chapter name",
  444. dacStartDownload:"Download selected",
  445. downloadShortcut:"Download chapter Shortcut",
  446. downloadSingleShortcut:"Download single page Shortcut",
  447. downloadCustomShortcut:"Custom download Shortcut"
  448. };
  449. break;
  450. }
  451. if (i18n.encode) {
  452. for (let k in i18n) {
  453. if (k != "encode") {
  454. i18n[k] = decodeURI(i18n[k]);
  455. }
  456. }
  457. }
  458. var firefox=navigator.userAgent.toLowerCase().indexOf('firefox')!=-1,curRequests=[],useIframe=false,iframeSandbox=false,iframeInit=false;
  459. var filterListContainer,txtDownContent,txtDownWords,txtDownQuit,dacLinksCon,dacUseIframe,shadowContainer,downTxtShadowContainer;
  460.  
  461. const escapeHTMLPolicy = (win.trustedTypes && win.trustedTypes.createPolicy) ? win.trustedTypes.createPolicy('dac_default', {
  462. createHTML: (string, sink) => string
  463. }) : null;
  464.  
  465. function createHTML(html) {
  466. return escapeHTMLPolicy ? escapeHTMLPolicy.createHTML(html) : html;
  467. }
  468.  
  469. function str2Num(str) {
  470. str = str.replace(/^番\s*外/, "99999+").replace(/[一①Ⅰ壹]/g, "1").replace(/[二②Ⅱ贰]/g, "2").replace(/[三③Ⅲ叁]/g, "3").replace(/[四④Ⅳ肆]/g, "4").replace(/[五⑤Ⅴ伍]/g, "5").replace(/[六⑥Ⅵ陆]/g, "6").replace(/[七⑦Ⅶ柒]/g, "7").replace(/[八⑧Ⅷ捌]/g, "8").replace(/[九⑨Ⅸ玖]/g, "9").replace(/[十⑩Ⅹ拾]/g, "*10+").replace(/[百佰]/g, "*100+").replace(/[千仟]/g, "*1000+").replace(/[万萬]/g, "*10000+").replace(/\s/g, "").match(/[\d\*\+]+/);
  471. if (!str) return 0;
  472. str = str[0];
  473. let mul = str.match(/(\d*)\*(\d+)/);
  474. while(mul) {
  475. let result = parseInt(mul[1] || 1) * parseInt(mul[2]);
  476. str = str.replace(mul[0], result);
  477. mul = str.match(/(\d+)\*(\d+)/);
  478. }
  479. let plus = str.match(/(\d+)\+(\d+)/);
  480. while(plus) {
  481. let result = parseInt(plus[1]) + parseInt(plus[2]);
  482. str = str.replace(plus[0], result);
  483. plus = str.match(/(\d+)\+(\d+)/);
  484. }
  485. return parseInt(str);
  486. }
  487.  
  488. var dragOverItem, dragFrom, linkDict;
  489. function createLinkItem(aEle) {
  490. let item = document.createElement("div");
  491. item.innerHTML = createHTML(`
  492. <input type="checkbox" checked>
  493. <a class="dacLink" draggable="false" target="_blank" href="${aEle.href}">${aEle.innerText || "📄"}</a>
  494. <span>🖱️</span>
  495. `);
  496. item.title = aEle.innerText;
  497. item.setAttribute("draggable", "true");
  498. item.addEventListener("dragover", e => {
  499. e.preventDefault();
  500. });
  501. item.addEventListener("dragenter", e => {
  502. if (dragOverItem) dragOverItem.style.opacity = "";
  503. item.style.opacity = 0.3;
  504. dragOverItem = item;
  505. });
  506. item.addEventListener('dragstart', e => {
  507. dragFrom = item;
  508. });
  509. item.addEventListener('drop', e => {
  510. if (!dragFrom) return;
  511. if (e.clientX < item.getBoundingClientRect().left + 142) {
  512. dacLinksCon.insertBefore(dragFrom, item);
  513. } else {
  514. if (item.nextElementSibling) {
  515. dacLinksCon.insertBefore(dragFrom, item.nextElementSibling);
  516. } else {
  517. dacLinksCon.appendChild(dragFrom);
  518. }
  519. }
  520. e.preventDefault();
  521. });
  522. linkDict[aEle.href] = item;
  523. dacLinksCon.appendChild(item);
  524. }
  525.  
  526. var saveAsZip = true;
  527. function filterList(list) {
  528. if (!GM_getValue("showFilterList")) {
  529. indexDownload(list);
  530. return;
  531. }
  532. if (txtDownContent) {
  533. txtDownContent.style.display = "none";
  534. }
  535. if (filterListContainer) {
  536. filterListContainer.style.display = "";
  537. filterListContainer.classList.remove("customRule");
  538. dacLinksCon.innerHTML = createHTML("");
  539. } else {
  540. document.addEventListener('dragend', e => {
  541. if (dragOverItem) dragOverItem.style.opacity = "";
  542. }, true);
  543. filterListContainer = document.createElement("div");
  544. filterListContainer.id = "filterListContainer";
  545. filterListContainer.innerHTML = createHTML(`
  546. <div id="dacFilterBg" style="height: 100%; width: 100%; position: fixed; top: 0; z-index: 2147483646; opacity: 0.3; filter: alpha(opacity=30); background-color: #000;"></div>
  547. <div id="filterListBody">
  548. <div class="dacCustomRule">
  549. ${i18n.custom}
  550. <textarea id="dacCustomInput"></textarea>
  551. <div class="fun">
  552. <input id="dacConfirmRule" value="${i18n.ok}" type="button"/>
  553. <input id="dacCustomClose" value="${i18n.close}" type="button"/>
  554. </div>
  555. </div>
  556. <div class="sort">
  557. <input id="dacSortByPos" value="${i18n.dacSortByPos}" type="button"/>
  558. <input id="dacSortByUrl" value="${i18n.dacSortByUrl}" type="button"/>
  559. <input id="dacSortByName" value="${i18n.dacSortByName}" type="button"/>
  560. <input id="reverse" value="${i18n.reverse}" type="button"/>
  561. </div>
  562. <div id="dacLinksCon" style="max-height: calc(80vh - 100px); min-height: 100px; display: grid; grid-template-columns: auto auto; width: 100%; overflow: auto; white-space: nowrap;"></div>
  563. <p style="margin: 5px; text-align: center; font-size: 14px; height: 20px;"><span><input id="dacUseIframe" type="checkbox"/><label for="dacUseIframe"> ${i18n.dacUseIframe}</label></span> <span style="display:${win.downloadAllContentSaveAsZip ? "inline" : "none"}"><input id="dacSaveAsZip" type="checkbox" checked="checked"/><label for="dacSaveAsZip"> ${i18n.dacSaveAsZip}</label></span></p>
  564. <div class="fun">
  565. <input id="dacSetCustomRule" value="${i18n.dacSetCustomRule}" type="button"/>
  566. <input id="dacAddUrl" value="${i18n.dacAddUrl}" type="button"/>
  567. <input id="dacStartDownload" value="${i18n.dacStartDownload}" type="button"/>
  568. <input id="dacLinksClose" value="${i18n.close}" type="button"/>
  569. </div>
  570. </div>`);
  571. let dacSortByPos = filterListContainer.querySelector("#dacSortByPos");
  572. let dacSortByUrl = filterListContainer.querySelector("#dacSortByUrl");
  573. let dacSortByName = filterListContainer.querySelector("#dacSortByName");
  574. let reverse = filterListContainer.querySelector("#reverse");
  575. let dacSetCustomRule = filterListContainer.querySelector("#dacSetCustomRule");
  576. let dacCustomInput = filterListContainer.querySelector("#dacCustomInput");
  577. let dacConfirmRule = filterListContainer.querySelector("#dacConfirmRule");
  578. let dacCustomClose = filterListContainer.querySelector("#dacCustomClose");
  579. let dacAddUrl = filterListContainer.querySelector("#dacAddUrl");
  580. let dacStartDownload = filterListContainer.querySelector("#dacStartDownload");
  581. let dacLinksClose = filterListContainer.querySelector("#dacLinksClose");
  582. let dacFilterBg = filterListContainer.querySelector("#dacFilterBg");
  583. let dacSaveAsZip = filterListContainer.querySelector("#dacSaveAsZip");
  584. dacUseIframe = filterListContainer.querySelector("#dacUseIframe");
  585. dacSaveAsZip.onchange = e => {
  586. saveAsZip = dacSaveAsZip.checked;
  587. };
  588. dacSortByPos.onclick = e => {
  589. let linkList = [].slice.call(dacLinksCon.children);
  590. if (linkList[0].children[1].href != list[0].href) {
  591. list.reverse().forEach(a => {
  592. let link = linkDict[a.href];
  593. if (!link) return;
  594. dacLinksCon.insertBefore(link, dacLinksCon.children[0]);
  595. });
  596. } else {
  597. list.forEach(a => {
  598. let link = linkDict[a.href];
  599. if (!link) return;
  600. dacLinksCon.insertBefore(link, dacLinksCon.children[0]);
  601. });
  602. }
  603. };
  604. dacSortByUrl.onclick = e => {
  605. let linkList = [].slice.call(dacLinksCon.children);
  606. linkList.sort((a, b) => {
  607. const nameA = a.children[1].href.toUpperCase();
  608. const nameB = b.children[1].href.toUpperCase();
  609. if (nameA < nameB) {
  610. return -1;
  611. }
  612. if (nameA > nameB) {
  613. return 1;
  614. }
  615. return 0;
  616. });
  617. if (linkList[0] == dacLinksCon.children[0]) {
  618. linkList = linkList.reverse();
  619. }
  620. linkList.forEach(link => {
  621. dacLinksCon.appendChild(link);
  622. });
  623. };
  624. dacSortByName.onclick = e => {
  625. let linkList = [].slice.call(dacLinksCon.children);
  626. linkList.sort((a, b) => {
  627. return str2Num(a.innerText) - str2Num(b.innerText);
  628. });
  629. if (linkList[0] == dacLinksCon.children[0]) {
  630. linkList = linkList.reverse();
  631. }
  632. linkList.forEach(link => {
  633. dacLinksCon.appendChild(link);
  634. });
  635. };
  636. reverse.onclick = e => {
  637. let linkList = [].slice.call(dacLinksCon.children);
  638. linkList.forEach(link => {
  639. link.children[0].checked=!link.children[0].checked;
  640. });
  641. };
  642. dacSetCustomRule.onclick = e => {
  643. filterListContainer.classList.add("customRule");
  644. dacCustomInput.value = GM_getValue("DACrules_" + document.domain) || "";
  645. };
  646. dacConfirmRule.onclick = e => {
  647. if (dacCustomInput.value) {
  648. customDown(dacCustomInput.value);
  649. }
  650. };
  651. dacCustomClose.onclick = e => {
  652. filterListContainer.classList.remove("customRule");
  653. };
  654. dacAddUrl.onclick = e => {
  655. let addUrls = window.prompt(i18n.customInfo, "https://xxx.xxx/book-[20-99].html, https://xxx.xxx/book-[01-10].html");
  656. if (!addUrls || !/^http|^ftp/.test(addUrls)) return;
  657. let index = 1;
  658. [].forEach.call(addUrls.split(","), function(i) {
  659. var curEle;
  660. var varNum = /\[\d+\-\d+\]/.exec(i);
  661. if (varNum) {
  662. varNum = varNum[0].trim();
  663. } else {
  664. curEle = document.createElement("a");
  665. curEle.href = i;
  666. curEle.innerText = "Added Url";
  667. createLinkItem(curEle);
  668. return;
  669. }
  670. var num1 = /\[(\d+)/.exec(varNum)[1].trim();
  671. var num2 = /(\d+)\]/.exec(varNum)[1].trim();
  672. var num1Int = parseInt(num1);
  673. var num2Int = parseInt(num2);
  674. var numLen = num1.length;
  675. var needAdd = num1.charAt(0) == "0";
  676. if (num1Int >= num2Int) return;
  677. for (var j = num1Int; j <= num2Int; j++) {
  678. var urlIndex = j.toString();
  679. if (needAdd) {
  680. while(urlIndex.length < numLen) urlIndex = "0" + urlIndex;
  681. }
  682. var curUrl = i.replace(/\[\d+\-\d+\]/, urlIndex).trim();
  683. curEle = document.createElement("a");
  684. curEle.href = curUrl;
  685. curEle.innerText = "Added Url " + index++;
  686. createLinkItem(curEle);
  687. }
  688. });
  689. };
  690. dacStartDownload.onclick = e => {
  691. let linkList = [].slice.call(dacLinksCon.querySelectorAll("input:checked+.dacLink"));
  692. useIframe = !!dacUseIframe.checked;
  693. indexDownload(linkList, true);
  694. };
  695. dacLinksClose.onclick = e => {
  696. filterListContainer.style.display = "none";
  697. };
  698. dacFilterBg.onclick = e => {
  699. filterListContainer.style.display = "none";
  700. };
  701. let listStyle = GM_addStyle(`
  702. #filterListContainer * {
  703. font-size: 13px;
  704. float: initial;
  705. background-image: initial;
  706. height: fit-content;
  707. color: black;
  708. }
  709. #filterListContainer.customRule .dacCustomRule {
  710. display: flex;
  711. }
  712. #filterListContainer .dacCustomRule>textarea {
  713. height: 300px;
  714. width: 100%;
  715. border: 1px #DADADA solid;
  716. background: #ededed70;
  717. margin: 5px;
  718. }
  719. #filterListContainer.customRule .dacCustomRule~* {
  720. display: none!important;
  721. }
  722. #dacLinksCon>div {
  723. padding: 5px 0;
  724. display: flex;
  725. }
  726. #dacLinksCon>div>a {
  727. max-width: 245px;
  728. display: inline-block;
  729. text-overflow: ellipsis;
  730. overflow: hidden;
  731. }
  732. #dacLinksCon>div>input {
  733. margin-right: 5px;
  734. }
  735. #filterListContainer .dacCustomRule {
  736. border-radius: 8px;
  737. font-weight: bold;
  738. font-size: 16px;
  739. outline: none;
  740. align-items: center;
  741. flex-wrap: nowrap;
  742. white-space: nowrap;
  743. flex-direction: column;
  744. display: none;
  745. }
  746. #filterListContainer input {
  747. border-width: 2px;
  748. border-style: outset;
  749. border-color: buttonface;
  750. border-image: initial;
  751. border: 1px #DADADA solid;
  752. padding: 5px;
  753. border-radius: 8px;
  754. font-weight: bold;
  755. font-size: 9pt;
  756. outline: none;
  757. cursor: pointer;
  758. line-height: initial;
  759. width: initial;
  760. min-width: initial;
  761. max-width: initial;
  762. height: initial;
  763. min-height: initial;
  764. max-height: initial;
  765. }
  766. #dacLinksCon>div:nth-of-type(4n),
  767. #dacLinksCon>div:nth-of-type(4n+1) {
  768. background: #ffffff;
  769. }
  770. #dacLinksCon>div:nth-of-type(4n+2),
  771. #dacLinksCon>div:nth-of-type(4n+3) {
  772. background: #f5f5f5;
  773. }
  774. #filterListContainer .fun,#filterListContainer .sort {
  775. display: flex;
  776. justify-content: space-around;
  777. flex-wrap: nowrap;
  778. width: 100%;
  779. height: 28px;
  780. }
  781. #filterListContainer input[type=button]:hover {
  782. border: 1px #C6C6C6 solid;
  783. box-shadow: 1px 1px 1px #EAEAEA;
  784. color: #333333;
  785. background: #F7F7F7;
  786. }
  787. #filterListContainer input[type=button]:active {
  788. box-shadow: inset 1px 1px 1px #DFDFDF;
  789. }
  790. #filterListBody {
  791. padding: 5px;
  792. box-sizing: border-box;
  793. overflow: hidden;
  794. width: 600px;
  795. height: auto;
  796. max-height: 80vh;
  797. min-height: 200px;
  798. position: fixed;
  799. left: 50%;
  800. top: 10%;
  801. margin-left: -300px;
  802. z-index: 2147483646;
  803. background-color: #ffffff;
  804. border: 1px solid #afb3b6;
  805. border-radius: 10px;
  806. opacity: 0.95;
  807. filter: alpha(opacity=95);
  808. box-shadow: 5px 5px 20px 0px #000;
  809. }
  810. @media screen and (max-width: 800px) {
  811. #filterListBody {
  812. width: 90%;
  813. margin-left: -45%;
  814. }
  815. }
  816. `);
  817. dacLinksCon = filterListContainer.querySelector("#dacLinksCon");
  818. shadowContainer = document.createElement("div");
  819. shadowContainer.id = "download-all-content";
  820. document.body.appendChild(shadowContainer);
  821. let shadow = shadowContainer.attachShadow({ mode: "open" });
  822. shadow.appendChild(listStyle);
  823. shadow.appendChild(filterListContainer);
  824. }
  825. if (shadowContainer.parentNode) shadowContainer.parentNode.removeChild(shadowContainer);
  826. linkDict = {};
  827. list.forEach(a => {
  828. createLinkItem(a);
  829. });
  830. dacUseIframe.checked = useIframe;
  831. document.body.appendChild(shadowContainer);
  832. }
  833.  
  834. function initTxtDownDiv() {
  835. if (txtDownContent) {
  836. txtDownContent.style.display = "";
  837. document.body.appendChild(downTxtShadowContainer);
  838. return;
  839. }
  840. txtDownContent = document.createElement("div");
  841. txtDownContent.id = "txtDownContent";
  842. downTxtShadowContainer = document.createElement("div");
  843. document.body.appendChild(downTxtShadowContainer);
  844. let shadow = downTxtShadowContainer.attachShadow({ mode: "open" });
  845. shadow.appendChild(txtDownContent);
  846. txtDownContent.innerHTML = createHTML(`
  847. <style>
  848. #txtDownContent>div{
  849. font-size:16px;
  850. color:#333333;
  851. width:342px;
  852. height:110px;
  853. position:fixed;
  854. left:50%;
  855. top:50%;
  856. margin-top:-25px;
  857. margin-left:-171px;
  858. z-index:2147483647;
  859. background-color:#ffffff;
  860. border:1px solid #afb3b6;
  861. border-radius:10px;
  862. opacity:0.95;
  863. filter:alpha(opacity=95);
  864. box-shadow:5px 5px 20px 0px #000;
  865. }
  866. #txtDownWords{
  867. position:absolute;
  868. width:275px;
  869. height: 90px;
  870. max-height: 90%;
  871. border: 1px solid #f3f1f1;
  872. padding: 8px;
  873. border-radius: 10px;
  874. overflow: auto;
  875. }
  876. #txtDownQuit{
  877. width: 30px;height: 30px;border-radius: 30px;position:absolute;right:2px;top:2px;cursor: pointer;background-color:#ff5a5a;
  878. }
  879. #txtDownQuit>span{
  880. height: 30px;line-height: 30px;display:block;color:#FFF;text-align:center;font-size: 12px;font-weight: bold;font-family: arial;background: initial; float: initial;
  881. }
  882. #txtDownQuit+div{
  883. position:absolute;right:0px;bottom:2px;cursor: pointer;max-width:85px;
  884. }
  885. #txtDownQuit+div>button{
  886. background: #008aff;border: 0;padding: 5px;border-radius: 6px;color: white;float: right;margin: 1px;height: 25px;line-height: 16px;cursor: pointer;overflow: hidden;
  887. }
  888. </style>
  889. <div>
  890. <div id="txtDownWords">
  891. Analysing......
  892. </div>
  893. <div id="txtDownQuit">
  894. <span>╳</span>
  895. </div>
  896. <div>
  897. <button id="abortRequest" style="display:none;">${getI18n('abort')}</button>
  898. <button id="tempSaveTxt">${getI18n('save')}</button>
  899. <button id="saveAsMd" title="${getI18n('saveAsMd')}">Markdown</button>
  900. <button id="saveAsJSON" title="${getI18n('saveAsJSON')}">JSON</button>
  901. </div>
  902. </div>`);
  903. txtDownWords=txtDownContent.querySelector("#txtDownWords");
  904. txtDownQuit=txtDownContent.querySelector("#txtDownQuit");
  905. txtDownQuit.onclick=function(){
  906. txtDownContent.style.display="none";
  907. };
  908. initTempSave(txtDownContent);
  909. win.txtDownWords = txtDownWords;
  910. }
  911.  
  912. function saveContent() {
  913. if (win.downloadAllContentSaveAsZip && saveAsZip) {
  914. win.downloadAllContentSaveAsZip(rCats, i18n.info.replace("#t#", location.href), content => {
  915. saveAs(content, document.title.replace(/[\*\/:<>\?\\\|\r\n,]/g, "_") + ".zip");
  916. });
  917. } else {
  918. var blob = new Blob([i18n.info.replace("#t#", location.href) + "\r\n\r\n" + rCats.join("\r\n\r\n")], {type: "text/plain;charset=utf-8"});
  919. saveAs(blob, document.title.replace(/[\*\/:<>\?\\\|\r\n,]/g, "_") + ".txt");
  920. }
  921. }
  922.  
  923. function initTempSave(txtDownContent){
  924. var tempSavebtn = txtDownContent.querySelector('#tempSaveTxt');
  925. var abortbtn = txtDownContent.querySelector('#abortRequest');
  926. var saveAsMd = txtDownContent.querySelector('#saveAsMd');
  927. var saveAsJSON = txtDownContent.querySelector('#saveAsJSON');
  928.  
  929. tempSavebtn.onclick = function(){
  930. saveContent();
  931. console.log(curRequests);
  932. }
  933. abortbtn.onclick = function(){
  934. let curRequest = curRequests.pop();
  935. if(curRequest)curRequest[1].abort();
  936. }
  937. saveAsMd.onclick = function(){
  938. let txt = i18n.info.replace("#t#", location.href)+"\n\n---\n"+document.title+"\n===\n";
  939. rCats.forEach(cat => {
  940. cat = cat.replace("\r\n", "\n---").replace(/(\r\n|\n\r)+/g, "\n\n").replace(/[\n\r]\t+/g, "\n");
  941. txt += '\n\n'+cat;
  942. });
  943. var blob = new Blob([txt], {type: "text/plain;charset=utf-8"});
  944. saveAs(blob, document.title.replace(/[\*\/:<>\?\\\|\r\n,]/g, "_") + ".md");
  945. }
  946. saveAsJSON.onclick = function(){
  947. let txt = [];
  948. rCats.forEach(cat => {
  949. let catArr = cat.split("\r\n", 3);
  950. let saveUrl = GM_getValue("saveUrl");
  951. let catJson = {
  952. title: catArr[0].trim(),
  953. content: catArr[1].trim()
  954. };
  955. if (saveUrl){
  956. catJson = {
  957. title: catArr[0].trim(),
  958. url: catArr[1].trim(),
  959. content: catArr[2].trim()
  960. };
  961. }
  962. txt.push(catJson);
  963. });
  964. txt = JSON.stringify(txt, null, 2);
  965. var blob = new Blob([txt], {type: "text/plain;charset=utf-8"});
  966. saveAs(blob, document.title.replace(/[\*\/:<>\?\\\|\r\n,]/g, "_") + ".json");
  967. }
  968. }
  969.  
  970. let charset = (document.characterSet || document.charset || document.inputEncoding);
  971. let equiv = document.querySelector('[http-equiv="Content-Type"]'), charsetValid = true;
  972. if (equiv && equiv.content) {
  973. let innerCharSet = equiv.content.match(/charset\=([^;]+)/);
  974. if (!innerCharSet) {
  975. charsetValid = false;
  976. } else if (innerCharSet[1].replace("-", "").toLowerCase() != charset.replace("-", "").toLowerCase()) {
  977. charsetValid = false;
  978. }
  979. } else charsetValid = false;
  980. function indexDownload(aEles, noSort){
  981. if(aEles.length<1)return;
  982. initTxtDownDiv();
  983. if(!noSort) {
  984. if(GM_getValue("contentSort")){
  985. aEles.sort((a, b) => {
  986. return str2Num(a.innerText) - str2Num(b.innerText);
  987. });
  988. }
  989. if(GM_getValue("contentSortUrl")){
  990. aEles.sort((a, b) => {
  991. const nameA = a.href.toUpperCase();
  992. const nameB = b.href.toUpperCase();
  993. if (nameA < nameB) {
  994. return -1;
  995. }
  996. if (nameA > nameB) {
  997. return 1;
  998. }
  999. return 0;
  1000. });
  1001. }
  1002. if(GM_getValue("reverse")){
  1003. aEles=aEles.reverse();
  1004. }
  1005. }
  1006. rCats=[];
  1007. const minute=60000;
  1008. var minTxtLength=GM_getValue("minTxtLength") || 100;
  1009. var customTitle=GM_getValue("customTitle");
  1010. var prefix=GM_getValue("prefix");
  1011. var disableNextPage=!!GM_getValue("disableNextPage");
  1012. var customNextPageReg=GM_getValue("nextPageReg");
  1013. var maxDlPerMin=GM_getValue("maxDlPerMin") || 0;
  1014. var dlCount=0;
  1015. if (customNextPageReg) {
  1016. try {
  1017. innerNextPage = new RegExp(customNextPageReg);
  1018. } catch(e) {
  1019. console.warn(e);
  1020. }
  1021. }
  1022. var linkIndex = 1;
  1023. function packLink(doc, item) {
  1024. if (customTitle) {
  1025. try {
  1026. let title = doc.querySelector(customTitle);
  1027. if (title && title.innerText) {
  1028. item.innerText = title.innerText;
  1029. }
  1030. } catch(e) {
  1031. console.warn(e);
  1032. }
  1033. }
  1034. if (prefix) {
  1035. item.innerText = prefix.replace(/\$i/g, linkIndex) + item.innerText;
  1036. linkIndex++;
  1037. }
  1038. }
  1039. var insertSigns=[];
  1040. // var j=0,rCats=[];
  1041. var downIndex=0,downNum=0,downOnce=function(wait){
  1042. if(downNum>=aEles.length)return;
  1043. if(maxDlPerMin){
  1044. if(dlCount===-1){
  1045. setTimeout(() => {
  1046. downOnce(wait);
  1047. }, minute);
  1048. return;
  1049. }else if(dlCount>=maxDlPerMin){
  1050. dlCount=-1;
  1051. setTimeout(() => {
  1052. dlCount=0;
  1053. downOnce(wait);
  1054. }, minute);
  1055. return;
  1056. }else dlCount++;
  1057. }
  1058. let curIndex=downIndex;
  1059. let aTag=aEles[curIndex];
  1060. let request=(aTag, curIndex)=>{
  1061. if (aTag && aTag.cloneNode) {
  1062. aTag = aTag.cloneNode(true);
  1063. }
  1064. let tryTimes=0;
  1065. let validTimes=0;
  1066. function requestDoc(_charset) {
  1067. if (!_charset) _charset = charset;
  1068. return GM_xmlhttpRequest({
  1069. method: 'GET',
  1070. url: aTag.href,
  1071. headers:{
  1072. referer:aTag.href,
  1073. "Content-Type":"text/html;charset="+_charset
  1074. },
  1075. timeout:10000,
  1076. overrideMimeType:"text/html;charset="+_charset,
  1077. onload: async function(result) {
  1078. let doc = getDocEle(result.responseText);
  1079. if (charsetValid) {
  1080. let equiv = doc.querySelector('[http-equiv="Content-Type"]');
  1081. if (equiv && equiv.content) {
  1082. let innerCharSet = equiv.content.match(/charset\=([^;]+)/);
  1083. if (innerCharSet && innerCharSet[1].replace("-", "").toLowerCase() != _charset.replace("-", "").toLowerCase()) {
  1084. charset = innerCharSet[1];
  1085. return requestDoc(charset);
  1086. }
  1087. }
  1088. }
  1089. downIndex++;
  1090. downNum++;
  1091. if (/^{/.test(result.responseText)) {
  1092. doc.json = () => {
  1093. try {
  1094. return JSON.parse(result.responseText);
  1095. } catch(e) {}
  1096. return {};
  1097. }
  1098. }
  1099. let base = doc.querySelector("base");
  1100. let nextPages = !disableNextPage && !processFunc && await checkNextPage(doc, base ? base.href : aTag.href);
  1101. if (nextPages) {
  1102. if (!nextPages.length) nextPages = [nextPages];
  1103. nextPages.forEach(nextPage => {
  1104. var inArr=false;
  1105. for(var ai=0;ai<aEles.length;ai++){
  1106. if(aEles[ai].href==nextPage.href){
  1107. inArr=true;
  1108. break;
  1109. }
  1110. }
  1111. if(!inArr){
  1112. nextPage.innerText=aTag.innerText+"\t>>";
  1113. aEles.push(nextPage);
  1114. let targetIndex = curIndex;
  1115. for(let a=0;a<insertSigns.length;a++){
  1116. let signs=insertSigns[a],breakSign=false;
  1117. if(signs){
  1118. for(let b=0;b<signs.length;b++){
  1119. let sign=signs[b];
  1120. if(sign==curIndex){
  1121. targetIndex=a;
  1122. breakSign=true;
  1123. break;
  1124. }
  1125. }
  1126. }
  1127. if(breakSign)break;
  1128. }
  1129. let insertSign = insertSigns[targetIndex];
  1130. if(!insertSign)insertSigns[targetIndex] = [];
  1131. insertSigns[targetIndex].push(aEles.length-1);
  1132. }
  1133. });
  1134. }
  1135. if (result.status >= 400) {
  1136. console.warn("error:", `status: ${result.status} from: ${aTag.href}`);
  1137. } else {
  1138. console.log(result.status);
  1139. }
  1140. packLink(doc, aTag);
  1141. let validData = processDoc(curIndex, aTag, doc, (result.status>=400?` status: ${result.status} from: ${aTag.href} `:""), validTimes < 5);
  1142. if (!validData && validTimes++ < 5) {
  1143. downIndex--;
  1144. downNum--;
  1145. setTimeout(() => {
  1146. requestDoc();
  1147. }, Math.random() * 500 + validTimes * 1000);
  1148. return;
  1149. }
  1150. if (wait) {
  1151. setTimeout(() => {
  1152. downOnce(wait);
  1153. }, wait);
  1154. } else downOnce();
  1155. },
  1156. onerror: function(e) {
  1157. console.warn("error:", e, aTag.href);
  1158. if(tryTimes++ < 5){
  1159. setTimeout(() => {
  1160. requestDoc();
  1161. }, Math.random() * 500 + tryTimes * 1000);
  1162. return;
  1163. }
  1164. downIndex++;
  1165. downNum++;
  1166. processDoc(curIndex, aTag, null, ` NETWORK ERROR: ${(e.response||e.responseText)} from: ${aTag.href} `);
  1167. if (wait) {
  1168. setTimeout(() => {
  1169. downOnce(wait);
  1170. }, wait);
  1171. } else downOnce();
  1172. },
  1173. ontimeout: function(e) {
  1174. console.warn("timeout: times="+(tryTimes+1)+" url="+aTag.href);
  1175. //console.log(e);
  1176. if(tryTimes++ < 5){
  1177. setTimeout(() => {
  1178. requestDoc();
  1179. }, Math.random() * 500 + tryTimes * 1000);
  1180. return;
  1181. }
  1182. downIndex++;
  1183. downNum++;
  1184. processDoc(curIndex, aTag, null, ` TIMEOUT: ${aTag.href} `);
  1185. if (wait) {
  1186. setTimeout(() => {
  1187. downOnce(wait);
  1188. }, wait);
  1189. } else downOnce();
  1190. }
  1191. });
  1192. };
  1193. if (useIframe) {
  1194. let iframe = document.createElement('iframe'), inited = false, failedTimes = 0;
  1195. let loadtimeout;
  1196. let loadIframe = src => {
  1197. iframe.src = src;
  1198. clearTimeout(loadtimeout);
  1199. loadtimeout = setTimeout(() => {
  1200. iframe.src = src;
  1201. }, 20000);
  1202. };
  1203. iframe.name = 'pagetual-iframe';
  1204. iframe.width = '100%';
  1205. iframe.height = '1000';
  1206. iframe.frameBorder = '0';
  1207. iframe.sandbox = iframeSandbox || "allow-same-origin allow-scripts allow-popups allow-forms";
  1208. iframe.style.cssText = 'margin:0!important;padding:0!important;visibility:hidden!important;flex:0;opacity:0!important;pointer-events:none!important;position:fixed;top:0px;left:0px;z-index:-2147483647;';
  1209. iframe.addEventListener('load', e => {
  1210. if (e.data != 'pagetual-iframe:DOMLoaded' && e.type != 'load') return;
  1211. if (inited) return;
  1212. inited = true;
  1213. clearTimeout(loadtimeout);
  1214. async function checkIframe() {
  1215. try {
  1216. let doc = iframe.contentDocument || iframe.contentWindow.document;
  1217. if (!doc || !doc.body) {
  1218. setTimeout(() => {
  1219. checkIframe();
  1220. }, 1000);
  1221. return;
  1222. }
  1223. doc.body.scrollTop = 9999999;
  1224. doc.documentElement.scrollTop = 9999999;
  1225. if (!processFunc && validTimes++ > 5 && failedTimes++ < 2) {
  1226. loadIframe(iframe.src);
  1227. validTimes = 0;
  1228. inited = false;
  1229. return;
  1230. }
  1231. let base = doc.querySelector("base");
  1232. let nextPages = !disableNextPage && !processFunc && await checkNextPage(doc, base ? base.href : aTag.href);
  1233. if (nextPages) {
  1234. if (!nextPages.length) nextPages = [nextPages];
  1235. nextPages.forEach(nextPage => {
  1236. var inArr=false;
  1237. for(var ai=0;ai<aEles.length;ai++){
  1238. if(aEles[ai].href==nextPage.href){
  1239. inArr=true;
  1240. break;
  1241. }
  1242. }
  1243. if(!inArr){
  1244. nextPage.innerText=aTag.innerText+"\t>>";
  1245. aEles.push(nextPage);
  1246. let targetIndex = curIndex;
  1247. for(let a=0;a<insertSigns.length;a++){
  1248. let signs=insertSigns[a],breakSign=false;
  1249. if(signs){
  1250. for(let b=0;b<signs.length;b++){
  1251. let sign=signs[b];
  1252. if(sign==curIndex){
  1253. targetIndex=a;
  1254. breakSign=true;
  1255. break;
  1256. }
  1257. }
  1258. }
  1259. if(breakSign)break;
  1260. }
  1261. let insertSign = insertSigns[targetIndex];
  1262. if(!insertSign)insertSigns[targetIndex] = [];
  1263. insertSigns[targetIndex].push(aEles.length-1);
  1264. }
  1265. });
  1266. }
  1267. packLink(doc, aTag);
  1268. downIndex++;
  1269. downNum++;
  1270. let validData = processDoc(curIndex, aTag, doc, "", failedTimes < 2);
  1271. if (!validData) {
  1272. downIndex--;
  1273. downNum--;
  1274. setTimeout(() => {
  1275. checkIframe();
  1276. }, 1000);
  1277. return;
  1278. }
  1279. if (wait) {
  1280. setTimeout(() => {
  1281. downOnce(wait);
  1282. }, wait);
  1283. } else downOnce();
  1284. } catch(e) {
  1285. console.debug("Stop as cors");
  1286. }
  1287. if (iframe && iframe.parentNode) iframe.parentNode.removeChild(iframe);
  1288. }
  1289. setTimeout(() => {
  1290. checkIframe();
  1291. }, 500);
  1292. }, false);
  1293. let checkReady = setInterval(() => {
  1294. let doc;
  1295. try {
  1296. doc = iframe.contentDocument || (iframe.contentWindow && iframe.contentWindow.document);
  1297. } catch(e) {
  1298. clearInterval(checkReady);
  1299. return;
  1300. }
  1301. if (doc) {
  1302. try {
  1303. Function('win', 'iframe', '"use strict";' + (iframeInit || "win.self=win.top;"))(iframe.contentWindow, iframe);
  1304. clearInterval(checkReady);
  1305. } catch(e) {
  1306. console.debug(e);
  1307. }
  1308. }
  1309. }, 50);
  1310. loadIframe(aTag.href);
  1311. document.body.appendChild(iframe);
  1312. return [curIndex, null, aTag.href];
  1313. } else {
  1314. return [curIndex, requestDoc(), aTag.href];
  1315. }
  1316. }
  1317. if(!aTag){
  1318. let waitAtagReadyInterval=setInterval(function(){
  1319. if(downNum>=aEles.length)clearInterval(waitAtagReadyInterval);
  1320. aTag=aEles[curIndex];
  1321. if(aTag){
  1322. clearInterval(waitAtagReadyInterval);
  1323. request(aTag, curIndex);
  1324. }
  1325. },1000);
  1326. return null;
  1327. }
  1328. let result = request(aTag, curIndex);
  1329. if (result) curRequests.push(result);
  1330. return result;
  1331. };
  1332. function getDocEle(str){
  1333. var doc = null;
  1334. try {
  1335. doc = document.implementation.createHTMLDocument('');
  1336. doc.documentElement.innerHTML = str;
  1337. }
  1338. catch (e) {
  1339. console.log('parse error');
  1340. }
  1341. return doc;
  1342. }
  1343. function sortInnerPage(){
  1344. var pageArrs=[],maxIndex=0,i,j;
  1345. for(i=0;i<insertSigns.length;i++){
  1346. var signs=insertSigns[i];
  1347. if(signs){
  1348. for(j=0;j<signs.length;j++){
  1349. var sign=signs[j];
  1350. var cat=rCats[sign];
  1351. rCats[sign]=null;
  1352. if(!pageArrs[i])pageArrs[i]=[];
  1353. pageArrs[i].push(cat);
  1354. }
  1355. }
  1356. }
  1357. for(i=pageArrs.length-1;i>=0;i--){
  1358. let pageArr=pageArrs[i];
  1359. if(pageArr){
  1360. for(j=pageArr.length-1;j>=0;j--){
  1361. rCats.splice(i+1, 0, pageArr[j]);
  1362. }
  1363. }
  1364. }
  1365. rCats = rCats.filter(function(e){return e!=null});
  1366. }
  1367. var waitForComplete;
  1368. function processDoc(i, aTag, doc, cause, check){
  1369. let cbFunc=content=>{
  1370. let isHref = "";
  1371. let saveUrl = GM_getValue("saveUrl");
  1372. if (saveUrl){
  1373. isHref = aTag.href + '\r\n';
  1374. }
  1375. rCats[i]=(aTag.innerText.replace(/[\r\n\t]/g, "") + "\r\n" + isHref + (cause || '') + content.replace(/\s*$/, ""));
  1376. curRequests = curRequests.filter(function(e){return e[0]!=i});
  1377. txtDownContent.style.display="block";
  1378. txtDownWords.innerHTML=getI18n("downloading",[downNum,(aEles.length-downNum),aTag.innerText]);
  1379. if(downNum==aEles.length){
  1380. if(waitForComplete) clearTimeout(waitForComplete);
  1381. waitForComplete=setTimeout(()=>{
  1382. if(downNum==aEles.length){
  1383. txtDownWords.innerHTML=getI18n("complete",[downNum]);
  1384. sortInnerPage();
  1385. var disableAutoStartSave = GM_getValue("disableAutoStartSave");
  1386. if(!disableAutoStartSave){
  1387. saveContent();
  1388. }
  1389. }
  1390. },3000);
  1391. }
  1392. };
  1393. let contentResult=getPageContent(doc, content=>{
  1394. cbFunc(content);
  1395. }, aTag.href);
  1396. if(contentResult!==false){
  1397. if(check && contentResult && contentResult.replace(/\s/g, "").length<minTxtLength){
  1398. return false;
  1399. }
  1400. cbFunc(contentResult);
  1401. }
  1402. return true;
  1403. }
  1404. var downThreadNum = parseInt(GM_getValue("downThreadNum"));
  1405. downThreadNum = downThreadNum || 20;
  1406. if (useIframe && downThreadNum > 5) {
  1407. downThreadNum = 5;
  1408. }
  1409. if (downThreadNum > 0) {
  1410. for (var i = 0; i < downThreadNum; i++) {
  1411. downOnce();
  1412. if (downIndex >= aEles.length - 1 || downIndex >= downThreadNum - 1) break;
  1413. else downIndex++;
  1414. }
  1415. } else {
  1416. downOnce(-downThreadNum * 1000);
  1417. }
  1418. }
  1419.  
  1420. function canonicalUri(src, baseUrl) {
  1421. if (!src) {
  1422. return "";
  1423. }
  1424. if (src.charAt(0) == "#") return baseUrl + src;
  1425. if (src.charAt(0) == "?") return baseUrl.replace(/^([^\?#]+).*/, "$1" + src);
  1426. let origin = location.protocol + '//' + location.host;
  1427. let url = baseUrl || origin;
  1428. url = url.replace(/(\?|#).*/, "");
  1429. if (/https?:\/\/[^\/]+$/.test(url)) url = url + '/';
  1430. if (url.indexOf("http") !== 0) url = origin + url;
  1431. var root_page = /^[^\?#]*\//.exec(url)[0],
  1432. root_domain = /^\w+\:\/\/\/?[^\/]+/.exec(root_page)[0],
  1433. absolute_regex = /^\w+\:\/\//;
  1434. while (src.indexOf("../") === 0) {
  1435. src = src.substr(3);
  1436. root_page = root_page.replace(/\/[^\/]+\/$/, "/");
  1437. }
  1438. src = src.replace(/\.\//, "");
  1439. if (/^\/\/\/?/.test(src)) {
  1440. src = location.protocol + src;
  1441. }
  1442. return (absolute_regex.test(src) ? src : ((src.charAt(0) == "/" ? root_domain : root_page) + src));
  1443. }
  1444.  
  1445. async function checkNextPage(doc, baseUrl) {
  1446. let nextPage = null;
  1447. if (nextPageFunc) {
  1448. nextPage = await nextPageFunc(doc, baseUrl);
  1449. if (nextPage && nextPage.length === 0) nextPage = null;
  1450. } else {
  1451. let aTags = doc.querySelectorAll("a");
  1452. for (var i = 0; i < aTags.length; i++) {
  1453. let aTag = aTags[i];
  1454. if (innerNextPage.test(aTag.innerText) && aTag.href && !/javascript:|#/.test(aTag.href)) {
  1455. let nextPageHref = canonicalUri(aTag.getAttribute("href"), baseUrl || location.href);
  1456. if (nextPageHref != location.href) {
  1457. nextPage = aTag;
  1458. nextPage.href = nextPageHref;
  1459. break;
  1460. }
  1461. }
  1462. }
  1463. }
  1464. return nextPage;
  1465. }
  1466.  
  1467. function textNodesUnder(el){
  1468. var n, a=[], walk=document.createTreeWalker(el,NodeFilter.SHOW_TEXT,null,false);
  1469. while(n=walk.nextNode()) a.push(n);
  1470. return a;
  1471. }
  1472.  
  1473. function getPageContent(doc, cb, url){
  1474. if(!doc)return i18n.error;
  1475. if(doc.body && !doc.body.children.length)return doc.body.innerText;
  1476. if(processFunc){
  1477. return processFunc(doc, cb, url);
  1478. }
  1479. [].forEach.call(doc.querySelectorAll("span,div,ul"),function(item){
  1480. var thisStyle=doc.defaultView?doc.defaultView.getComputedStyle(item):item.style;
  1481. if(thisStyle && (thisStyle.display=="none" || (item.nodeName=="SPAN" && thisStyle.fontSize=="0px"))){
  1482. item.innerHTML="";
  1483. }
  1484. });
  1485. var i,j,k,rStr="",pageData=(doc.body?doc.body:doc).cloneNode(true);
  1486. pageData.innerHTML=pageData.innerHTML.replace(/\<\!\-\-((.|[\n|\r|\r\n])*?)\-\-\>/g,"");
  1487. [].forEach.call(pageData.querySelectorAll("font.jammer"),function(item){
  1488. item.innerHTML="";
  1489. });
  1490. var selectors=GM_getValue("selectors");
  1491. if(selectors){
  1492. [].forEach.call(pageData.querySelectorAll(selectors),function(item){
  1493. item.innerHTML="";
  1494. });
  1495. }
  1496. [].forEach.call(pageData.querySelectorAll("script,style,link,noscript,iframe"),function(item){
  1497. if (item && item.parentNode) {
  1498. item.parentNode.removeChild(item);
  1499. }
  1500. });
  1501. var endEle = ele => {
  1502. return /^(I|STRONG|B|FONT|P|DL|DD|H\d)$/.test(ele.nodeName) && ele.children.length <= 1;
  1503. };
  1504. var largestContent,contents=pageData.querySelectorAll("span,div,article,p,td,pre"),largestNum=0;
  1505. for(i=0;i<contents.length;i++){
  1506. let content=contents[i],hasText=false,allSingle=true,item,curNum=0;
  1507. if(/footer/.test(content.className))continue;
  1508. for(j=content.childNodes.length-1;j>=0;j--){
  1509. item=content.childNodes[j];
  1510. if(item.nodeType==3){
  1511. if(/^\s*$/.test(item.data)){
  1512. item.innerHTML="";
  1513. }else hasText=true;
  1514. }else if(/^(I|A|STRONG|B|FONT|P|DL|DD|H\d)$/.test(item.nodeName)){
  1515. hasText=true;
  1516. }else if(item.nodeType==1&&item.children.length==1&&/^(I|A|STRONG|B|FONT|P|DL|DD|H\d)$/.test(item.children[0].nodeName)){
  1517. hasText=true;
  1518. }
  1519. }
  1520. for(j=content.childNodes.length-1;j>=0;j--){
  1521. item=content.childNodes[j];
  1522. if(item.nodeType==1 && !/^(I|A|STRONG|B|FONT|BR)$/.test(item.nodeName) && /^[\s\-\_\?\>\|]*$/.test(item.innerHTML)){
  1523. item.innerHTML="";
  1524. }
  1525. }
  1526. if(content.childNodes.length>1){
  1527. let indexItem=0;
  1528. for(j=0;j<content.childNodes.length;j++){
  1529. item=content.childNodes[j];
  1530. if(item.nodeType==1){
  1531. if(item.innerText && item.innerText.length<50 && indexReg.test(item.innerText))indexItem++;
  1532. for(k=0;k<item.childNodes.length;k++){
  1533. var childNode=item.childNodes[k];
  1534. if(childNode.nodeType!=3 && !/^(I|A|STRONG|B|FONT|BR)$/.test(childNode.nodeName)){
  1535. allSingle=false;
  1536. break;
  1537. }
  1538. }
  1539. if(!allSingle)break;
  1540. }
  1541. }
  1542. if(indexItem>=5)continue;
  1543. }else{
  1544. allSingle=false;
  1545. }
  1546. if(!allSingle && !hasText){
  1547. continue;
  1548. }else {
  1549. if(pageData==document && content.offsetWidth<=0 && content.offsetHeight<=0){
  1550. continue;
  1551. }
  1552. [].forEach.call(content.childNodes,function(item){
  1553. if(item.nodeType==3)curNum+=item.data.trim().length;
  1554. else if(endEle(item) || (item.nodeType == 1 && item.children.length == 1 && endEle(item.children[0]))) curNum += (firefox ? item.textContent.trim().length : item.innerText.trim().length);
  1555. });
  1556. }
  1557. if(curNum>largestNum){
  1558. largestNum=curNum;
  1559. largestContent=content;
  1560. }
  1561. }
  1562. if(!largestContent)return i18n.error+" : NO TEXT CONTENT";
  1563. var retainImage=!!GM_getValue("retainImage");
  1564. function getContentByLargest() {
  1565. var childlist=pageData.querySelectorAll(largestContent.nodeName);//+(largestContent.className?"."+largestContent.className.replace(/(^\s*)|(\s*$)/g, '').replace(/\s+/g, '.'):""));
  1566. function getRightStr(ele, noTextEnable){
  1567. [].forEach.call(ele.querySelectorAll("a[href]"), a => {
  1568. a.parentNode && a.parentNode.removeChild(a);
  1569. });
  1570. if(retainImage){
  1571. [].forEach.call(ele.querySelectorAll("img[src]"), img => {
  1572. let imgTxtNode=document.createTextNode(`![img](${canonicalUri(img.getAttribute("src"), url || location.href)})`);
  1573. img.parentNode.replaceChild(imgTxtNode, img);
  1574. });
  1575. }
  1576. let childNodes=ele.childNodes,cStr="\r\n",hasText=false;
  1577. for(let j=0;j<childNodes.length;j++){
  1578. let childNode=childNodes[j];
  1579. if(childNode.nodeType==3 && childNode.data && !/^[\s\-\_\?\>\|]*$/.test(childNode.data))hasText=true;
  1580. if(childNode.innerHTML){
  1581. childNode.innerHTML=childNode.innerHTML.replace(/\<\s*br\s*\>/gi,"\r\n").replace(/\n+/gi,"\n").replace(/\r+/gi,"\r");
  1582. }
  1583. let content=childNode.textContent;
  1584. if(content){
  1585. if(!content.trim())continue;
  1586. cStr+=content.replace(/[\uFEFF\xA0 ]+/g," ").replace(/([^\r]|^)\n([^\r]|$)/gi,"$1\r\n$2");
  1587. }
  1588. if(childNode.nodeType!=3 && !/^(I|A|STRONG|B|FONT|IMG)$/.test(childNode.nodeName))cStr+="\r\n";
  1589. else if(childNode.nextSibling && /^P$/.test(childNode.nextSibling.nodeName))cStr+="\r\n";
  1590. }
  1591. if(hasText || noTextEnable || ele==largestContent)rStr+=cStr+"\r\n";
  1592. }
  1593. var sameDepthChildren=[];
  1594. for(i=0;i<childlist.length;i++){
  1595. var child=childlist[i];
  1596. if(getDepth(child)==getDepth(largestContent)){
  1597. if(largestContent.className != child.className)continue;
  1598. sameDepthChildren.push(child);
  1599. }
  1600. }
  1601. var minLength = largestNum>>2;
  1602. var tooShort = sameDepthChildren.length <= 3;
  1603. sameDepthChildren.forEach(child => {
  1604. if(tooShort && child.innerText.length < minLength) return;
  1605. if((largestContent.className && largestContent.className == child.className) || largestContent.parentNode == child.parentNode){
  1606. getRightStr(child, true);
  1607. }else {
  1608. getRightStr(child, false);
  1609. }
  1610. });
  1611. rStr = rStr.replace(/[\n\r]+/g,"\n\r");
  1612. }
  1613. getContentByLargest();
  1614. if (rStr.length < 100) {
  1615. let articles = pageData.querySelectorAll("article");
  1616. if (articles && articles.length == 1) {
  1617. largestContent = articles[0];
  1618. largestNum = largestContent.innerText.length;
  1619. if (largestNum > 100) {
  1620. rStr = "";
  1621. getContentByLargest();
  1622. }
  1623. }
  1624. }
  1625. return rStr;
  1626. }
  1627.  
  1628. function getI18n(key, args){
  1629. var resultStr=i18n[key];
  1630. if(args && args.length>0){
  1631. args.forEach(function(item){
  1632. resultStr=resultStr.replace(/%s/,item);
  1633. });
  1634. }
  1635. return resultStr;
  1636. }
  1637.  
  1638. function getDepth(dom){
  1639. var pa=dom,i=0;
  1640. while(pa.parentNode){
  1641. pa=pa.parentNode;
  1642. i++;
  1643. }
  1644. return i;
  1645. }
  1646.  
  1647. async function sleep(time) {
  1648. await new Promise((resolve) => {
  1649. setTimeout(() => {
  1650. resolve();
  1651. }, time);
  1652. })
  1653. }
  1654.  
  1655. async function fetch(forceSingle){
  1656. forceSingle=forceSingle===true;
  1657. processFunc=null;
  1658. initTxtDownDiv();
  1659. var aEles=document.body.querySelectorAll("a"),list=[];
  1660. txtDownWords.innerHTML=`Analysing ( 1/${aEles.length} )......`;
  1661. txtDownContent.style.pointerEvents="none";
  1662. for(var i=0;i<aEles.length;i++){
  1663. if (i % 100 == 0) {
  1664. await sleep(1);
  1665. }
  1666. txtDownWords.innerHTML=`Analysing ( ${i + 1}/${aEles.length} )......`;
  1667. var aEle=aEles[i],has=false;
  1668. if(aEle.dataset.href && (!aEle.href || aEle.href.indexOf("javascript")!=-1)){
  1669. aEle.href=aEle.dataset.href;
  1670. }
  1671. if(aEle.href==location.href)continue;
  1672. for(var j=0;j<list.length;j++){
  1673. if(list[j].href==aEle.href){
  1674. aEle=list[j];
  1675. list.splice(j,1);
  1676. list.push(aEle);
  1677. has=true;
  1678. break;
  1679. }
  1680. }
  1681. if(!has && aEle.href && /^http/i.test(aEle.href) && ((aEle.innerText.trim()!="" && indexReg.test(aEle.innerText.trim())) || /chapter[\-_]?\d/.test(aEle.href))){
  1682. list.push(aEle);
  1683. }
  1684. }
  1685. txtDownContent.style.display="none";
  1686. txtDownContent.style.pointerEvents="";
  1687. txtDownWords.innerHTML="Analysing......";
  1688. if(list.length>2 && !forceSingle){
  1689. useIframe = false;
  1690. filterList(list);
  1691. }else{
  1692. var blob = new Blob([i18n.info.replace("#t#", location.href)+"\r\n\r\n"+document.title+"\r\n\r\n"+getPageContent(document)], {type: "text/plain;charset=utf-8"});
  1693. saveAs(blob, document.title+".txt");
  1694. }
  1695. }
  1696.  
  1697. function customDown(urls){
  1698. processFunc = null;
  1699. useIframe = false;
  1700. if(urls){
  1701. urls=decodeURIComponent(urls.replace(/%/g,'%25'));
  1702. GM_setValue("DACrules_"+document.domain, urls);
  1703. var processEles=[];
  1704. let urlsArr=urls.split("@@"),eles=[];
  1705. if(/^http|^ftp/.test(urlsArr[0])){
  1706. [].forEach.call(urlsArr[0].split(","),function(i){
  1707. var curEle;
  1708. var varNum=/\[\d+\-\d+\]/.exec(i);
  1709. if(varNum){
  1710. varNum=varNum[0].trim();
  1711. }else{
  1712. curEle=document.createElement("a");
  1713. curEle.href=i;
  1714. curEle.innerText="Added Url";
  1715. processEles.push(curEle);
  1716. return;
  1717. }
  1718. var num1=/\[(\d+)/.exec(varNum)[1].trim();
  1719. var num2=/(\d+)\]/.exec(varNum)[1].trim();
  1720. var num1Int=parseInt(num1);
  1721. var num2Int=parseInt(num2);
  1722. var numLen=num1.length;
  1723. var needAdd=num1.charAt(0)=="0";
  1724. if(num1Int>=num2Int)return;
  1725. for(var j=num1Int;j<=num2Int;j++){
  1726. var urlIndex=j.toString();
  1727. if(needAdd){
  1728. while(urlIndex.length<numLen)urlIndex="0"+urlIndex;
  1729. }
  1730. var curUrl=i.replace(/\[\d+\-\d+\]/,urlIndex).trim();
  1731. curEle=document.createElement("a");
  1732. curEle.href=curUrl;
  1733. curEle.innerText="Added Url " + processEles.length.toString();
  1734. processEles.push(curEle);
  1735. }
  1736. });
  1737. }else{
  1738. let urlSel=urlsArr[0].split(">>");
  1739. try{
  1740. eles=document.querySelectorAll(urlSel[0]);
  1741. eles=[].filter.call(eles, ele=>{
  1742. return ele.nodeName=='BODY'||(!!ele.offsetParent&&getComputedStyle(ele).display!=='none');
  1743. })
  1744. }catch(e){}
  1745. if(eles.length==0){
  1746. eles=[];
  1747. var eleTxts=urlsArr[0].split(/(?<=[^\\])[,,]/),exmpEles=[],excludeTxts={};
  1748. [].forEach.call(document.querySelectorAll("a"),function(item){
  1749. if(!item.offsetParent)return;
  1750. eleTxts.forEach(txt=>{
  1751. var txtArr=txt.split("!");
  1752. if(item.innerText.indexOf(txtArr[0])!=-1){
  1753. exmpEles.push(item);
  1754. excludeTxts[item]=txtArr.splice(1);
  1755. }
  1756. });
  1757. })
  1758. exmpEles.forEach(e=>{
  1759. var cssSelStr="a",pa=e.parentNode,excludeTxt=excludeTxts[e];
  1760. if(e.className)cssSelStr+="."+CSS.escape(e.className.replace(/\s+/g, ".")).replace(/\\\./g, '.');
  1761. while(pa && pa.nodeName!="BODY"){
  1762. cssSelStr=pa.nodeName+">"+cssSelStr;
  1763. pa=pa.parentNode;
  1764. }
  1765. cssSelStr="body>"+cssSelStr;;
  1766. [].forEach.call(document.querySelectorAll(cssSelStr),function(item){
  1767. if(!item.offsetParent)return;
  1768. var isExclude=false;
  1769. for(var t in excludeTxt){
  1770. if(item.innerText.indexOf(excludeTxt[t])!=-1){
  1771. isExclude=true;
  1772. break;
  1773. }
  1774. }
  1775. if(!isExclude && eles.indexOf(item)==-1){
  1776. eles.push(item);
  1777. }
  1778. });
  1779. });
  1780. }
  1781. function addItem(item) {
  1782. let has=false;
  1783. for(var j=0;j<processEles.length;j++){
  1784. if(processEles[j].href==item.href){
  1785. processEles.splice(j,1);
  1786. processEles.push(item);
  1787. has=true;
  1788. break;
  1789. }
  1790. }
  1791. if((!item.href || item.href.indexOf("javascript")!=-1) && item.dataset.href){
  1792. item.href=item.dataset.href;
  1793. }
  1794. if(!has && item.href && /^http/i.test(item.href)){
  1795. processEles.push(item.cloneNode(1));
  1796. }
  1797. }
  1798. [].forEach.call(eles,function(item){
  1799. if(urlSel[1]){
  1800. item=Function("item",urlSel[1])(item);
  1801. let items;
  1802. if (Array.isArray(item)) {
  1803. items = item;
  1804. } else items = [item];
  1805. items.forEach(item => {
  1806. if(!item || !item.href)return;
  1807. if(!item.nodeName || item.nodeName!="A"){
  1808. let href=item.href;
  1809. let innerText=item.innerText;
  1810. item=document.createElement("a");
  1811. item.href=href;
  1812. item.innerText=innerText;
  1813. }
  1814. addItem(item);
  1815. });
  1816. } else {
  1817. addItem(item);
  1818. }
  1819. });
  1820. }
  1821. if(urlsArr[1]){
  1822. processEles.forEach(ele=>{
  1823. ele.href=ele.href.replace(new RegExp(urlsArr[1]), urlsArr[2]);
  1824. });
  1825. }
  1826. var retainImage=!!GM_getValue("retainImage");
  1827. var evalCode = urlsArr[3];
  1828. if (evalCode) {
  1829. evalCode = evalCode.trim();
  1830. if (/^iframe:/.test(evalCode)) {
  1831. evalCode = evalCode.replace("iframe:", "");
  1832. useIframe = true;
  1833. iframeSandbox = false;
  1834. iframeInit = false;
  1835. while (/^(sandbox|init):/.test(evalCode)) {
  1836. iframeSandbox = evalCode.match(/^sandbox:\{(.*?)\}/);
  1837. if (iframeSandbox) {
  1838. evalCode = evalCode.replace(iframeSandbox[0], "");
  1839. iframeSandbox = iframeSandbox[1];
  1840. }
  1841. iframeInit = evalCode.match(/^init:\{(.*?)\}/);
  1842. if (iframeInit) {
  1843. evalCode = evalCode.replace(iframeInit[0], "");
  1844. iframeInit = iframeInit[1];
  1845. }
  1846. }
  1847. }
  1848. let charsetMatch = evalCode.match(/^charset:{(.+?)}/);
  1849. if (charsetMatch) {
  1850. charset = charsetMatch[1];
  1851. evalCode = evalCode.replace(charsetMatch[0], "");
  1852. }
  1853. let nextMatch = evalCode.match(/^next:(\{+)/);
  1854. if (nextMatch) {
  1855. let splitLen = nextMatch[1].length;
  1856. nextMatch = evalCode.match(new RegExp(`^next:\\{{${splitLen}}(.*?)\\}{${splitLen}}`));
  1857. if (nextMatch) {
  1858. let nextCode = nextMatch[1];
  1859. evalCode = evalCode.replace(nextMatch[0], "");
  1860. nextPageFunc = async (doc, url) => {
  1861. let result;
  1862. if (/\breturn\b/.test(nextCode)) {
  1863. result = await new AsyncFunction('doc', 'url', '"use strict";' + nextCode)(doc, url);
  1864. } else {
  1865. try {
  1866. result = doc.querySelectorAll(nextCode);
  1867. if (result && result.length) {
  1868. [].forEach.call(result, ele => {
  1869. ele.href = canonicalUri(ele.getAttribute("href"), url || location.href);
  1870. });
  1871. } else result = null;
  1872. } catch(e) {}
  1873. }
  1874. return result;
  1875. }
  1876. }
  1877. }
  1878. }
  1879. if(evalCode){
  1880. processFunc=(data, cb, url)=>{
  1881. let doc=data;
  1882. if(evalCode.indexOf("return ")==-1){
  1883. if(evalCode.indexOf("@")==0){
  1884. let content="";
  1885. var selectors=GM_getValue("selectors");
  1886. if(selectors){
  1887. [].forEach.call(data.querySelectorAll(selectors),function(item){
  1888. item.innerHTML="";
  1889. });
  1890. }
  1891. [].forEach.call(data.querySelectorAll("script,style,link,noscript,iframe"),function(item){
  1892. if (item && item.parentNode) {
  1893. item.parentNode.removeChild(item);
  1894. }
  1895. });
  1896. if(retainImage){
  1897. [].forEach.call(data.querySelectorAll("img[src]"), img => {
  1898. let imgTxt=`![img](${canonicalUri(img.getAttribute("src"), location.href)})`;
  1899. let imgTxtNode=document.createTextNode(imgTxt);
  1900. img.parentNode.replaceChild(imgTxtNode, img);
  1901. });
  1902. }
  1903. [].forEach.call(data.querySelectorAll(evalCode.slice(1)), ele=>{
  1904. [].forEach.call(ele.childNodes, child=>{
  1905. if(child.innerHTML){
  1906. child.innerHTML=child.innerHTML.replace(/\<\s*br\s*\>/gi,"\r\n").replace(/\n+/gi,"\n").replace(/\r+/gi,"\r");
  1907. }
  1908. if(child.textContent){
  1909. content+=(child.textContent.replace(/ +/g," ").replace(/([^\r]|^)\n([^\r]|$)/gi,"$1\r\n$2")+"\r\n");
  1910. }
  1911. });
  1912. content+="\r\n";
  1913. });
  1914. return content;
  1915. }else return eval(evalCode);
  1916. }else{
  1917. return Function("data", "doc", "cb", "url", evalCode)(data, doc, cb, url);
  1918. }
  1919. };
  1920. }else{
  1921. if(win.dacProcess){
  1922. processFunc=win.dacProcess;
  1923. }
  1924. }
  1925. filterList(processEles);
  1926. }
  1927. }
  1928. const configPage = "https://hoothin.github.io/UserScripts/DownloadAllContent/";
  1929. const copySvg = '<svg aria-hidden="true" height="16" viewBox="0 0 16 16" version="1.1" width="16" data-view-component="true" style="transition: all ease 0.5s;top: 5px;right: 5px;position: absolute;cursor: pointer;"><title>Copy</title><path d="M0 6.75C0 5.784.784 5 1.75 5h1.5a.75.75 0 0 1 0 1.5h-1.5a.25.25 0 0 0-.25.25v7.5c0 .138.112.25.25.25h7.5a.25.25 0 0 0 .25-.25v-1.5a.75.75 0 0 1 1.5 0v1.5A1.75 1.75 0 0 1 9.25 16h-7.5A1.75 1.75 0 0 1 0 14.25Z"></path><path d="M5 1.75C5 .784 5.784 0 6.75 0h7.5C15.216 0 16 .784 16 1.75v7.5A1.75 1.75 0 0 1 14.25 11h-7.5A1.75 1.75 0 0 1 5 9.25Zm1.75-.25a.25.25 0 0 0-.25.25v7.5c0 .138.112.25.25.25h7.5a.25.25 0 0 0 .25-.25v-7.5a.25.25 0 0 0-.25-.25Z"></path></svg>';
  1930. function searchRule(){
  1931. GM_openInTab(configPage + "#@" + location.hostname, {active: true});
  1932. }
  1933. var downloadShortcut = GM_getValue("downloadShortcut") || {ctrlKey: true, shiftKey: false, altKey: false, metaKey: false, key: 'F9'};
  1934. var downloadSingleShortcut = GM_getValue("downloadSingleShortcut") || {ctrlKey: true, shiftKey: true, altKey: false, metaKey: false, key: 'F9'};
  1935. var downloadCustomShortcut = GM_getValue("downloadCustomShortcut") || {ctrlKey: true, shiftKey: false, altKey: true, metaKey: false, key: 'F9'};
  1936.  
  1937. var enableTouch = GM_getValue("enableTouch");
  1938. if (enableTouch) {
  1939. const minLength = 256, tg = 0.5, atg = 2;
  1940. var lastX, lastY, signs, lastSign;
  1941. function tracer(e) {
  1942. let curX = e.changedTouches[0].clientX, curY = e.changedTouches[0].clientY;
  1943. let distanceX = curX - lastX, distanceY = curY - lastY;
  1944. let distance = distanceX * distanceX + distanceY * distanceY;
  1945. if (distance > minLength) {
  1946. lastX = curX;
  1947. lastY = curY;
  1948. let direction = "";
  1949. let slope = Math.abs(distanceY / distanceX);
  1950. if (slope > atg) {
  1951. if (distanceY > 0) {
  1952. direction = "↓";
  1953. } else {
  1954. direction = "↑";
  1955. }
  1956. } else if (slope < tg) {
  1957. if (distanceX > 0) {
  1958. direction = "→";
  1959. } else {
  1960. direction = "←";
  1961. }
  1962. }
  1963. if (direction && lastSign != direction) {
  1964. signs += direction;
  1965. lastSign = direction;
  1966. }
  1967. }
  1968. }
  1969. document.addEventListener("touchstart", function(e) {
  1970. lastX = e.changedTouches[0].clientX;
  1971. lastY = e.changedTouches[0].clientY;
  1972. lastSign = signs = "";
  1973. document.addEventListener("touchmove", tracer, false);
  1974. }, false);
  1975. document.addEventListener("touchend", function(e) {
  1976. document.removeEventListener("touchmove", tracer, false);
  1977. if (signs == "→↓←↑") {
  1978. e.stopPropagation();
  1979. e.preventDefault();
  1980. startCustom();
  1981. }
  1982. }, false);
  1983. }
  1984.  
  1985. if (location.origin + location.pathname == configPage) {
  1986. let exampleNode = document.getElementById("example");
  1987. if (!exampleNode) return;
  1988.  
  1989. exampleNode = exampleNode.parentNode;
  1990. let ruleList = exampleNode.nextElementSibling.nextElementSibling;
  1991. let searchInput = document.createElement("input");
  1992. let inputTimer;
  1993. function searchByInput() {
  1994. clearTimeout(inputTimer);
  1995. inputTimer = setTimeout(() => {
  1996. let curValue = searchInput.value;
  1997. let matchRules = [];
  1998. let dontMatchRules = [];
  1999. if (curValue) {
  2000. for (let i = 0; i < ruleList.children.length; i++) {
  2001. let curRule = ruleList.children[i];
  2002. let aHref = curRule.firstChild.href;
  2003. if (aHref.indexOf(curValue) == -1) {
  2004. dontMatchRules.push(curRule);
  2005. } else {
  2006. matchRules.push(curRule);
  2007. }
  2008. }
  2009. } else {
  2010. dontMatchRules = ruleList.children;
  2011. }
  2012. if (matchRules.length) {
  2013. for (let i = 0; i < dontMatchRules.length; i++) {
  2014. let curRule = dontMatchRules[i];
  2015. curRule.style.display = "none";
  2016. }
  2017. for (let i = 0; i < matchRules.length; i++) {
  2018. let curRule = matchRules[i];
  2019. curRule.style.display = "";
  2020. }
  2021. } else {
  2022. for (let i = 0; i < dontMatchRules.length; i++) {
  2023. let curRule = dontMatchRules[i];
  2024. curRule.style.display = "";
  2025. }
  2026. }
  2027. }, 500);
  2028. }
  2029. searchInput.style.margin = "10px";
  2030. searchInput.style.width = "100%";
  2031. searchInput.placeholder = i18n.searchRule;
  2032. searchInput.addEventListener("input", function(e) {
  2033. searchByInput();
  2034. });
  2035. if (location.hash) {
  2036. let hash = location.hash.slice(1);
  2037. if (hash.indexOf("@") == 0) {
  2038. setTimeout(() => {
  2039. exampleNode.scrollIntoView();
  2040. }, 500);
  2041. searchInput.value = hash.slice(1);
  2042. searchByInput();
  2043. }
  2044. }
  2045. [].forEach.call(ruleList.querySelectorAll("div.highlight"), highlight => {
  2046. highlight.style.position = "relative";
  2047. highlight.innerHTML = highlight.innerHTML + copySvg;
  2048. let svg = highlight.children[1];
  2049. svg.addEventListener("click", function(e) {
  2050. GM_setClipboard(highlight.children[0].innerText);
  2051. svg.style.opacity = 0;
  2052. setTimeout(() => {
  2053. svg.style.opacity = 1;
  2054. }, 1000);
  2055. });
  2056. });
  2057. exampleNode.parentNode.insertBefore(searchInput, ruleList);
  2058.  
  2059.  
  2060. let donateNode = document.querySelector("[alt='donate']");
  2061. if (!donateNode) return;
  2062. let insertPos = donateNode.parentNode.nextElementSibling;
  2063. let radioIndex = 0;
  2064. function createOption(_name, _value, _type) {
  2065. if (!_type) _type = "input";
  2066. let con = document.createElement("div");
  2067. let option = document.createElement("input");
  2068. let cap = document.createElement("b");
  2069. option.type = _type;
  2070. option.value = _value;
  2071. option.checked = _value;
  2072. cap.style.margin = "0px 10px 0px 0px";
  2073. if (_type == "radio") {
  2074. let label = document.createElement("label");
  2075. label.innerText = _name;
  2076. radioIndex++;
  2077. option.id = "radio" + radioIndex;
  2078. label.setAttribute("for", option.id);
  2079. cap.appendChild(label);
  2080. } else {
  2081. if (_type == "input") {
  2082. option.style.flexGrow = "1";
  2083. }
  2084. cap.innerText = _name;
  2085. }
  2086. con.style.margin = "10px 0";
  2087. con.style.display = "flex";
  2088. con.style.alignItems = "center";
  2089. con.appendChild(cap);
  2090. con.appendChild(option);
  2091. insertPos.parentNode.insertBefore(con, insertPos);
  2092. return option;
  2093. }
  2094. function formatShortcut(e) {
  2095. let result = [];
  2096. if (e.ctrlKey) {
  2097. result.push("Ctrl");
  2098. }
  2099. if (e.shiftKey) {
  2100. result.push("Shift");
  2101. }
  2102. if (e.altKey) {
  2103. result.push("Alt");
  2104. }
  2105. if (e.metaKey) {
  2106. result.push("Meta");
  2107. }
  2108. result.push(e.key);
  2109. return result.join(" + ");
  2110. }
  2111. function geneShortcutData(str) {
  2112. if (!str) return "";
  2113. let result = {ctrlKey: false, shiftKey: false, altKey: false, metaKey: false, key: ''};
  2114. str.split(" + ").forEach(item => {
  2115. switch(item) {
  2116. case "Ctrl":
  2117. result.ctrlKey = true;
  2118. break;
  2119. case "Shift":
  2120. result.shiftKey = true;
  2121. break;
  2122. case "Alt":
  2123. result.altKey = true;
  2124. break;
  2125. case "Meta":
  2126. result.metaKey = true;
  2127. break;
  2128. default:
  2129. result.key = item;
  2130. break;
  2131. }
  2132. });
  2133. return result;
  2134. }
  2135. let showFilterList = createOption(i18n.showFilterList, !!GM_getValue("showFilterList"), "checkbox");
  2136. let downloadShortcutInput = createOption(i18n.downloadShortcut, formatShortcut(downloadShortcut) || "");
  2137. let downloadSingleShortcutInput = createOption(i18n.downloadSingleShortcut, formatShortcut(downloadSingleShortcut) || "");
  2138. let downloadCustomShortcutInput = createOption(i18n.downloadCustomShortcut, formatShortcut(downloadCustomShortcut) || "");
  2139. downloadShortcutInput.setAttribute("readonly", "true");
  2140. downloadSingleShortcutInput.setAttribute("readonly", "true");
  2141. downloadCustomShortcutInput.setAttribute("readonly", "true");
  2142. downloadShortcutInput.style.cursor = "cell";
  2143. downloadSingleShortcutInput.style.cursor = "cell";
  2144. downloadCustomShortcutInput.style.cursor = "cell";
  2145. let keydonwHandler = e => {
  2146. if (e.key) {
  2147. if (e.key == "Backspace") {
  2148. e.target.value = "";
  2149. } else if (e.key != "Control" && e.key != "Shift" && e.key != "Alt" && e.key != "Meta") {
  2150. e.target.value = formatShortcut(e);
  2151. }
  2152. }
  2153. e.preventDefault();
  2154. e.stopPropagation();
  2155. };
  2156. downloadShortcutInput.addEventListener("keydown", keydonwHandler);
  2157. downloadSingleShortcutInput.addEventListener("keydown", keydonwHandler);
  2158. downloadCustomShortcutInput.addEventListener("keydown", keydonwHandler);
  2159. let enableTouchInput = createOption(i18n.enableTouch, !!enableTouch, "checkbox");
  2160.  
  2161. let delSelector = createOption(i18n.del, GM_getValue("selectors") || "");
  2162. delSelector.setAttribute("placeHolder", ".mask,.ksam");
  2163. let downThreadNum = createOption(i18n.downThreadNum, GM_getValue("downThreadNum") || "20", "number");
  2164. let maxDlPerMin = createOption(i18n.maxDlPerMin, GM_getValue("maxDlPerMin") || "0", "number");
  2165. let customTitle = createOption(i18n.customTitle, GM_getValue("customTitle") || "");
  2166. let saveUrl = createOption(i18n.saveUrl, !!GM_getValue("saveUrl"), "checkbox");
  2167. let disableAutoStartSave = createOption(i18n.disableAutoStartSave, !!GM_getValue("disableAutoStartSave"), "checkbox");
  2168. customTitle.setAttribute("placeHolder", "title");
  2169. let minTxtLength = createOption(i18n.minTxtLength, GM_getValue("minTxtLength") || "100", "number");
  2170. let contentSortUrlValue = GM_getValue("contentSortUrl") || false;
  2171. let contentSortValue = GM_getValue("contentSort") || false;
  2172. let reSortDefault = createOption(i18n.reSortDefault, !contentSortUrlValue && !contentSortValue, "radio");
  2173. let reSortUrl = createOption(i18n.reSortUrl, contentSortUrlValue || false, "radio");
  2174. let contentSort = createOption(i18n.reSort, contentSortValue || false, "radio");
  2175. reSortDefault.name = "sort";
  2176. reSortUrl.name = "sort";
  2177. contentSort.name = "sort";
  2178. let reverse = createOption(i18n.reverseOrder, !!GM_getValue("reverse"), "checkbox");
  2179. let prefix = createOption(i18n.prefix, GM_getValue("prefix") || "");
  2180. let disableNextPage = !!GM_getValue("disableNextPage");
  2181. let nextPage = createOption(i18n.nextPage, !disableNextPage, "checkbox");
  2182. let nextPageReg = createOption(i18n.nextPageReg, GM_getValue("nextPageReg") || "");
  2183. let retainImage = createOption(i18n.retainImage, !!GM_getValue("retainImage"), "checkbox");
  2184. prefix.setAttribute("placeHolder", "第 $i 章:");
  2185. nextPageReg.setAttribute("placeHolder", "^\\s*(下一[页頁张張]|next\\s*page|次のページ)");
  2186. if (disableNextPage) {
  2187. nextPageReg.parentNode.style.display = "none";
  2188. }
  2189. nextPage.onclick = e => {
  2190. nextPageReg.parentNode.style.display = nextPage.checked ? "flex" : "none";
  2191. }
  2192. let saveBtn = document.createElement("button");
  2193. saveBtn.innerText = i18n.saveBtn;
  2194. saveBtn.style.margin = "0 0 20px 0";
  2195. insertPos.parentNode.insertBefore(saveBtn, insertPos);
  2196. saveBtn.onclick = e => {
  2197. GM_setValue("selectors", delSelector.value || "");
  2198. GM_setValue("downThreadNum", parseInt(downThreadNum.value || 20));
  2199. GM_setValue("maxDlPerMin", parseInt(maxDlPerMin.value || 20));
  2200. GM_setValue("minTxtLength", parseInt(minTxtLength.value || 100));
  2201. GM_setValue("customTitle", customTitle.value || "");
  2202. GM_setValue("disableAutoStartSave", disableAutoStartSave.checked);
  2203. GM_setValue("saveUrl", saveUrl.checked);
  2204. if (reSortUrl.checked) {
  2205. GM_setValue("contentSortUrl", true);
  2206. GM_setValue("contentSort", false);
  2207. } else if (contentSort.checked) {
  2208. GM_setValue("contentSortUrl", false);
  2209. GM_setValue("contentSort", true);
  2210. } else {
  2211. GM_setValue("contentSortUrl", false);
  2212. GM_setValue("contentSort", false);
  2213. }
  2214. GM_setValue("reverse", reverse.checked);
  2215. GM_setValue("enableTouch", enableTouchInput.checked);
  2216. GM_setValue("retainImage", retainImage.checked);
  2217. GM_setValue("showFilterList", showFilterList.checked);
  2218. GM_setValue("disableNextPage", !nextPage.checked);
  2219. GM_setValue("nextPageReg", nextPageReg.value || "");
  2220. GM_setValue("prefix", prefix.value || "");
  2221. GM_setValue("downloadShortcut", geneShortcutData(downloadShortcutInput.value) || "");
  2222. GM_setValue("downloadSingleShortcut", geneShortcutData(downloadSingleShortcutInput.value) || "");
  2223. GM_setValue("downloadCustomShortcut", geneShortcutData(downloadCustomShortcutInput.value) || "");
  2224. alert(i18n.saveOk);
  2225. };
  2226. return;
  2227. }
  2228.  
  2229. function setDel(){
  2230. GM_openInTab(configPage + "#操作說明", {active: true});
  2231. }
  2232.  
  2233. function checkKey(shortcut1, shortcut2) {
  2234. return shortcut1.ctrlKey == shortcut2.ctrlKey && shortcut1.shiftKey == shortcut2.shiftKey && shortcut1.altKey == shortcut2.altKey && shortcut1.metaKey == shortcut2.metaKey && shortcut1.key == shortcut2.key;
  2235. }
  2236.  
  2237. function startCustom() {
  2238. var customRules = GM_getValue("DACrules_" + document.domain);
  2239. var urls = window.prompt(i18n.customInfo + ":\nhttps://xxx.xxx/book-[20-99].html, https://xxx.xxx/book-[01-10].html", customRules || "");
  2240. if (urls) {
  2241. customDown(urls);
  2242. }
  2243. }
  2244.  
  2245. document.addEventListener("keydown", function(e) {
  2246. if (checkKey(downloadCustomShortcut, e)) {
  2247. startCustom();
  2248. } else if (checkKey(downloadSingleShortcut, e)) {
  2249. fetch(true);
  2250. } else if (checkKey(downloadShortcut, e)) {
  2251. fetch(false);
  2252. }
  2253. });
  2254. GM_registerMenuCommand(i18n.custom, () => {
  2255. startCustom();
  2256. });
  2257. GM_registerMenuCommand(i18n.fetch, fetch);
  2258. GM_registerMenuCommand(i18n.setting, setDel);
  2259. GM_registerMenuCommand(i18n.searchRule, searchRule);
  2260. })();