cmoa.jp downloader

Download comic from cmoa.jp

  1. // ==UserScript==
  2. // @name cmoa.jp downloader
  3. // @namespace https://blog.bgme.me
  4. // @match https://www.cmoa.jp/bib/speedreader/*
  5. // @grant none
  6. // @require https://cdn.jsdelivr.net/npm/file-saver@2.0.2/dist/FileSaver.min.js
  7. // @require https://cdn.jsdelivr.net/npm/jszip@3.2.1/dist/jszip.min.js
  8. // @run-at document-end
  9. // @version 1.0
  10. // @author bgme
  11. // @description Download comic from cmoa.jp
  12. // @supportURL https://github.com/yingziwu/Greasemonkey/issues
  13. // @icon https://www.cmoa.jp/favicon.ico
  14. // @license AGPL-3.0-or-later
  15. // ==/UserScript==
  16.  
  17. "use strict";
  18.  
  19. /*
  20. // 在 Console 中使用时请去除此段注释
  21. ['https://cdn.jsdelivr.net/npm/file-saver@2.0.2/dist/FileSaver.min.js', 'https://cdn.jsdelivr.net/npm/jszip@3.2.1/dist/jszip.min.js'].forEach(item => {
  22. let script = document.createElement('script');
  23. script.src = item;
  24. document.body.append(script);
  25. });
  26. */
  27.  
  28. function addButton() {
  29. let button = document.createElement("button");
  30. button.className = "icon_pc";
  31. button.style.cssText = `position: fixed;
  32. top: 15%;
  33. right: 5%;
  34. z-index: 99;
  35. background-color: #0000;
  36. border-style: none;
  37. text-align:center;
  38. vertical-align:baseline;`;
  39.  
  40. let img = document.createElement("img");
  41. img.src =
  42. "";
  43. img.style.cssText = "height: 2em;";
  44.  
  45. button.onclick = function () {
  46. console.log("Start download……");
  47. window.zip = new JSZip();
  48. window.fileNameList = new Array();
  49. setEvent();
  50.  
  51. img.src =
  52. "";
  53. button.onclick = function () {
  54. alert(`Downloaded ${window.fileNameList.length / 3} files.`);
  55. if (confirm("Save to disk?")) {
  56. saveZip();
  57. }
  58. };
  59. };
  60.  
  61. button.appendChild(img);
  62. document.body.appendChild(button);
  63. console.log("Start Load……");
  64. }
  65.  
  66. function setEvent() {
  67. let MutationObserver =
  68. window.MutationObserver ||
  69. window.WebKitMutationObserver ||
  70. window.MozMutationObserver;
  71. const element = document.getElementById("content");
  72. let observer = new MutationObserver(function (mutations) {
  73. mutations.forEach(function (mutation) {
  74. if (mutation.type == "attributes") {
  75. catchPages();
  76. }
  77. });
  78. });
  79.  
  80. observer.observe(element, {
  81. attributes: true, //configure it to listen to attribute changes
  82. attributeFilter: ["style"],
  83. });
  84. }
  85.  
  86. function catchPages() {
  87. const pt_imgs = document.querySelectorAll(".pt-img");
  88. pt_imgs.forEach((pt_img) => {
  89. catchPage(pt_img);
  90. });
  91. }
  92.  
  93. async function catchPage(pt_img) {
  94. let pageNum = pt_img.parentElement.getAttribute("id").match(/\d+$/);
  95. let imgs = pt_img.querySelectorAll("img");
  96. let imgNum = 1;
  97.  
  98. for (let img of imgs) {
  99. let blobUri = img.getAttribute("src");
  100. let blobName = `p${pageNum}i${imgNum}.jpg`;
  101. if (window.fileNameList.includes(blobName)) {
  102. continue;
  103. }
  104. window.fileNameList.push(blobName);
  105. await fetch(blobUri)
  106. .then((response) => response.blob())
  107. .then((blob) => window.zip.file(blobName, blob))
  108. .catch((err) => console.log("catchPage: " + err));
  109. console.log("Downloading " + blobName);
  110. imgNum++;
  111. }
  112. }
  113.  
  114. function saveZip() {
  115. console.log("Save to disk……");
  116. // console.log(zip);
  117. let title = document.querySelector("title").text;
  118. window.zip
  119. .generateAsync({ type: "blob" })
  120. .then((blob) => {
  121. saveAs(blob, `${title}.zip`);
  122. })
  123. .catch((err) => console.log("saveZip: " + err));
  124. }
  125.  
  126. // Run script
  127. window.addEventListener("load", function () {
  128. addButton();
  129. });

QingJ © 2025

镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址