Copy Magnet Links

Copy all unique magnet links on a webpage with a button(一键复制所有磁力链接)

  1. // ==UserScript==
  2. // @name Copy Magnet Links
  3. // @namespace http://your.namespace.com
  4. // @version 0.8
  5. // @description Copy all unique magnet links on a webpage with a button(一键复制所有磁力链接)
  6. // @author 0x0413
  7. // @match https://www.javbus.com/*
  8. // @grant none
  9. // @license GPL-3.0-only
  10. // ==/UserScript==
  11.  
  12.  
  13.  
  14.  
  15.  
  16. (function() {
  17. 'use strict';
  18.  
  19. // 配置选项
  20. const config = {
  21. buttonPosition: {
  22. top: '80px',
  23. left: '10px'
  24. },
  25. buttonStyle: {
  26. backgroundColor: '#1a1a1a',
  27. color: '#ffffff',
  28. padding: '8px 16px',
  29. border: 'none',
  30. borderRadius: '4px',
  31. fontSize: '14px',
  32. fontWeight: '600',
  33. cursor: 'pointer',
  34. transition: 'all 0.3s ease'
  35. }
  36. };
  37.  
  38. // 定义磁力链接的正则表达式
  39. const magnetRegex = /magnet:?\?xt=urn:btih:[0-9a-f]{40}/ig;
  40.  
  41. // 创建通知元素
  42. function createNotification(message, type = 'success') {
  43. const notification = document.createElement('div');
  44. notification.textContent = message;
  45. notification.style.position = 'fixed';
  46. notification.style.top = '20px';
  47. notification.style.right = '20px';
  48. notification.style.padding = '12px 24px';
  49. notification.style.borderRadius = '4px';
  50. notification.style.zIndex = '10000';
  51. notification.style.animation = 'fadeInOut 2.5s ease';
  52. notification.style.backgroundColor = type === 'success' ? '#4caf50' : '#f44336';
  53. notification.style.color = '#ffffff';
  54. notification.style.boxShadow = '0 2px 5px rgba(0,0,0,0.2)';
  55.  
  56. // 添加动画样式
  57. const style = document.createElement('style');
  58. style.textContent = `
  59. @keyframes fadeInOut {
  60. 0% { opacity: 0; transform: translateY(-20px); }
  61. 15% { opacity: 1; transform: translateY(0); }
  62. 85% { opacity: 1; transform: translateY(0); }
  63. 100% { opacity: 0; transform: translateY(-20px); }
  64. }
  65. `;
  66. document.head.appendChild(style);
  67.  
  68. document.body.appendChild(notification);
  69. setTimeout(() => notification.remove(), 2500);
  70. }
  71.  
  72. // 添加复制按钮
  73. function addCopyButton() {
  74. const copyButton = document.createElement('button');
  75. copyButton.textContent = '复制磁力链接';
  76. copyButton.style.position = 'fixed';
  77. copyButton.style.top = config.buttonPosition.top;
  78. copyButton.style.left = config.buttonPosition.left;
  79. copyButton.style.zIndex = '9999';
  80.  
  81. // 应用按钮样式
  82. Object.assign(copyButton.style, config.buttonStyle);
  83.  
  84. // 添加悬停效果
  85. copyButton.addEventListener('mouseover', () => {
  86. copyButton.style.backgroundColor = '#333333';
  87. copyButton.style.transform = 'translateY(-1px)';
  88. });
  89.  
  90. copyButton.addEventListener('mouseout', () => {
  91. copyButton.style.backgroundColor = config.buttonStyle.backgroundColor;
  92. copyButton.style.transform = 'translateY(0)';
  93. });
  94.  
  95. copyButton.addEventListener('click', copyAllMagnetLinks);
  96. document.body.appendChild(copyButton);
  97. return copyButton;
  98. }
  99.  
  100. // 获取所有分页URL
  101. function getAllPageUrls() {
  102. const pageLinks = document.querySelectorAll('.pg a');
  103. const urls = Array.from(pageLinks)
  104. .filter(link => link.textContent.trim() !== '下一頁')
  105. .map(link => link.href);
  106. return [...new Set(urls)]; // 去除重复的URL
  107. }
  108.  
  109. // 从单个页面获取磁力链接
  110. async function getMagnetLinksFromPage(url) {
  111. const magnetLinksSet = new Set();
  112.  
  113. try {
  114. const response = await fetch(url);
  115. const text = await response.text();
  116. const parser = new DOMParser();
  117. const doc = parser.parseFromString(text, 'text/html');
  118.  
  119. const elements = doc.querySelectorAll('a[href^="magnet:"], div, span, p');
  120. elements.forEach(element => {
  121. const text = element.textContent || '';
  122. const matches = text.match(magnetRegex);
  123. if (matches) {
  124. matches.forEach(match => magnetLinksSet.add(match));
  125. }
  126.  
  127. if (element.href && element.href.match(magnetRegex)) {
  128. magnetLinksSet.add(element.href);
  129. }
  130. });
  131.  
  132. return Array.from(magnetLinksSet);
  133. } catch (error) {
  134. console.error(`获取页面 ${url} 失败:`, error);
  135. return [];
  136. }
  137. }
  138.  
  139. // 复制所有磁力链接
  140. async function copyAllMagnetLinks() {
  141. const button = this;
  142. button.disabled = true;
  143. button.style.opacity = '0.7';
  144. button.textContent = '正在复制...';
  145.  
  146. try {
  147. const urls = getAllPageUrls();
  148. const currentUrl = window.location.href;
  149. if (!urls.includes(currentUrl)) {
  150. urls.unshift(currentUrl);
  151. }
  152.  
  153. const allMagnetLinks = new Set();
  154. let processedPages = 0;
  155.  
  156. // 使用 Promise.all 并发处理所有页面
  157. const updateProgress = () => {
  158. processedPages++;
  159. button.textContent = `正在复制... (${processedPages}/${urls.length}页)`;
  160. };
  161.  
  162. const pageResults = await Promise.all(
  163. urls.map(async url => {
  164. const links = await getMagnetLinksFromPage(url);
  165. updateProgress();
  166. return links;
  167. })
  168. );
  169.  
  170. // 合并所有页面的结果
  171. pageResults.flat().forEach(link => allMagnetLinks.add(link));
  172.  
  173. if (allMagnetLinks.size > 0) {
  174. const allLinks = Array.from(allMagnetLinks).join('\n');
  175. await navigator.clipboard.writeText(allLinks);
  176. createNotification(`成功复制 ${allMagnetLinks.size} 个磁力链接!`);
  177. } else {
  178. createNotification('未找到磁力链接', 'error');
  179. }
  180. } catch (error) {
  181. console.error('复制失败:', error);
  182. createNotification('复制失败,请重试', 'error');
  183. } finally {
  184. // 恢复按钮状态
  185. setTimeout(() => {
  186. button.disabled = false;
  187. button.style.opacity = '1';
  188. button.textContent = '复制磁力链接';
  189. }, 1000);
  190. }
  191. }
  192.  
  193. // 初始化
  194. if (document.readyState === 'loading') {
  195. document.addEventListener('DOMContentLoaded', addCopyButton);
  196. } else {
  197. addCopyButton();
  198. }
  199. })();