Magnet Copy & Preview

Enhanced magnet link handling with copy, preview, and image URL features//Add buttons to copy magnet links and preview on magnet.pics, works on 1cili.com and other sites

// ==UserScript==
// @name             		Magnet Copy & Preview
// @namespace        	    http://tampermonkey.net/
// @version          		2025.09.12
// @description      		Enhanced magnet link handling with copy, preview, and image URL features//Add buttons to copy magnet links and preview on magnet.pics, works on 1cili.com and other sites
// @author           		庄引
// @icon             		
// @match            		*://u3c3.com/*
// @match            		*://hjd2048.com/*
// @match            		*://*.cctv10.cc/*
// @match            		*://*.cctv12.cc/*
// @match            		*://*.1cili.com/!*
// @match            		*://sukebei.nyaa.si/*
// @match           		*://btdig.com/*
// @match            		*://whatslink.info*
// @match            		*://btsow.pics/*
// @match            		*://*.sis001.com/*
// @grant            		GM_setClipboard
// @grant           		GM_openInTab
// @grant           		GM_xmlhttpRequest
// @grant            		GM_addElement
// @grant            		GM_addStyle
// @run-at           		document-idle
// @license MIT
// ==/UserScript==

(function () {
	'use strict';

	// Debug flag to reduce noisy logs in production
	const DEBUG = false;

	// Schedule DOM writes safely to avoid ResizeObserver loops
	const schedule = (callback) => {
		return requestAnimationFrame(() => {
			try { callback(); } catch (error) { console.error(error); }
		});
	};

	/**
	* 创建并添加HTML元素到指定父节点
	* @param {string} tagName - HTML标签名称,默认为'button'
	* @param {string} innerHTML - 元素的内部HTML内容
	* @param {Object} options - 要应用到元素上的属性
	* @param {HTMLElement} parentNode - 父节点元素
	* @param {boolean} flag - 是否添加到父节点末尾,true使用append,false使用prepend
	*/
	const addElement = (tagName = `button`, innerHTML, options, parentNode, flag = true) => {
		if (!parentNode) {
			console.warn('addElement: parentNode is null or undefined');
			return;
		}

		const el = document.createElement(tagName);
		el.innerHTML = innerHTML;
		Object.assign(el, options);

		// 使用 requestAnimationFrame 来避免 ResizeObserver 循环问题
		schedule(() => {
			try {
				parentNode[flag ? 'prepend' : 'append'](el);
				if (DEBUG) console.log(parentNode);
			} catch (error) {
				console.error('addElement error:', error);
			}
		});
	};

	/**
	 * 监听并复制图片URL
	 * @param {string} parentDiv - 父元素选择器
	 * @param {string} targetNode - 图片元素选择器
	 * @param {string} attribute - 要监听的属性名
	 * @param {string} displayDiv - 显示元素选择器
	 * @returns {boolean} 是否成功设置监听
	 */
	const monitorAndCopyImageUrls = (parentDiv, targetNode, attribute, displayDiv) => {
		const searchDiv = document.querySelector(parentDiv);
		if (!searchDiv) {
			console.error(`找不到父元素: ${parentDiv}`);
			return false;
		}

		// 防抖函数,避免频繁更新
		let debounceTimer;
		const debounce = (fn, delay) => {
			return function (...args) {
				clearTimeout(debounceTimer);
				debounceTimer = setTimeout(() => fn.apply(this, args), delay);
			};
		};

		// 处理图片URL的函数
		const processImages = () => {
			try {
				const images = document.querySelectorAll(targetNode);
				if (images.length === 0) {
					return;
				}

				// 收集所有图片URL
				const urls = Array.from(images).map((img) => img[attribute]);
				const displayElement = document.querySelector(displayDiv);
				if (!displayElement) {
					return;
				}

				// 使用 DocumentFragment 来批量操作 DOM,减少重排
				const fragment = document.createDocumentFragment();

				// 创建或更新显示区域
				let section = displayElement.querySelector('section');
				if (!section) {
					section = document.createElement("section");
					fragment.appendChild(section);
				}

				// 更新URL列表
				section.innerHTML = '';
				urls.forEach((url) => {
					const p = document.createElement("p");
					p.textContent = url;
					section.appendChild(p);
				});

				// 如果section是新创建的,添加到fragment
				if (!displayElement.querySelector('section')) {
					displayElement.appendChild(fragment);
				}

				// 添加或更新复制按钮
				let copyButton = displayElement.querySelector('.copy-urls-button');
				if (!copyButton) {
					// 使用 setTimeout 来延迟按钮创建,避免在 MutationObserver 回调中直接修改 DOM
					setTimeout(() => {
						addElement(
							'button',
							'复制所有URL',
							{
								className: 'copy-urls-button',
								onclick: (e) => {
									try {
										const currentImages = document.querySelectorAll(targetNode);
										const currentUrls = Array.from(currentImages).map((img) => img[attribute]);
										GM_setClipboard(currentUrls.join("\n"), "text");
										e.target.textContent = '已复制所有URL';
										setTimeout(() => {
											e.target.textContent = '复制所有URL';
										}, 3000);
									} catch (error) {
										console.error("复制到剪贴板时出错:", error);
										e.target.textContent = '复制失败';
										setTimeout(() => {
											e.target.textContent = '复制所有URL';
										}, 3000);
									}
								}
							},
							displayElement
						);
					}, 0);
				}
			} catch (error) {
				console.error("处理图片时出错:", error);
			}
		};

		// 使用防抖处理图片更新
		const debouncedProcessImages = debounce(processImages, 1000);

		// 创建MutationObserver监听DOM变化
		const observer = new MutationObserver((mutations) => {
			const hasRelevantChanges = mutations.some((mutation) => {
				return (
					(mutation.type === "attributes" &&
						mutation.attributeName === attribute) ||
					mutation.type === "childList"
				);
			});

			if (hasRelevantChanges) {
				// 使用 requestAnimationFrame 来避免 ResizeObserver 循环问题
				schedule(() => {
					debouncedProcessImages();
				});
			}
		});

		// 配置观察选项
		const config = {
			childList: true,
			subtree: true,
			attributes: true,
			attributeFilter: [attribute],
		};

		// 开始观察
		observer.observe(searchDiv, config);

		// 初始处理
		processImages();

		return true;
	};



	// Function to extract hash from magnet URL
	function extractHashFromMagnet(magnetUrl) {
		const btihMatch = magnetUrl.match(/urn:btih:([a-zA-Z0-9]+)/i);
		if (btihMatch && btihMatch[1]) {
			return btihMatch[1].toLowerCase();
		}
		return '';
	}
	class Drawer {
		static instance = null;

		constructor(options = {}) {
			this.direction = options.direction || 'left';
			this.width = options.width || '300px';
			this.title = options.title || 'Drawer';
			this.init();
		}

		init() {
			// 如果存在之前的实例,先移除它
			if (Drawer.instance) {
				document.body.removeChild(Drawer.instance.drawer);
			}

			// Create drawer container
			this.drawer = document.createElement('div');
			this.drawer.style.cssText = `
                position: fixed;
                top: 0;
                ${this.direction}: 0;
                width: ${this.width};
                height: 100vh;
                background: white;
                box-shadow: 0 0 10px rgba(0,0,0,0.1);
                transform: translateX(${this.direction === 'left' ? '-100%' : '100%'});
                transition: transform 0.3s ease;
                z-index: 1000;
            `;

			// Create header
			const header = document.createElement('div');
			header.style.cssText = `
                display: flex;
                justify-content: space-between;
                align-items: center;
                padding: 16px;
                border-bottom: 1px solid #eee;
            `;

			// Create title
			const title = document.createElement('h5');
			title.textContent = this.title;
			title.style.margin = '0';

			// Create close button
			const closeBtn = document.createElement('button');
			closeBtn.textContent = '×';
			closeBtn.style.cssText = `
                background: none;
                border: none;
                font-size: 24px;
                cursor: pointer;
                padding: 0;
                color: #666;
            `;
			closeBtn.onclick = () => this.close();

			// Create content container
			this.content = document.createElement('div');
			this.content.style.cssText = `
                padding: 16px;
                height: calc(100vh - 60px);
                overflow-y: auto;
            `;

			// Assemble drawer
			header.appendChild(title);
			header.appendChild(closeBtn);
			this.drawer.appendChild(header);
			this.drawer.appendChild(this.content);
			document.body.appendChild(this.drawer);

			// 保存当前实例
			Drawer.instance = this;
		}

		open() {
			this.drawer.style.transform = 'translateX(0)';
		}

		close() {
			this.drawer.style.transform = `translateX(${this.direction === 'left' ? '-100%' : '100%'})`;
			// Clear content after animation completes
			setTimeout(() => {
				this.content.innerHTML = '';
			}, 300); // Match the transition duration
		}

		setContent(content) {
			this.content.innerHTML = content;
		}
	}

	// Usage example
	function openDrawer(url, title, width = '40vw', direction = 'right') {
		// 如果已经存在drawer实例,先清空内容
		const existingDrawer = document.querySelector('.drawer-container');
		if (existingDrawer) {
			const content = existingDrawer.querySelector('.drawer-content');
			if (content) {
				content.innerHTML = '';
			}
		}

		const drawer = new Drawer({
			direction: direction,
			width: `${width}`,
			title: `${title} `
		});
		drawer.setContent(`<iframe width='100%;' height='100%' src=${url}></iframe>`);
		drawer.open();
	}
	// Special handling for 1cili.com
	function handle1CiliSite() {
		const magnetBoxes = document.querySelectorAll('.magnet-box');

		if (magnetBoxes.length > 0) {
			// For each magnet box
			magnetBoxes.forEach(box => {
				const inputField = box.querySelector('#input-magnet');
				if (!inputField) return;

				const magnetUrl = inputField.value;
				const hash = extractHashFromMagnet(magnetUrl);

				if (!hash) return;

				// Get title from page
				const pageTitle = document.querySelector('.magnet-title')?.textContent || 'Magnet Preview';

				// Find the input-group-btn div where the existing buttons are
				const btnGroup = box.querySelector('.input-group-btn');

				if (btnGroup) {
					// 使用 requestAnimationFrame 来避免 ResizeObserver 循环问题
					requestAnimationFrame(() => {
						try {
							// Create a new preview button
							const previewBtn = document.createElement('a');
							previewBtn.className = 'btn preview-button-1cili';
							previewBtn.innerHTML = '<svg class="svg-icon"><use xlink:href="/assets/icons.svg#icon-search"></use></svg>';
							previewBtn.title = '预览 Preview';
							previewBtn.href = 'javascript:void(0);';

							// Add click event for preview
							previewBtn.addEventListener('click', function (e) {
								e.preventDefault();
								e.stopPropagation();

								// Open preview in side panel
								openDrawer(`https://magnet.pics/m/${hash}`, pageTitle);
							});

							// Add the button to the button group
							btnGroup.appendChild(previewBtn);
						} catch (error) {
							console.error('handle1CiliSite error:', error);
						}
					});
				}
			});
		}
	}

	/**
	 * 处理常规网站的磁力链接功能
	 * @param {string} datalist - 要处理的元素列表选择器
	 * @param {string} hash - 包含磁力链接hash的元素选择器
	 * @param {string} title - 标题元素选择器
	 * @param {string} size - 文件大小元素选择器
	 * @param {string} date - 日期元素选择器
	 * @param {string} magnet - 磁力链接元素选择器
	 * @param {boolean} flag - 是否将按钮添加到元素末尾,true使用append,false使用prepend
	 */
	const handleRegularSites = (datalist, hash, title, size, date, magnet, flag) => {
		if (DEBUG) console.log(datalist, hash, title, size, date, magnet, flag);
		if (DEBUG) console.log(document.querySelectorAll(datalist));
		//在磁力链接详情页面(.fa-magnet)也添加预览按钮
		if (/[0-9a-fA-F]{40}/.test(window.location.href)) {
			// 使用 requestAnimationFrame 来避免 ResizeObserver 循环问题
			schedule(() => {
				try {
					// 构建完整的磁力链接,包含标题、大小和日期信息
					const link = `magnet:?xt=urn:btih:${window.location.href.match(/[0-9a-fA-F]{40}/)[0].toLowerCase()}&dn=${document.querySelector('tbody > tr:nth-child(5) > td:nth-child(2)').innerText}🔞Size=${document.querySelector('tbody > tr:nth-child(6) > td:nth-child(2)').innerText}🔞Date=${document.querySelector(' tbody > tr:nth-child(7) > td:nth-child(2)').innerText}`;
					document.querySelector('tbody > tr:nth-child(4) > td:nth-child(2) > div > a').textContent = link;

					// 添加复制按钮(幂等)
					const detailContainer = document.querySelector('.fa-magnet');
					if (!detailContainer) return;
					if (!detailContainer.querySelector('.magnet-btn.copy-btn')) {
						addElement('a', '📋',
							{
								className: 'magnet-btn copy-btn',
								title: 'Copy Magnet Link',
								onclick: (e) => {
									e.preventDefault();
									e.stopPropagation();
									GM_setClipboard(decodeURIComponent(link));
								},
							},
							detailContainer
						)
					}
					// 添加预览按钮(幂等)
					if (!detailContainer.querySelector('.magnet-btn.preview-btn')) {
						addElement('a', '👁️',
							{
								className: 'magnet-btn preview-btn',
								title: 'Preview on magnet.pics',
								onclick: (e) => {
									e.preventDefault();
									e.stopPropagation();
									openDrawer(`https://magnet.pics/m/${window.location.href.match(/[0-9a-fA-F]{40}/)[0].toLowerCase()}`, document.querySelector('.fa-folder-open').innerText);
								},
							},
							detailContainer
						)
					}
				} catch (error) {
					console.error('磁力链接详情页面处理错误:', error);
				}
			});
		};
		// 遍历所有匹配的元素
		document.querySelectorAll(datalist).forEach((element, index) => {
			if (DEBUG) console.log(element);

			const hashElement = element.querySelector(hash);
			if (!hashElement) return;

			// 从链接中提取40位的磁力链接hash
			const hashMatch = hashElement.href.match(/[0-9a-fA-F]{40}/);
			if (!hashMatch) return;

			// 构建完整的磁力链接,包含标题、大小和日期信息
			const link = `magnet:?xt=urn:btih:${hashMatch[0].toLowerCase()}&dn=${element.querySelector(title).innerText}🔞Size=${element.querySelector(size).innerText}🔞Date=${element.querySelector(date).innerText}`;

			// 使用 schedule 来避免 ResizeObserver 循环问题
			schedule(() => {
				try {
					const magnetContainer = element.querySelector(magnet);
					if (!magnetContainer) return;
					magnetContainer.textContent = link;

					// 添加复制按钮(幂等)
					if (!magnetContainer.querySelector('.magnet-btn.copy-btn')) {
						addElement('a', '📋',
							{
								className: 'magnet-btn copy-btn',
								title: 'Copy Magnet Link',
								onclick: (e) => {
									e.preventDefault();
									e.stopPropagation();
									GM_setClipboard(decodeURIComponent(link));
								},
							},
							magnetContainer
						)
					}

					// 添加预览按钮(幂等)
					if (!magnetContainer.querySelector('.magnet-btn.preview-btn')) {
						addElement('a', '👁️',
							{
								className: 'magnet-btn preview-btn',
								title: 'Preview on magnet.pics',
								onclick: (e) => {
									e.preventDefault();
									e.stopPropagation();
									openDrawer(`https://magnet.pics/m/${hashMatch[0].toLowerCase()}`, element.querySelector(title).innerText);
								},
							},
							magnetContainer
						)
					}
				} catch (error) {
					console.error('handleRegularSites error:', error);
				}
			});
		})
	}
	const isIncludes = (str) => window.location.hostname.includes(str);
	switch (true) {
		case isIncludes('1cili'):
			handle1CiliSite();
			break;
		case isIncludes('whatslink.info'):
			// 自定义样式和图片URL复制功能
			GM_addStyle(`.img {flex-shrink: 0;height: auto;width: auto;margin-right: 12px;}
        .banner-title,.banner,.disc,div.wrapper:nth-child(5), #app > div > div:nth-child(7),#app > div > div.footer {display:none}
        #app {max-width: 100vw !important; padding: 5px !important;}
        .content {padding: 5px !important;}
        .wrapper {margin: 0 !important;}
        .el-input-group {width: 100vw;}
        body {place-items: baseline !important;}`);
			monitorAndCopyImageUrls(
				"div.search",
				".image-list .img img",
				"src",
				"div.search"
			);
			break;
		case isIncludes('u3c3'):
		case isIncludes("sukebei.nyaa.si"):
			GM_addStyle(`/*u9a9*/
                .container .ad,.hdr-link,tr td:nth-child(1),tr td:nth-child(3),tr td:nth-child(6),tr td:nth-child(7),.text-center{display:none !important;}
                .table {width: 100%;}
                .torrent-list>tbody>tr>td { white-space: normal;}
                .torrent-list > tbody > tr > td {max-width:90vw;white-space: normal !important;}
                .data-list .row {padding: 0;}
                .navbar-form .input-group {position: fixed; left:0;width: 100vw;}`
			);
			handleRegularSites("tbody tr",
				"td:nth-child(3) a:last-child",
				"td:nth-child(2) a",
				"td:nth-child(4)",
				"td:nth-child(5)",
				"td:nth-child(2) a:last-child");
			break;
		case isIncludes('btdig.com'):
			// GM_addStyle(`center div div {width: 100vw;}`);
			handleRegularSites(
				".one_result > div",
				".torrent_name a",
				".torrent_name a",
				".torrent_size",
				".torrent_age",
				".torrent_magnet");
			break;
		case isIncludes('btsow.pics'):
			// GM_addStyle(`center div div {width: 100vw;}`);
			GM_addStyle(`/*btsow*/
                .search {position: sticky !important;top: 80px !important;}
                .form-inline .input-group {width: 100%;}
                .hidden-xs:not(.tags-box,.text-right,.search,.search-container),.data-list:not(.detail) .size,.data-list:not(.detail)  .date{ display: none !important;}`);

			// 为 btsow.pics 添加等待机制,确保内容加载完成
			const waitForBtsowContent = () => {
				const elements = document.querySelectorAll(".q-infinite-scroll .row");
				if (elements.length > 0) {
					console.log("btsow.pics 内容已加载,找到元素数量:", elements.length);
					// 使用 requestAnimationFrame 来避免 ResizeObserver 循环问题
					requestAnimationFrame(() => {
						handleRegularSites(
							".q-infinite-scroll .row",
							"a",
							"a",
							".size",
							".date",
							"a"
						);
					});
				} else {
					console.log("btsow.pics 内容未加载,等待中...");
					setTimeout(waitForBtsowContent, 2000); // 每2秒检查一次
				}
			};

			// 开始等待内容加载
			waitForBtsowContent();
			break;
		case isIncludes('hjd2048.com'):
		case isIncludes('cctv10.cc'):
		case isIncludes('cctv12.cc'):
			break;
		// case isIncludes('javdb'):
		// 	GM_addStyle(`.container:not(.is-max-desktop):not(.is-max-widescreen) {max-width: 100%;}
		//     .movie-list .item .video-title {white-space: normal;}`);
		// 	replaceImg();
		// 	break;
		case isIncludes('javbus'):
			GM_addStyle(`.masonry #waterfall {display: grid; grid-template-columns: repeat(auto-fill, minmax(465px, 1fr)); gap: 0px; padding: 0px;}
            .movie-box{width:465px !important;height:400px !important;margin:0 !important;}
            #waterfall .masonry-brick{position:relative !important; top: 0px !important;  left: 0 !important;margin:5px}
            .item-tag {display: inline-block;}.movie-box .photo-frame {height: auto !important;margin:0 !important;}.movie-box img {height: auto !important;}`);
			break;
		default:
			GM_addStyle(`* {-webkit-touch-callout: text;-webkit-user-select: text !important;-moz-user-select: text;user-select: text;}`);;
	}

})();