// ==UserScript== // @name 复制授权拦截器 // @namespace https://viayoo.com/ // @version 3.20 // @description 复制必弹窗,2次允许后开始自由复制,3次拒绝永久禁用(变灰,点击灰点恢复变红),点击小红点恢复,红绿切换自由复制。 // @author Aloazny & Deepseek // @match *://*/* // @run-at document-start // @icon  // @grant GM_setClipboard // @license MIT // ==/UserScript== (function () { const KEY = 'copyMonitor:v3'; if (window[KEY]) return; window[KEY] = true; const CONFIG = { COPY_TIME_THRESHOLD: 2000, COPY_COUNT_THRESHOLD: 5, QUICK_COPY_INTERVAL: 50, LONG_PRESS_THRESHOLD: 500 }; let copyRecords = [], lastTouchTime = 0, longPressTimer = null, hasUserAction = false, isMonitoring = true, lastUserClickTime = 0; const setupUserActionDetection = () => { document.addEventListener('click', (e) => { hasUserAction = true; lastUserClickTime = Date.now(); const isCopyButton = e.target.closest && (e.target.closest('[data-action="copy"]') || e.target.closest('[onclick*="copy"]')); if (isCopyButton) setTimeout(() => {}, 1000); }, { capture: true, passive: true }); document.addEventListener('touchstart', (e) => { hasUserAction = true; lastTouchTime = Date.now(); longPressTimer = setTimeout(() => { hasUserAction = true; }, CONFIG.LONG_PRESS_THRESHOLD); }, { capture: true, passive: true }); document.addEventListener('touchend', () => { clearTimeout(longPressTimer); }, { capture: true, passive: true }); document.addEventListener('mousedown', () => { hasUserAction = true; }, { capture: true, passive: true }); document.addEventListener('selectionchange', () => { const selection = window.getSelection().toString(); if (selection.length > 0) hasUserAction = true; }, { capture: true, passive: true }); }; const recordCopy = (method) => { if (!isMonitoring) return; const now = Date.now(), isRecentUserAction = now - lastUserClickTime < 1000; copyRecords.push({ timestamp: now, method: method, hasUserAction: hasUserAction || isRecentUserAction }); copyRecords = copyRecords.filter(record => now - record.timestamp <= CONFIG.COPY_TIME_THRESHOLD); detectMaliciousCopy(); }; const detectMaliciousCopy = () => { const now = Date.now(); const copiesWithoutUserAction = copyRecords.filter(record => !record.hasUserAction && now - record.timestamp <= CONFIG.COPY_TIME_THRESHOLD); if (copiesWithoutUserAction.length >= CONFIG.COPY_COUNT_THRESHOLD) { const hasMaliciousPattern = checkMaliciousPattern(copiesWithoutUserAction); if (hasMaliciousPattern) blockMaliciousCopy("检测到异常复制行为:短时间内自动复制次数过多"); return; } if (copiesWithoutUserAction.length >= 2) { const hasQuickCopy = checkMaliciousPattern(copiesWithoutUserAction); if (hasQuickCopy) blockMaliciousCopy("检测到异常复制行为:快速连续自动复制"); } }; const checkMaliciousPattern = (copies) => { if (copies.length < 2) return false; for (let i = 1; i < copies.length; i++) { const timeDiff = copies[i].timestamp - copies[i - 1].timestamp; if (timeDiff <= CONFIG.QUICK_COPY_INTERVAL) return true; } return false; }; const blockMaliciousCopy = (reason) => { console.warn('复制拦截:', reason); isMonitoring = false; showWarning(reason); setTimeout(() => { isMonitoring = true; copyRecords = []; }, 10000); }; const showWarning = (message) => { if (document.querySelector('.copy-monitor-warning')) return; const modal = document.createElement('div'); modal.className = 'copy-monitor-warning'; modal.style.cssText = `position:fixed;top:0;left:0;width:100%;height:100%;background:rgba(0,0,0,0.5);display:flex;align-items:center;justify-content:center;z-index:2147483647;font-family:-apple-system,BlinkMacSystemFont,"Segoe UI",Roboto,sans-serif;`; const dialog = document.createElement('div'); dialog.style.cssText = `background:white;padding:20px;border-radius:12px;max-width:80%;text-align:center;box-shadow:0 4px 20px rgba(0,0,0,0.15);`; const title = document.createElement('h3'); title.textContent = '安全警告'; title.style.cssText = 'color:#ff3b30;margin:0 0 10px 0;'; const msg = document.createElement('p'); msg.textContent = message; msg.style.cssText = 'margin:0 0 20px 0;color:#333;'; const button = document.createElement('button'); button.textContent = '我知道了'; button.style.cssText = `background:#007AFF;color:white;border:none;padding:10px 20px;border-radius:8px;cursor:pointer;font-size:14px;`; button.onclick = () => { modal.remove(); }; dialog.append(title, msg, button); modal.appendChild(dialog); document.body.appendChild(modal); }; const monitorCopyMethods = () => { const originalExecCommand = document.execCommand; const originalWriteText = navigator.clipboard?.writeText; const originalWrite = navigator.clipboard?.write; document.execCommand = function (command, showUI, value) { if (command === 'copy') { const now = Date.now(), isRecentUserAction = now - lastUserClickTime < 1000; if (hasUserAction || isRecentUserAction) { recordCopy('execCommand'); return originalExecCommand.apply(this, arguments); } else { recordCopy('execCommand'); return askForCopyPermission().then((allowed) => { return allowed ? originalExecCommand.apply(this, arguments) : false; }).catch(() => false); } } return originalExecCommand.apply(this, arguments); }; if (navigator.clipboard && originalWriteText) { navigator.clipboard.writeText = function (text) { const now = Date.now(), isRecentUserAction = now - lastUserClickTime < 1000; recordCopy('clipboard.writeText'); if (hasUserAction || isRecentUserAction) return originalWriteText.call(this, text); return askForCopyPermission().then((allowed) => { return allowed ? originalWriteText.call(this, text) : Promise.reject(new Error('用户拒绝复制')); }); }; } if (navigator.clipboard && originalWrite) { navigator.clipboard.write = function (data) { const now = Date.now(), isRecentUserAction = now - lastUserClickTime < 1000; recordCopy('clipboard.write'); if (hasUserAction || isRecentUserAction) return originalWrite.call(this, data); return askForCopyPermission().then((allowed) => { return allowed ? originalWrite.call(this, data) : Promise.reject(new Error('用户拒绝复制')); }); }; } document.addEventListener('copy', (e) => { const now = Date.now(), isRecentUserAction = now - lastUserClickTime < 1000; recordCopy('copy-event'); if (hasUserAction || isRecentUserAction) return; e.preventDefault(); e.stopImmediatePropagation(); askForCopyPermission().then((allowed) => { if (allowed && e.clipboardData) { const selection = window.getSelection().toString(); if (selection) e.clipboardData.setData('text/plain', selection); } }); }, { capture: true }); }; const askForCopyPermission = () => { return new Promise((resolve) => { if (document.querySelector('.copy-permission-dialog')) { resolve(false); return; } const modal = document.createElement('div'); modal.className = 'copy-permission-dialog'; modal.style.cssText = `position:fixed;top:0;left:0;width:100%;height:100%;background:rgba(0,0,0,0.5);display:flex;align-items:center;justify-content:center;z-index:2147483647;font-family:-apple-system,BlinkMacSystemFont,"Segoe UI",Roboto,sans-serif;`; const dialog = document.createElement('div'); dialog.style.cssText = `background:white;padding:20px;border-radius:12px;max-width:80%;text-align:center;box-shadow:0 4px 20px rgba(0,0,0,0.15);`; const title = document.createElement('h3'); title.textContent = '复制确认'; title.style.margin = '0 0 10px 0'; const message = document.createElement('p'); message.textContent = '检测到自动复制操作,是否允许?'; message.style.margin = '0 0 20px 0'; const buttonContainer = document.createElement('div'); buttonContainer.style.cssText = 'display:flex;gap:10px;justify-content:center;'; const allowButton = document.createElement('button'); allowButton.textContent = '允许'; allowButton.style.cssText = `background:#34C759;color:white;border:none;padding:10px 20px;border-radius:8px;cursor:pointer;`; const denyButton = document.createElement('button'); denyButton.textContent = '拒绝'; denyButton.style.cssText = `background:#FF3B30;color:white;border:none;padding:10px 20px;border-radius:8px;cursor:pointer;`; const cleanup = () => { modal.remove(); setTimeout(() => { hasUserAction = false; }, 100); }; allowButton.onclick = () => { cleanup(); resolve(true); }; denyButton.onclick = () => { cleanup(); resolve(false); }; buttonContainer.append(denyButton, allowButton); dialog.append(title, message, buttonContainer); modal.appendChild(dialog); document.body.appendChild(modal); }); }; const init = () => { setupUserActionDetection(); monitorCopyMethods(); console.log('复制行为监控器已启动'); }; if (document.readyState === 'loading') document.addEventListener('DOMContentLoaded', init); else init(); })();