懒人小说下载器

通用网站内容抓取工具,可批量抓取小说、论坛内容等并保存为TXT文档

目前为 2016-11-24 提交的版本,查看 最新版本

  1. // ==UserScript==
  2. // @name DownloadAllContent
  3. // @name:zh-CN 懒人小说下载器
  4. // @name:zh-TW 懶人小説下載器
  5. // @name:ja 怠惰者小説ダウンロードツール
  6. // @namespace hoothin
  7. // @version 0.5
  8. // @description Fetch and download main content on current page
  9. // @description:zh-CN 通用网站内容抓取工具,可批量抓取小说、论坛内容等并保存为TXT文档
  10. // @description:zh-TW 通用網站內容抓取工具,可批量抓取小說、論壇內容等並保存為TXT文檔
  11. // @description:ja ユニバーサルサイトコンテンツクロールツール、クロール、フォーラム内容など
  12. // @author hoothin
  13. // @include *
  14. // @grant GM_xmlhttpRequest
  15. // @grant GM_registerMenuCommand
  16. // @require https://cdnjs.cloudflare.com/ajax/libs/FileSaver.js/1.3.3/FileSaver.min.js
  17. // @license MIT License
  18. // @compatible chrome
  19. // @compatible firefox
  20. // @compatible opera 未测试
  21. // @compatible safari 未测试
  22. // @contributionURL https://www.paypal.com/cgi-bin/webscr?cmd=_donations&business=rixixi@sina.com&item_name=Greasy+Fork+donation
  23. // @contributionAmount 1
  24. // ==/UserScript==
  25.  
  26. (function() {
  27. 'use strict';
  28. var lang = navigator.appName=="Netscape"?navigator.language:navigator.userLanguage;
  29. var i18n={};
  30. switch (lang){
  31. case "zh-CN":
  32. i18n={
  33. fetch:"开始下载小说或其他",
  34. info:"本文使用DownloadAllContent脚本下载"
  35. };
  36. break;
  37. default:
  38. i18n={
  39. fetch:"Download All Content",
  40. info:"The TXT is downloaded with 'DownloadAllContent'"
  41. };
  42. break;
  43. }
  44.  
  45. function indexDownload(aEles){
  46. var rocketContent=document.createElement("div");
  47. document.body.appendChild(rocketContent);
  48. rocketContent.outerHTML=`
  49. <div id="txtDownContent" style="display: none;">
  50. <div style="width:300px;height:50px;position:fixed;left:50%;top:50%;margin-top:-25px;margin-left:-150px;z-index:100000;background-color:#ffffff;border:1px solid #afb3b6;border-radius:10px;opacity:0.95;filter:alpha(opacity=95);box-shadow:5px 5px 20px 0px #000;">
  51. <div id="txtDownWords" style="position:absolute;left:20px;top:10px;width:260px;">
  52. </div>
  53. <img src="" id="txtDownQuit" style="position:absolute;right:0px;top:0px;cursor: pointer;" />
  54. </div>
  55. </div>`;
  56. var txtDownContent=document.querySelector("#txtDownContent");
  57. var txtDownWords=document.querySelector("#txtDownWords");
  58. var txtDownQuit=document.querySelector("#txtDownQuit");
  59. txtDownQuit.onclick=function(){
  60. txtDownContent.style.display="none";
  61. txtDownContent.parentNode.removeChild(txtDownContent);
  62. };
  63. var j=0,rCats=[];
  64. function getDocEle(str){
  65. var doc = null;
  66. try {
  67. doc = document.implementation.createHTMLDocument('');
  68. doc.documentElement.innerHTML = str;
  69. }
  70. catch (e) {
  71. console.log('parse error');
  72. }
  73. return doc;
  74. }
  75. function processDoc(i, aTag, doc){
  76. j++;
  77. rCats[i]=(aTag.textContent+"\r\n"+getPageContent(doc));
  78. txtDownContent.style.display="block";
  79. txtDownWords.innerHTML="已下载完成 "+j+" 段,剩余 "+(aEles.length-j)+" 段";
  80. if(j==aEles.length){
  81. txtDownWords.innerHTML="已全部下载完成,共 "+j+" 段";
  82. var blob = new Blob([i18n.info+"\r\n"+document.title+"\r\n\r\n"+rCats.join("\r\n\r\n")], {type: "text/plain;charset=utf-8"});
  83. saveAs(blob, document.title+".txt");
  84. }
  85. }
  86. for(let i=0;i<aEles.length;i++){
  87. let aTag=aEles[i];
  88. GM_xmlhttpRequest({
  89. method: 'GET',
  90. url: aTag.href,
  91. onload: function(result) {
  92. var doc = getDocEle(result.responseText);
  93. if (!doc) {
  94. return;
  95. }
  96. var contentType=doc.querySelector("meta[http-equiv=Content-Type]");
  97. if(contentType && !/utf\-?8/i.test(contentType.content)){
  98. GM_xmlhttpRequest({
  99. method: 'GET',
  100. url: aTag.href,
  101. overrideMimeType:contentType.content,
  102. onload: function(result) {
  103. var doc = getDocEle(result.responseText);
  104. if (!doc) {
  105. return;
  106. }
  107. processDoc(i, aTag, doc);
  108. }
  109. });
  110. }else{
  111. processDoc(i, aTag, doc);
  112. }
  113. }
  114. });
  115. }
  116. }
  117.  
  118. function getPageContent(pageData){
  119. var i,rStr="";
  120. var largestContent,contents=pageData.querySelectorAll("span,div,article,p,td");
  121. for(i=0;i<contents.length;i++){
  122. var content=contents[i];
  123. if(content.firstChild && (
  124. (content.firstChild.nodeType!=3 && !/^[I|A]$/.test(content.firstChild.tagName)) ||
  125. (/^\s*$/.test(content.firstChild.data) &&
  126. (!content.childNodes[1]||!/^[I|A]$/.test(content.childNodes[1].tagName)))
  127. ))
  128. continue;
  129. if(pageData==document && content.offsetWidth <= 0 && content.offsetHeight <= 0)
  130. continue;
  131. if(navigator.userAgent.toLowerCase().indexOf('firefox')!=-1){
  132. if(!largestContent || largestContent.textContent.length<content.textContent.length){
  133. largestContent=content;
  134. }
  135. }else{
  136. if(!largestContent || largestContent.innerText.length<content.innerText.length){
  137. largestContent=content;
  138. }
  139. }
  140. }
  141. var childlist=pageData.querySelectorAll(largestContent.tagName);
  142. for(i=0;i<childlist.length;i++){
  143. var child=childlist[i];
  144. if(largestContent.className && largestContent.className==child.className){
  145. }else if(child.firstChild && ((child.firstChild.nodeType!=3 && !/^[I|A]$/.test(child.firstChild.tagName)) || (/^\s*$/.test(child.firstChild.data) && (!child.childNodes[1] || !/^[I|A]$/.test(child.childNodes[1].tagName)))))continue;
  146. if(getDepth(child)==getDepth(largestContent)){
  147. let childNodes=child.childNodes,cStr="\r\n",hasText=false;
  148. for(var j=0;j<childNodes.length;j++){
  149. var childNode=childNodes[j];
  150. if(childNode.nodeType==3 && childNode.data && !/^\s*$/.test(childNode.data))hasText=true;
  151. if(childNode.tagName=="BR")cStr+="\r\n";
  152. else if(!/SCRIPT|STYLE/.test(childNode.tagName) && childNode.textContent)cStr+=childNode.textContent.replace(/\s*/," ");
  153. }
  154. if(hasText)rStr+=cStr;
  155. }
  156. }
  157. return rStr;
  158. }
  159.  
  160. function getDepth(dom){
  161. var pa=dom,i=0;
  162. while(pa.parentNode){
  163. pa=pa.parentNode;
  164. i++;
  165. }
  166. return i;
  167. }
  168.  
  169. function fetch(){
  170. var aEles=document.querySelectorAll("a"),list=[];
  171. for(var i=0;i<aEles.length;i++){
  172. var aEle=aEles[i];
  173. if(/第[\d|〇|零|一|二|三|四|五|六|七|八|九|十|百|千|万|萬]+[章|节|回|卷|折|篇|幕|集]|序|序\s*言|序\s*章|前\s*言|引\s*言|引\s*子|摘\s*要|楔\s*子|后\s*记|附\s*言|结\s*语/.test(aEle.innerHTML)){
  174. list.push(aEle);
  175. }
  176. }
  177. if(list.length>2){
  178. indexDownload(list);
  179. }else{
  180. var blob = new Blob([i18n.info+"\r\n"+document.title+"\r\n\r\n"+getPageContent(document)], {type: "text/plain;charset=utf-8"});
  181. saveAs(blob, document.title+".txt");
  182. }
  183. }
  184.  
  185. document.addEventListener("keydown", function(e) {
  186. if(e.keyCode == 120 && e.ctrlKey) {
  187. fetch();
  188. }
  189. });
  190. GM_registerMenuCommand(i18n.fetch, fetch);
  191. })();