复制授权拦截器

复制必弹窗,2次允许后开始自由复制,3次拒绝永久禁用(变灰,点击灰点恢复变红),点击小红点恢复,红绿切换自由复制。

您需要先安装一个扩展,例如 篡改猴Greasemonkey暴力猴,之后才能安装此脚本。

您需要先安装一个扩展,例如 篡改猴暴力猴,之后才能安装此脚本。

您需要先安装一个扩展,例如 篡改猴暴力猴,之后才能安装此脚本。

您需要先安装一个扩展,例如 篡改猴Userscripts ,之后才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。

您需要先安装用户脚本管理器扩展后才能安装此脚本。

(我已经安装了用户脚本管理器,让我安装!)

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

(我已经安装了用户样式管理器,让我安装!)

// ==UserScript==
// @name         复制授权拦截器
// @namespace    https://viayoo.com/
// @version      3.28
// @description  复制必弹窗,2次允许后开始自由复制,3次拒绝永久禁用(变灰,点击灰点恢复变红),点击小红点恢复,红绿切换自由复制。
// @author       Aloazny & Deepseek
// @match        *://*/*
// @run-at       document-start
// @icon       
// @grant        none
// @license      MIT
// ==/UserScript==

(function () {
    const KEY = 'copyAuth:v3'; if (window[KEY]) return; window[KEY] = true;
    let denyCount = 0, allowCount = 0, enabled = true, freeCopyMode = false, isShowingModal = false;
    const MAX_DENY = 3, AUTO_FREE_AFTER_ALLOW = 2, COPY_TIME_THRESHOLD = 1000, COPY_COUNT_THRESHOLD = 3;
    let copyRecords = [], dot = null, hideTimer = null;
    const originalExec = document.execCommand;
    const origWriteText = navigator.clipboard?.writeText;
    const origWrite = navigator.clipboard?.write;

    const disableAllCopyAPIs = () => {
        const emptyFunc = () => false;
        const emptyPromise = () => Promise.reject('复制功能已被永久禁用');

        Object.defineProperty(document, 'execCommand', {
            value: emptyFunc,
            writable: false,
            configurable: false
        });

        if (navigator.clipboard) {
            Object.defineProperty(navigator.clipboard, 'writeText', {
                value: emptyPromise,
                writable: false,
                configurable: false
            });
            if (navigator.clipboard.write) {
                Object.defineProperty(navigator.clipboard, 'write', {
                    value: emptyPromise,
                    writable: false,
                    configurable: false
                });
            }
        }

        setInterval(() => {
            document.execCommand = emptyFunc;
            if (navigator.clipboard) {
                navigator.clipboard.writeText = emptyPromise;
                navigator.clipboard.write = emptyPromise;
            }
        }, 800);
    };

    const showToast = (msg) => {
        if (window.via && typeof window.via.toast === 'function') {
            window.via.toast(msg);
        } else {
            const t = document.createElement('div');
            t.textContent = msg;
            t.style.cssText = 'position:fixed;top:50%;left:50%;transform:translate(-50%,-50%);background:rgba(0,0,0,.8);color:white;padding:12px 20px;border-radius:8px;z-index:2147483647;font-size:14px;';
            document.body.appendChild(t);
            setTimeout(() => t.remove(), 2000);
        }
    };

    const showConfirm = (msg, txt = '') => {
        if (isShowingModal) return Promise.resolve(false);
        return new Promise(r => { isShowingModal = true;
            const modal = document.createElement('div'), dialog = document.createElement('div'), title = document.createElement('h3'), pre = document.createElement('div'), btns = document.createElement('div'), allow = document.createElement('button'), deny = document.createElement('button');
            modal.style.cssText = 'position:fixed;top:0;left:0;width:100%;height:100%;background:rgba(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;backdrop-filter:blur(8px);padding:16px;box-sizing:border-box;';
            dialog.style.cssText = 'background:rgba(255,255,255,.94);border-radius:18px;width:88vw;max-width:420px;text-align:center;box-shadow:0 12px 40px rgba(0,0,0,.25);display:flex;flex-direction:column;gap:16px;border:1px solid rgba(0,0,0,.06);overflow:hidden;';
            title.style.cssText = 'margin:0;padding:0 20px;font-size:18px;font-weight:700;color:#111;line-height:1.3;';
            pre.style.cssText = 'margin:0 20px;padding:14px;background:#f8f8f8;border-radius:12px;font-family:monospace;font-size:13px;color:#111;max-height:140px;overflow:auto;word-break:break-all;display:' + (txt?'block':'none') + ';line-height:1.4;';
            btns.style.cssText = 'display:flex;gap:12px;margin:0 20px;padding-bottom:8px;';
            allow.style.cssText = 'flex:1;height:48px;border:none;border-radius:14px;background:#007AFF;color:#fff;font-size:16px;font-weight:600;cursor:pointer;box-shadow:0 4px 14px rgba(0,122,255,.35);transition:transform .12s;';
            deny.style.cssText = 'flex:1;height:48px;border:none;border-radius:14px;background:#FF3B30;color:#fff;font-size:16px;font-weight:600;cursor:pointer;box-shadow:0 4px 14px rgba(255,59,48,.35);transition:transform .12s;';
            title.textContent = msg; pre.textContent = txt.slice(0,200)+(txt.length>200?'...':''); allow.textContent='允许'; deny.textContent='拒绝';
            allow.onclick = () => { clean(); r(true); };
            deny.onclick = () => { clean(); r(false); };
            modal.onclick = e => e.target===modal && deny.onclick();
            const clean = () => { modal.remove(); isShowingModal = false; };
            ['touchstart','mousedown'].forEach(ev => {
                allow.addEventListener(ev, () => allow.style.transform = 'translateY(1px)', {passive:true});
                deny.addEventListener(ev, () => deny.style.transform = 'translateY(1px)', {passive:true});
                document.addEventListener(ev==='touchstart'?'touchend':'mouseup', () => { allow.style.transform = deny.style.transform = ''; }, {once:true, passive:true});
            });
            dialog.append(title, pre, btns); btns.append(deny, allow); modal.appendChild(dialog); document.body.appendChild(modal); allow.focus();
        });
    };

    const updateDot = () => { if (!dot) return; dot.style.backgroundColor = freeCopyMode ? '#34C759' : (!enabled ? '#8E8E93' : '#FF3B30'); };

    const showDot = () => {
        if (dot) { clearTimeout(hideTimer); return; }
        dot = document.createElement('div');
        dot.style.cssText = 'position:fixed;bottom:45%;right:10px;width:20px;height:20px;border-radius:10px;background:#FF3B30;z-index:2147483647;opacity:0;transform:scale(0);border:2px solid #fff;box-shadow:0 2px 8px rgba(0,0,0,.3);transition:all .3s cubic-bezier(0.34,1.56,0.64,1);cursor:pointer;touch-action:none;';
        document.body.appendChild(dot);
        requestAnimationFrame(() => { dot.style.opacity = '.8'; dot.style.transform = 'scale(1)'; });
        updateDot();
        let taps = 0, timer = null;
        const handleTap = () => {
            taps++; clearTimeout(timer);
            timer = setTimeout(() => {
                if (taps === 1) {
                    if (enabled && !freeCopyMode) { freeCopyMode = true; enabled = true; updateDot(); }
                    else if (freeCopyMode) { freeCopyMode = false; updateDot(); }
                    else { enabled = true; denyCount = 0; updateDot(); }
                } else if (taps === 2) { freeCopyMode = enabled = true; denyCount = 0; updateDot(); }
                taps = 0;
            }, 300);
        };
        dot.onclick = handleTap; dot.ontouchstart = e => { e.preventDefault(); handleTap(); };
    };

    const tryAutoFreeOnAllow = () => { allowCount++; if (allowCount >= AUTO_FREE_AFTER_ALLOW && !freeCopyMode) { freeCopyMode = true; enabled = true; if (dot) updateDot(); } };

    const hideDotIfNeeded = () => {
        if (!dot) return; clearTimeout(hideTimer);
        if (freeCopyMode || !enabled) return;
        hideTimer = setTimeout(() => { if (dot) { dot.remove(); dot = null; } }, 3000);
    };

    const recordCopy = (method) => {
        const now = Date.now();
        copyRecords.push({ timestamp: now, method });

        copyRecords = copyRecords.filter(r => now - r.timestamp <= COPY_TIME_THRESHOLD);

        if (copyRecords.length >= COPY_COUNT_THRESHOLD) {
            enabled = false;
            updateDot();
            copyRecords = [];
            disableAllCopyAPIs();
            showToast('检测到恶意复制行为,复制功能已被永久禁用');
            return false;
        }
        return true;
    };

    const hookAPI = () => {
        if (origWriteText) navigator.clipboard.writeText = txt => new Promise((res, rej) => {
            if (!recordCopy('writeText')) return rej(new Error('恶意复制已被阻止'));
            if (freeCopyMode) return origWriteText.call(navigator.clipboard, txt).then(res).catch(rej);
            if (!enabled || denyCount >= MAX_DENY) return rej();
            showDot();
            showConfirm('允许复制内容?', txt).then(ok => {
                if (ok) { origWriteText.call(navigator.clipboard, txt).then(res).catch(rej); tryAutoFreeOnAllow(); }
                else { denyCount++; if (denyCount >= MAX_DENY) { enabled = false; updateDot(); disableAllCopyAPIs(); } rej(new Error('用户拒绝复制')); }
                hideDotIfNeeded();
            });
        });

        if (origWrite) navigator.clipboard.write = data => new Promise((res, rej) => {
            if (!recordCopy('write')) return rej(new Error('恶意复制已被阻止'));
            if (freeCopyMode) return origWrite.call(navigator.clipboard, data).then(res).catch(rej);
            if (!enabled || denyCount >= MAX_DENY) return rej();
            const txt = data[0]?.type === 'text/plain' ? data[0].getData('text/plain') : '';
            showDot();
            showConfirm('允许复制内容?', txt).then(ok => {
                if (ok) { origWrite.call(navigator.clipboard, data).then(res).catch(rej); tryAutoFreeOnAllow(); }
                else { denyCount++; if (denyCount >= MAX_DENY) { enabled = false; updateDot(); disableAllCopyAPIs(); } rej(new Error('用户拒绝复制')); }
                hideDotIfNeeded();
            });
        });
    };

    const hookExec = () => {
        document.execCommand = (cmd, ui, val) => {
            if (cmd === 'copy') {
                if (!recordCopy('execCommand')) return false;
                if (freeCopyMode) return originalExec.call(document, cmd, ui, val);
                if (!enabled || denyCount >= MAX_DENY) return false;
                const sel = window.getSelection().toString();
                if (!sel) return originalExec.apply(document, arguments);
                showDot();
                showConfirm('允许复制内容?', sel).then(ok => {
                    if (ok) { 
                        const ta = document.createElement('textarea'); 
                        ta.value = sel; ta.style = 'position:fixed;opacity:0;'; 
                        document.body.appendChild(ta); ta.select(); 
                        originalExec.call(document, 'copy'); 
                        ta.remove(); 
                        tryAutoFreeOnAllow(); 
                    } else { 
                        denyCount++; 
                        if (denyCount >= MAX_DENY) { enabled = false; updateDot(); disableAllCopyAPIs(); } 
                    }
                    hideDotIfNeeded();
                });
                return false;
            }
            return originalExec.apply(document, arguments);
        };
    };

    const setupFrame = iframe => {
        try {
            const doc = iframe.contentDocument; if (!doc) return;
            const win = doc.defaultView; if (!win) return;
            const frameOrigExec = doc.execCommand;
            doc.execCommand = (cmd, ui, val) => {
                if (cmd === 'copy') {
                    if (!recordCopy('frameExecCommand')) return false;
                    if (freeCopyMode) return frameOrigExec.call(doc, cmd, ui, val);
                    if (!enabled || denyCount >= MAX_DENY) return false;
                    const sel = win.getSelection().toString();
                    if (!sel) return frameOrigExec.apply(doc, arguments);
                    showDot();
                    showConfirm('允许复制内容?', sel).then(ok => {
                        if (ok) {
                            const ta = doc.createElement('textarea');
                            ta.value = sel; ta.style = 'position:fixed;opacity:0;';
                            doc.body.appendChild(ta); ta.select();
                            frameOrigExec.call(doc, 'copy');
                            ta.remove();
                            tryAutoFreeOnAllow();
                        } else {
                            denyCount++;
                            if (denyCount >= MAX_DENY) { enabled = false; updateDot(); disableAllCopyAPIs(); }
                        }
                        hideDotIfNeeded();
                    });
                    return false;
                }
                return frameOrigExec.apply(doc, arguments);
            };
        } catch(e) {}
    };

    const obs = new MutationObserver(m => {
        for (const r of m) for (const n of r.addedNodes)
            if (n.nodeType === 1 && n.tagName === 'IFRAME') setupFrame(n);
    });
    obs.observe(document.documentElement, { childList: true, subtree: true });
    document.querySelectorAll('iframe').forEach(setupFrame);

    const init = () => { hookAPI(); hookExec(); };
    if (document.readyState === 'loading') {
        document.addEventListener('DOMContentLoaded', () => setTimeout(init, 0));
    } else {
        setTimeout(init, 0);
    }

    const prot = setInterval(() => {
        if (navigator.clipboard && navigator.clipboard.writeText === origWriteText) hookAPI();
        if (document.execCommand === originalExec) hookExec();
    }, 1000);
    window.addEventListener('beforeunload', () => clearInterval(prot));
})();