懒人小说下载器

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

当前为 2016-11-23 提交的版本,查看 最新版本

  1. // ==UserScript==
  2. // @name DownloadAllContent
  3. // @name:zh-CN 懒人小说下载器
  4. // @name:zh-TW 懶人小説下載器
  5. // @name:ja 怠惰者小説ダウンロードツール
  6. // @namespace hoothin
  7. // @version 0.3
  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;"></div>
  52. <img src="" id="txtDownQuit" style="position:absolute;right:0px;top:0px;cursor: pointer;" />
  53. </div>
  54. </div>`;
  55. var txtDownContent=document.querySelector("#txtDownContent");
  56. var txtDownWords=document.querySelector("#txtDownWords");
  57. var txtDownQuit=document.querySelector("#txtDownQuit");
  58. txtDownQuit.onclick=function(){
  59. txtDownContent.style.display="none";
  60. txtDownContent.parentNode.removeChild(txtDownContent);
  61. };
  62. var j=0,rCats=[];
  63. for(let i=0;i<aEles.length;i++){
  64. let aTag=aEles[i];
  65. GM_xmlhttpRequest({
  66. method: 'GET',
  67. url: aTag.href,
  68. onload: function(result) {
  69. var doc = null;
  70. try {
  71. doc = document.implementation.createHTMLDocument('');
  72. doc.documentElement.innerHTML = result.responseText;
  73. }
  74. catch (e) {
  75. console.log('parse error');
  76. }
  77. if (!doc) {
  78. return;
  79. }
  80. j++;
  81. rCats[i]=(aTag.textContent+"\r\n"+getPageContent(doc));
  82. txtDownContent.style.display="block";
  83. txtDownWords.innerHTML="已下载完成 "+j+" 段,剩余 "+(aEles.length-j)+" 段";
  84. if(j==aEles.length){
  85. txtDownWords.innerHTML="已全部下载完成,共 "+j+" 段";
  86. 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"});
  87. saveAs(blob, document.title+".txt");
  88. }
  89. }
  90. });
  91. }
  92. }
  93.  
  94. function getPageContent(pageData){
  95. var i,rStr="";
  96. var largestContent,contents=pageData.querySelectorAll("span,div,article,p,td");
  97. for(i=0;i<contents.length;i++){
  98. var content=contents[i];
  99. if(content.firstChild && (
  100. (content.firstChild.nodeType!=3 && !/^[I|A]$/.test(content.firstChild.tagName)) ||
  101. (/^\s*$/.test(content.firstChild.data) &&
  102. (!content.childNodes[1]||!/^[I|A]$/.test(content.childNodes[1].tagName)))
  103. ))
  104. continue;
  105. if(pageData==document && content.offsetWidth <= 0 && content.offsetHeight <= 0)
  106. continue;
  107. if(navigator.userAgent.toLowerCase().indexOf('firefox')!=-1){
  108. if(!largestContent || largestContent.textContent.length<content.textContent.length){
  109. largestContent=content;
  110. }
  111. }else{
  112. if(!largestContent || largestContent.innerText.length<content.innerText.length){
  113. largestContent=content;
  114. }
  115. }
  116. }
  117. var childlist=pageData.querySelectorAll(largestContent.tagName);
  118. for(i=0;i<childlist.length;i++){
  119. var child=childlist[i];
  120. if(largestContent.className && largestContent.className==child.className){
  121. }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;
  122. if(getDepth(child)==getDepth(largestContent)){
  123. let childNodes=child.childNodes,cStr="\r\n",hasText=false;
  124. for(var j=0;j<childNodes.length;j++){
  125. var childNode=childNodes[j];
  126. if(childNode.nodeType==3 && childNode.data && !/^\s*$/.test(childNode.data))hasText=true;
  127. if(childNode.tagName=="BR")cStr+="\r\n";
  128. else if(!/SCRIPT|STYLE/.test(childNode.tagName) && childNode.textContent)cStr+=childNode.textContent.replace(/\s*/," ");
  129. }
  130. if(hasText)rStr+=cStr;
  131. }
  132. }
  133. return rStr;
  134. }
  135.  
  136. function getDepth(dom){
  137. var pa=dom,i=0;
  138. while(pa.parentNode){
  139. pa=pa.parentNode;
  140. i++;
  141. }
  142. return i;
  143. }
  144.  
  145. function fetch(){
  146. var aEles=document.querySelectorAll("a"),list=[];
  147. for(var i=0;i<aEles.length;i++){
  148. var aEle=aEles[i];
  149. if(/第[\d|〇|零|一|二|三|四|五|六|七|八|九|十|百|千|万|萬]+[章|节|回|卷|折|篇|幕|集]|序|序\s*言|序\s*章|前\s*言|引\s*言|引\s*子|摘\s*要|楔\s*子|后\s*记|附\s*言|结\s*语/.test(aEle.innerHTML)){
  150. list.push(aEle);
  151. }
  152. }
  153. if(list.length>2){
  154. indexDownload(list);
  155. }else{
  156. var blob = new Blob([i18n.info+"\r\n"+document.title+"\r\n\r\n"+getPageContent(document)], {type: "text/plain;charset=utf-8"});
  157. saveAs(blob, document.title+".txt");
  158. }
  159. }
  160.  
  161. document.addEventListener("keydown", function(e) {
  162. if(e.keyCode == 120 && e.ctrlKey) {
  163. fetch();
  164. }
  165. });
  166. GM_registerMenuCommand(i18n.fetch, fetch);
  167. })();