懒人小说下载器

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

当前为 2021-12-29 提交的版本,查看 最新版本

  1. // ==UserScript==
  2. // @name DownloadAllContent
  3. // @name:zh-CN 懒人小说下载器
  4. // @name:zh-TW 懶人小説下載器
  5. // @name:ja 怠惰者小説ダウンロードツール
  6. // @namespace hoothin
  7. // @version 2.6.1
  8. // @description Fetch and download main content on current page, provide special support for chinese novel
  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. // @grant GM_setValue
  17. // @grant GM_getValue
  18. // @require https://cdnjs.cloudflare.com/ajax/libs/FileSaver.js/1.3.8/FileSaver.min.js
  19. // @license MIT License
  20. // @compatible chrome
  21. // @compatible firefox
  22. // @compatible opera 未测试
  23. // @compatible safari 未测试
  24. // @contributionURL https://www.paypal.com/cgi-bin/webscr?cmd=_donations&business=rixixi@sina.com&item_name=Greasy+Fork+donation
  25. // @contributionAmount 1
  26. // ==/UserScript==
  27.  
  28. (function() {
  29. 'use strict';
  30. var indexReg=/PART\b|^Prologue|分卷|Chapter\s*[\-_]?\d+|^序$|^序\s*言|^序\s*章|^前\s*言|^引\s*言|^引\s*子|^摘\s*要|^楔\s*子|^契\s*子|^后\s*记|^後\s*記|^附\s*言|^结\s*语|^結\s*語|^尾\s*声|^最終話|^最终话|^番\s*外|^\d+\s*\D*[^\d#\.]$|^[第(][\d〇零一二三四五六七八九十百千万萬-]+\s*(、|)|章|节|節|回|卷|折|篇|幕|集|话|話)/i;
  31. var innerNextPage=/下一(页|张)|next\s*page/i;
  32. var lang = navigator.appName=="Netscape"?navigator.language:navigator.userLanguage;
  33. var i18n={};
  34. var rCats=[];
  35. var processFunc;
  36. switch (lang){
  37. case "zh-CN":
  38. case "zh-SG":
  39. i18n={
  40. fetch:"开始下载小说或其他【Ctrl+F9】",
  41. info:"本文是使用懒人小说下载器(DownloadAllContent)下载的",
  42. error:"该段内容获取失败",
  43. downloading:"已下载完成 %s 段,剩余 %s 段<br>正在下载 %s",
  44. complete:"已全部下载完成,共 %s 段",
  45. del:"设置文本干扰码的CSS选择器",
  46. custom:"自定义下载",
  47. customInfo:"输入网址或者章节CSS选择器",
  48. reSort:"按标题名重新排序",
  49. setting:"懒人小说下载设置",
  50. abort:"跳过此章",
  51. save:"临时保存",
  52. downThreadNum:"设置同时下载的线程数"
  53. };
  54. break;
  55. case "zh-TW":
  56. case "zh-HK":
  57. i18n={
  58. fetch:"開始下載小說或其他【Ctrl+F9】",
  59. info:"本文是使用懶人小說下載器(DownloadAllContent)下載的",
  60. error:"該段內容獲取失敗",
  61. downloading:"已下載完成 %s 段,剩餘 %s 段<br>正在下載 %s",
  62. complete:"已全部下載完成,共 %s 段",
  63. del:"設置文本干擾碼的CSS選擇器",
  64. custom:"自定義下載",
  65. customInfo:"輸入網址或者章節CSS選擇器",
  66. reSort:"按標題名重新排序",
  67. setting:"懶人小說下載設置",
  68. abort:"跳過此章",
  69. save:"保存當前",
  70. downThreadNum:"設置同時下載的綫程數"
  71. };
  72. break;
  73. default:
  74. i18n={
  75. fetch:"Download All Content[Ctrl+F9]",
  76. info:"The TXT is downloaded by 'DownloadAllContent'",
  77. error:"Failed in downloading current chapter",
  78. downloading:"%s pages are downloaded, there are still %s pages left<br>Downloading %s ......",
  79. complete:"Completed! Get %s pages in total",
  80. del:"Set css selectors for ignore",
  81. custom:"Custom to download",
  82. customInfo:"Input urls OR sss selectors for chapter links",
  83. reSort:"ReSort by title",
  84. setting:"DownloadAllContent Setting",
  85. abort:"Abort",
  86. save:"Save",
  87. downThreadNum:"Set threadNum for download"
  88. };
  89. break;
  90. }
  91. var firefox=navigator.userAgent.toLowerCase().indexOf('firefox')!=-1,curRequests=[];
  92. var rocketContent,txtDownContent,txtDownWords,txtDownQuit,txtDownDivInited=false;
  93.  
  94. function initTxtDownDiv(){
  95. if(txtDownDivInited)return;
  96. txtDownDivInited=true;
  97. rocketContent=document.createElement("div");
  98. document.body.appendChild(rocketContent);
  99. rocketContent.outerHTML=`
  100. <div id="txtDownContent">
  101. <div style="font-size:16px;color:#333333;width:360px;height:90px;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;">
  102. <div id="txtDownWords" style="position:absolute;width:275px;max-height: 90%;border: 1px solid #f3f1f1;padding: 8px;border-radius: 10px;overflow: auto;">
  103. Downloading......
  104. </div>
  105. <div id="txtDownQuit" style="width:36px;height:28px;border-radius:10px;position:absolute;right:2px;top:2px;cursor: pointer;background-color:#ff5a5a;">
  106. <span style="height:28px;line-height:28px;display:block;color:#FFF;text-align:center;font-size:20px;">╳</span>
  107. </div>
  108. <div style="position:absolute;right:0px;bottom:2px;cursor: pointer;max-width:85px">
  109. <button id="abortRequest" style="background: #008aff;border: 0;padding: 5px;border-radius: 10px;color: white;float: right;margin: 1px;height: 25px;display:none;">${getI18n('abort')}</button>
  110. <button id="tempSaveTxt" style="background: #008aff;border: 0;padding: 5px;border-radius: 10px;color: white;float: right;margin: 1px;height: 25px;">${getI18n('save')}</button>
  111. </div>
  112. </div>
  113. </div>`;
  114. txtDownContent=document.querySelector("#txtDownContent");
  115. txtDownWords=document.querySelector("#txtDownWords");
  116. txtDownQuit=document.querySelector("#txtDownQuit");
  117. txtDownQuit.onclick=function(){
  118. txtDownContent.style.display="none";
  119. txtDownContent.parentNode.removeChild(txtDownContent);
  120. };
  121. initTempSave();
  122. }
  123.  
  124. function initTempSave(){
  125. var tempSavebtn = document.getElementById('tempSaveTxt');
  126. var abortbtn = document.getElementById('abortRequest');
  127. tempSavebtn.onclick = function(){
  128. 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"});
  129. saveAs(blob, document.title+".txt");
  130. }
  131. abortbtn.onclick = function(){
  132. let curRequest = curRequests.pop();
  133. if(curRequest)curRequest[1].abort();
  134. }
  135. }
  136.  
  137. function indexDownload(aEles){
  138. if(aEles.length<1)return;
  139. initTxtDownDiv();
  140. if(GM_getValue("contentSort")){
  141. aEles.sort(function(a,b){
  142. return parseInt(a.innerText.replace(/[^0-9]/ig,"")) - parseInt(b.innerText.replace(/[^0-9]/ig,""));
  143. });
  144. }
  145. rCats=[];
  146. var insertSigns=[];
  147. // var j=0,rCats=[];
  148. var downIndex=0,downNum=0,downOnce=function(){
  149. if(downNum>=aEles.length)return;
  150. let curIndex=downIndex;
  151. let aTag=aEles[curIndex];
  152. let request=(aTag, curIndex)=>{
  153. return [curIndex,GM_xmlhttpRequest({
  154. method: 'GET',
  155. url: aTag.href,
  156. headers:{referer:aTag.href},
  157. timeout:15000,
  158. overrideMimeType:"text/html;charset="+document.charset,
  159. onload: function(result) {
  160. var doc = getDocEle(result.responseText);
  161. let nextPage=checkNextPage(doc);
  162. if(nextPage){
  163. var inArr=false;
  164. for(var ai=0;ai<aEles.length;ai++){
  165. if(aEles[ai].href==nextPage.href){
  166. inArr=true;
  167. break;
  168. }
  169. }
  170. if(!inArr){
  171. nextPage.innerText=aTag.innerText+"\t>>";
  172. aEles.push(nextPage);
  173. let targetIndex = curIndex;
  174. for(let a=0;a<insertSigns.length;a++){
  175. let signs=insertSigns[a],breakSign=false;
  176. if(signs){
  177. for(let b=0;b<signs.length;b++){
  178. let sign=signs[b];
  179. if(sign==curIndex){
  180. targetIndex=a;
  181. breakSign=true;
  182. break;
  183. }
  184. }
  185. }
  186. if(breakSign)break;
  187. }
  188. let insertSign = insertSigns[targetIndex];
  189. if(!insertSign)insertSigns[targetIndex] = [];
  190. insertSigns[targetIndex].push(aEles.length-1);
  191. }
  192. }
  193. downIndex++;
  194. downNum++;
  195. processDoc(curIndex, aTag, doc);
  196. let request=downOnce();
  197. if(request)curRequests.push(request);
  198. },
  199. onerror: function(e) {
  200. console.warn("error:");
  201. console.log(e);
  202. downIndex++;
  203. downNum++;
  204. processDoc(curIndex, aTag, null);
  205. let request=downOnce();
  206. if(request)curRequests.push(request);
  207. },
  208. ontimeout: function(e) {
  209. console.warn("timeout:");
  210. console.log(e);
  211. downIndex++;
  212. downNum++;
  213. processDoc(curIndex, aTag, null);
  214. let request=downOnce();
  215. if(request)curRequests.push(request);
  216. },
  217. })];
  218. }
  219. if(!aTag){
  220. let waitAtagReadyInterval=setInterval(function(){
  221. if(downNum>=aEles.length)clearInterval(waitAtagReadyInterval);
  222. aTag=aEles[curIndex];
  223. if(aTag){
  224. clearInterval(waitAtagReadyInterval);
  225. request(aTag, curIndex);
  226. }
  227. },1000);
  228. return null;
  229. }
  230. return request(aTag, curIndex);
  231. };
  232. function getDocEle(str){
  233. var doc = null;
  234. try {
  235. doc = document.implementation.createHTMLDocument('');
  236. doc.documentElement.innerHTML = str;
  237. }
  238. catch (e) {
  239. console.log('parse error');
  240. }
  241. return doc;
  242. }
  243. function sortInnerPage(){
  244. var pageArrs=[],maxIndex=0,i,j;
  245. for(i=0;i<insertSigns.length;i++){
  246. var signs=insertSigns[i];
  247. if(signs){
  248. for(j=0;j<signs.length;j++){
  249. var sign=signs[j];
  250. var cat=rCats[sign];
  251. rCats[sign]=null;
  252. if(!pageArrs[i])pageArrs[i]=[];
  253. pageArrs[i].push(cat);
  254. }
  255. }
  256. }
  257. for(i=pageArrs.length-1;i>=0;i--){
  258. let pageArr=pageArrs[i];
  259. if(pageArr){
  260. for(j=pageArr.length-1;j>=0;j--){
  261. rCats.splice(i+1, 0, pageArr[j]);
  262. }
  263. }
  264. }
  265. rCats = rCats.filter(function(e){return e!=null});
  266. }
  267. function processDoc(i, aTag, doc){
  268. curRequests = curRequests.filter(function(e){return e[0]!=i});
  269. rCats[i]=(aTag.innerText+"\r\n"+getPageContent(doc));
  270. txtDownContent.style.display="block";
  271. txtDownWords.innerHTML=getI18n("downloading",[downNum,(aEles.length-downNum),aTag.innerText]);
  272. if(downNum==aEles.length){
  273. txtDownWords.innerHTML=getI18n("complete",[downNum]);
  274. sortInnerPage();
  275. 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"});
  276. saveAs(blob, document.title+".txt");
  277. }
  278. }
  279. var downThreadNum = parseInt(GM_getValue("downThreadNum"));
  280. downThreadNum=downThreadNum>0?downThreadNum:20;
  281. for(var i=0;i<downThreadNum;i++){
  282. let request=downOnce();
  283. if(request)curRequests.push(request);
  284. if(downIndex>=aEles.length-1 || downIndex>=downThreadNum-1)break;
  285. else downIndex++;
  286. }
  287.  
  288. /*for(let i=0;i<aEles.length;i++){
  289. let aTag=aEles[i];
  290. GM_xmlhttpRequest({
  291. method: 'GET',
  292. url: aTag.href,
  293. overrideMimeType:"text/html;charset="+document.charset,
  294. onload: function(result) {
  295. var doc = getDocEle(result.responseText);
  296. processDoc(i, aTag, doc);
  297. }
  298. });
  299. }*/
  300. }
  301.  
  302. function checkNextPage(doc){
  303. let aTags=doc.querySelectorAll("a"),nextPage=null;
  304. for(var i=0;i<aTags.length;i++){
  305. let aTag=aTags[i];
  306. if(innerNextPage.test(aTag.innerText) && /^http/i.test(aTag.href)){
  307. nextPage=aTag;
  308. break;
  309. }
  310. }
  311. return nextPage;
  312. }
  313.  
  314. function getPageContent(doc){
  315. if(!doc)return i18n.error;
  316. if(processFunc){
  317. return processFunc(doc);
  318. }
  319. if(doc.defaultView){
  320. [].forEach.call(doc.querySelectorAll("span,div,ul"),function(item){
  321. var thisStyle=doc.defaultView.getComputedStyle(item);
  322. if(thisStyle && (thisStyle.display=="none" || (item.tagName=="SPAN" && thisStyle.fontSize=="0px")))
  323. item.innerHTML="";
  324. });
  325. }
  326. var i,j,k,rStr="",pageData=(doc.body?doc.body:doc).cloneNode(true),delList=[];
  327. pageData.innerHTML=pageData.innerHTML.replace(/\<\!\-\-((.|[\n|\r|\r\n])*?)\-\-\>/g,"");
  328. [].forEach.call(pageData.querySelectorAll("font.jammer"),function(item){
  329. item.innerHTML="";
  330. });
  331. var selectors=GM_getValue("selectors");
  332. if(selectors){
  333. [].forEach.call(pageData.querySelectorAll(selectors),function(item){
  334. item.innerHTML="";
  335. });
  336. }
  337. [].forEach.call(pageData.querySelectorAll("script,style,link,img,noscript,iframe"),function(item){delList.push(item);});
  338. [].forEach.call(delList,function(item){item.innerHTML="";});
  339. var largestContent,contents=pageData.querySelectorAll("span,div,article,p,td"),largestNum=0;
  340. for(i=0;i<contents.length;i++){
  341. let content=contents[i],hasText=false,allSingle=true,item,curNum=0;
  342. for(j=content.childNodes.length-1;j>=0;j--){
  343. item=content.childNodes[j];
  344. if(item.nodeType==3){
  345. if(/^\s*$/.test(item.data))
  346. item.innerHTML="";
  347. else hasText=true;
  348. }else if(/^(I|A|STRONG|B|FONT|P|DL|DD|H\d)$/.test(item.tagName))hasText=true;
  349. }
  350. for(j=content.childNodes.length-1;j>=0;j--){
  351. item=content.childNodes[j];
  352. if(item.nodeType==1 && !/^(I|A|STRONG|B|FONT|BR)$/.test(item.tagName) && /^[\s\-\_\?\>\|]*$/.test(item.innerHTML))
  353. item.innerHTML="";
  354. }
  355. if(content.childNodes.length>1){
  356. for(j=0;j<content.childNodes.length;j++){
  357. item=content.childNodes[j];
  358. if(item.nodeType==1){
  359. for(k=0;k<item.childNodes.length;k++){
  360. var childNode=item.childNodes[k];
  361. if(childNode.nodeType!=3 && !/^(I|A|STRONG|B|FONT|BR)$/.test(childNode.tagName)){
  362. allSingle=false;
  363. break;
  364. }
  365. }
  366. if(!allSingle)break;
  367. }
  368. }
  369. }else{
  370. allSingle=false;
  371. }
  372. if(allSingle){
  373. curNum=(firefox?content.textContent.length:content.innerText.length);
  374. }else {
  375. if(!hasText)continue;
  376. if(pageData==document && content.offsetWidth<=0 && content.offsetHeight<=0)
  377. continue;
  378. [].forEach.call(content.childNodes,function(item){
  379. if(item.nodeType==3)curNum+=item.data.length;
  380. else if(/^(I|A|STRONG|B|FONT|P|DL|DD|H\d)$/.test(item.tagName))curNum+=(firefox?item.textContent.length:item.innerText.length);
  381. });
  382. }
  383. if(curNum>largestNum){
  384. largestNum=curNum;
  385. largestContent=content;
  386. }
  387. }
  388. if(!largestContent)return i18n.error;
  389. var childlist=pageData.querySelectorAll(largestContent.tagName);//+(largestContent.className?"."+largestContent.className.replace(/(^\s*)|(\s*$)/g, '').replace(/\s+/g, '.'):""));
  390. function getRightStr(ele, noTextEnable){
  391. let childNodes=ele.childNodes,cStr="\r\n",hasText=false;
  392. for(let j=0;j<childNodes.length;j++){
  393. let childNode=childNodes[j];
  394. if(childNode.nodeType==3 && childNode.data && !/^[\s\-\_\?\>\|]*$/.test(childNode.data))hasText=true;
  395. if(childNode.innerHTML){
  396. childNode.innerHTML=childNode.innerHTML.replace(/\<\s*br\s*\>/gi,"\r\n").replace(/\n+/gi,"\n").replace(/\r+/gi,"\r");
  397. }
  398. if(childNode.textContent){
  399. cStr+=childNode.textContent.replace(/ +/g," ").replace(/([^\r]|^)\n([^\r]|$)/gi,"$1\r\n$2");
  400. }
  401. if(childNode.nodeType!=3 && !/^(I|A|STRONG|B|FONT)$/.test(childNode.tagName))cStr+="\r\n";
  402. }
  403. if(hasText || noTextEnable || ele==largestContent)rStr+=cStr+"\r\n";
  404. }
  405. for(i=0;i<childlist.length;i++){
  406. var child=childlist[i];
  407. if(getDepth(child)==getDepth(largestContent)){
  408. if((!largestContent.className && child.className) || (largestContent.className && !child.className) || (largestContent.className && child.className && largestContent.className != child.className))continue;
  409. if((largestContent.className && largestContent.className==child.className)||largestContent.parentNode ==child.parentNode){
  410. getRightStr(child, true);
  411. }else {
  412. getRightStr(child, false);
  413. }
  414. }
  415. }
  416. return rStr.replace(/[\n\r]+$/,"");
  417. }
  418.  
  419. function getI18n(key, args){
  420. var resultStr=i18n[key];
  421. if(args && args.length>0){
  422. args.forEach(function(item){
  423. resultStr=resultStr.replace(/%s/,item);
  424. });
  425. }
  426. return resultStr;
  427. }
  428.  
  429. function getDepth(dom){
  430. var pa=dom,i=0;
  431. while(pa.parentNode){
  432. pa=pa.parentNode;
  433. i++;
  434. }
  435. return i;
  436. }
  437.  
  438. function fetch(forceSingle){
  439. processFunc=null;
  440. var aEles=document.querySelectorAll("a"),list=[];
  441. for(var i=0;i<aEles.length;i++){
  442. var aEle=aEles[i],has=false;
  443. if((!aEle.href || aEle.href.indexOf("javascript")!=-1) && aEle.dataset.href){
  444. aEle.href=aEle.dataset.href;
  445. }
  446. for(var j=0;j<list.length;j++){
  447. if(list[j].href==aEle.href){
  448. list.splice(j,1);
  449. list.push(aEle);
  450. has=true;
  451. break;
  452. }
  453. }
  454. if(!has && aEle.href && /^http/i.test(aEle.href) && ((aEle.innerText.trim()!="" && indexReg.test(aEle.innerText.trim())) || /chapter[\-_]?\d/.test(aEle.href))){
  455. list.push(aEle);
  456. }
  457. }
  458. if(list.length>2 && !forceSingle){
  459. indexDownload(list);
  460. }else{
  461. var blob = new Blob([i18n.info+"\r\n"+document.title+"\r\n\r\n"+getPageContent(document)], {type: "text/plain;charset=utf-8"});
  462. saveAs(blob, document.title+".txt");
  463. }
  464. }
  465.  
  466. document.addEventListener("keydown", function(e) {
  467. if(e.keyCode == 120 && e.ctrlKey) {
  468. fetch(e.shiftKey);
  469. }
  470. });
  471. function setDel(){
  472. var selValue=GM_getValue("selectors");
  473. var selectors=prompt(i18n.del,selValue?selValue:"");
  474. GM_setValue("selectors",selectors);
  475. selValue=GM_getValue("downThreadNum");
  476. var downThreadNum=prompt(i18n.downThreadNum,selValue?selValue:"20");
  477. GM_setValue("downThreadNum",downThreadNum);
  478. if(window.confirm(i18n.reSort)){
  479. GM_setValue("contentSort", true);
  480. }else{
  481. GM_setValue("contentSort", false);
  482. }
  483. }
  484. function customDown(){
  485. processFunc=null;
  486. var customRules=GM_getValue("DACrules_"+document.domain);
  487. var urls=window.prompt(i18n.customInfo,customRules?customRules:"https://xxx.xxx/book-[20-99].html, https://xxx.xxx/book-[01-10].html");
  488. if(urls){
  489. GM_setValue("DACrules_"+document.domain, urls);
  490. var processEles=[];
  491. if(/^http|^ftp/.test(urls)){
  492. [].forEach.call(urls.split(","),function(i){
  493. var varNum=/\[\d+\-\d+\]/.exec(i)[0].trim();
  494. var num1=/\[(\d+)/.exec(varNum)[1].trim();
  495. var num2=/(\d+)\]/.exec(varNum)[1].trim();
  496. var num1Int=parseInt(num1);
  497. var num2Int=parseInt(num2);
  498. var numLen=num1.length;
  499. var needAdd=num1.charAt(0)=="0";
  500. if(num1Int>=num2Int)return;
  501. for(var j=num1Int;j<=num2Int;j++){
  502. var urlIndex=j.toString();
  503. if(needAdd){
  504. while(urlIndex.length<numLen)urlIndex="0"+urlIndex;
  505. }
  506. var curUrl=i.replace(/\[\d+\-\d+\]/,urlIndex).trim();
  507. var curEle=document.createElement("a");
  508. curEle.href=curUrl;
  509. processEles.push(curEle);
  510. curEle.innerText=processEles.length.toString();
  511. }
  512. });
  513. }else{
  514. let urlsArr=urls.split("@@");
  515. [].forEach.call(document.querySelectorAll(urlsArr[0]),function(item){
  516. let has=false;
  517. for(var j=0;j<processEles.length;j++){
  518. if(processEles[j].href==item.href){
  519. processEles.splice(j,1);
  520. processEles.push(item);
  521. has=true;
  522. break;
  523. }
  524. }
  525. if((!item.href || item.href.indexOf("javascript")!=-1) && item.dataset.href){
  526. item.href=item.dataset.href;
  527. }
  528. if(!has && item.href && /^http/i.test(item.href)){
  529. processEles.push(item.cloneNode(1));
  530. }
  531. });
  532. if(urlsArr[1]){
  533. processEles.forEach(ele=>{
  534. ele.href=ele.href.replace(new RegExp(urlsArr[1]), urlsArr[2]);
  535. });
  536. }
  537. if(urlsArr[3]){
  538. processFunc=data=>{return eval(urlsArr[3])};
  539. }
  540. }
  541. indexDownload(processEles);
  542. }
  543. }
  544. GM_registerMenuCommand(i18n.fetch, fetch);
  545. GM_registerMenuCommand(i18n.custom, customDown);
  546. GM_registerMenuCommand(i18n.setting, setDel);
  547. })();