V2PH Album Downloader

Album Downloader for v2ph.com. After logging in and opening the album page, click the "Download all images" option in the script menu to start downloading.

  1. // ==UserScript==
  2. // @name V2PH Album Downloader
  3. // @namespace https://greasyfork.org/zh-CN/users/220174-linepro
  4. // @match https://www.v2ph.com/album/*
  5. // @grant GM_download
  6. // @grant GM_xmlhttpRequest
  7. // @grant GM_registerMenuCommand
  8. // @grant GM_unregisterMenuCommand
  9. // @grant GM_addStyle
  10. // @version 1.2
  11. // @author LinePro
  12. // @license MIT
  13. // @description Album Downloader for v2ph.com. After logging in and opening the album page, click the "Download all images" option in the script menu to start downloading.
  14. // @run-at document-idle
  15. // @require https://update.greasyfork.org/scripts/473358/1237031/JSZip.js
  16. // @require https://cdn.jsdelivr.net/npm/idb-keyval@6/dist/umd.js
  17. // ==/UserScript==
  18. (async () => {
  19. const jQuery = unsafeWindow.jQuery;
  20. GM_addStyle(`
  21. .spinner-border {
  22. display: inline-block;
  23. width: 1.25rem;
  24. height: 1.25rem;
  25. vertical-align: -.125em;
  26. border: .175em solid currentColor;
  27. border-right-color: transparent;
  28. border-radius: 50%;
  29. -webkit-animation: .75s linear infinite spinner-border;
  30. animation: .75s linear infinite spinner-border
  31. }
  32. .sr-only {
  33. position: absolute;
  34. width: 1px;
  35. height: 1px;
  36. padding: 0;
  37. margin: -1px;
  38. overflow: hidden;
  39. clip: rect(0,0,0,0);
  40. white-space: nowrap;
  41. border: 0
  42. }
  43.  
  44. .modal-loading {
  45. margin-left: 0.5rem;
  46. }
  47. `);
  48. class BootStrap4Model {
  49. constructor(title = 'Modal', text = '') {
  50. const wrap = (() => {
  51. const modal = document.createElement('div');
  52. modal.className = 'modal';
  53. modal.setAttribute('tabindex', '-1');
  54. modal.dataset.backdrop = 'static';
  55. modal.dataset.keyboard = 'false';
  56. const modalDialog = document.createElement('div')
  57. modalDialog.className = 'modal-dialog';
  58. const modalContent = document.createElement('div')
  59. modalContent.className = 'modal-content';
  60. const modalHeader = document.createElement('div');
  61. modalHeader.className = 'modal-header';
  62. const modalTitle = document.createElement('h5');
  63. this.modalTitle = modalTitle;
  64. modalTitle.className = 'modal-title';
  65. modalTitle.innerText = title;
  66. const modalLoading = document.createElement('div');
  67. this.modalLoading = modalLoading;
  68. modalLoading.className = 'modal-loading spinner-border';
  69. modalLoading.style.display = 'none';
  70. modalLoading.setAttribute('role', 'status');
  71. const modalLoadingSpan = document.createElement('span');
  72. modalLoadingSpan.className = 'sr-only';
  73. modalLoadingSpan.innerText = 'Loading...';
  74. modalLoading.appendChild(modalLoadingSpan);
  75. modalTitle.appendChild(modalLoading);
  76. modalHeader.appendChild(modalTitle);
  77. const modalBody = document.createElement('div');
  78. this.modalBody = modalBody;
  79. modalBody.className = 'modal-body';
  80. modalBody.innerText = text;
  81. const modalFooter = document.createElement('div');
  82. modalFooter.className = 'modal-footer';
  83. const closeBtn = document.createElement('button');
  84. this.closeBtn = closeBtn;
  85. closeBtn.className = 'btn btn-danger';
  86. closeBtn.setAttribute('data-dismiss', 'modal');
  87. closeBtn.innerText = 'Close';
  88. modalFooter.appendChild(closeBtn);
  89. modalContent.appendChild(modalHeader);
  90. modalContent.appendChild(modalBody);
  91. modalContent.appendChild(modalFooter);
  92. modalDialog.appendChild(modalContent);
  93. modal.appendChild(modalDialog);
  94. return modal;
  95. })();
  96. document.body.appendChild(wrap);
  97. this.wrap = wrap;
  98. }
  99.  
  100. show() {
  101. jQuery(this.wrap).modal('show');
  102. }
  103.  
  104. hide() {
  105. jQuery(this.wrap).modal('hide');
  106. }
  107.  
  108. setTitle(title) {
  109. this.modalTitle.innerText = title;
  110. }
  111.  
  112. setText(text) {
  113. this.modalBody.innerText = text;
  114. }
  115.  
  116. setLoading(loading) {
  117. this.modalLoading.style.display = loading ? '' : 'none';
  118. }
  119.  
  120. setCloseBtn(text, callback) {
  121. this.closeBtn.innerText = text;
  122. this.closeBtn.onclick = callback;
  123. }
  124. }
  125.  
  126. const getPageFromUrl = (url) => {
  127. const search = new URL(url).searchParams;
  128. const page = search.get("page");
  129. if (page) {
  130. return parseInt(page);
  131. }
  132. return 1;
  133. }
  134.  
  135. const fetch = (url) => {
  136. return new Promise((resolve, reject) => {
  137. GM_xmlhttpRequest({
  138. method: "GET",
  139. url: url,
  140. headers: {
  141. Referer: location.href
  142. },
  143. responseType: 'blob',
  144. onload: response => resolve(response.response),
  145. onerror: error => reject(error)
  146. });
  147. });
  148. }
  149.  
  150. const downloadingStatus = await idbKeyval.get("downloadingStatus");
  151.  
  152. const downloadingAlbumId = downloadingStatus?.albumId;
  153. const downloadingPage = downloadingStatus?.page;
  154.  
  155. const currentAlbumId = location.pathname.split("/").pop();
  156.  
  157. const page = getPageFromUrl(location.href);
  158.  
  159. const registerMenu = () => {
  160. const caption = 'Download all images';
  161. GM_registerMenuCommand(caption, async () => {
  162. GM_unregisterMenuCommand(caption);
  163. if (page !== 1) {
  164. await idbKeyval.set("downloadingStatus", {
  165. albumId: currentAlbumId,
  166. page: 1
  167. });
  168. location.search = '';
  169. } else {
  170. await startDownload();
  171. }
  172. });
  173. };
  174.  
  175. const startDownload = async () => {
  176. let cancelled = false;
  177. const modal = new BootStrap4Model('Downloading');
  178. modal.show();
  179. modal.setCloseBtn('Cancel', async () => {
  180. cancelled = true;
  181. await idbKeyval.clear();
  182. registerMenu();
  183. });
  184.  
  185. const values = [];
  186. let index = 1;
  187. const imgList = document.querySelectorAll("img.album-photo");
  188. for (const element of imgList) {
  189. const url = element.dataset.src;
  190. const fileName = url.split("/").pop();
  191. modal.setText(`Downloading ${page}-${index}-${fileName}...`);
  192. modal.setLoading(true);
  193.  
  194. const blob = await fetch(url);
  195. values.push({
  196. name: `${page}-${index++}-${fileName}`,
  197. blob
  198. });
  199. await new Promise(resolve => setTimeout(resolve, 500));
  200. if (cancelled) {
  201. return;
  202. }
  203. }
  204. await idbKeyval.set(`page-${page}`, values);
  205. const lastPageUrl = document.querySelector(".page-item:last-of-type a")?.href;
  206. if (page === getPageFromUrl(lastPageUrl)) {
  207. const zip = new JSZip();
  208. const values = (await idbKeyval.values()).flat().slice(1);
  209. for (const { name, blob } of values) {
  210. zip.file(name, blob);
  211. }
  212. const zipBlob = await zip.generateAsync({ type: "blob" })
  213. const url = URL.createObjectURL(zipBlob);
  214. const title = document.querySelector(".h5.text-center.mb-3").textContent || 'album';
  215. GM_download(url, `${title}.zip`);
  216. await idbKeyval.clear();
  217. modal.setTitle('Downloaded Complete');
  218. modal.setText(`${values.length} images downloaded.`);
  219. modal.setLoading(false);
  220. modal.setCloseBtn('Close', () => modal.hide());
  221. } else {
  222. const nextPage = page + 1;
  223. await idbKeyval.set("downloadingStatus", {
  224. albumId: currentAlbumId,
  225. page: nextPage
  226. });
  227. location.search = `?page=${nextPage}`;
  228. }
  229. };
  230.  
  231. if (downloadingAlbumId === currentAlbumId && downloadingPage === page) {
  232. await startDownload();
  233. } else {
  234. await idbKeyval.clear();
  235. registerMenu();
  236. }
  237. })();