Greasy Fork

来自缓存

Greasy Fork is available in English.

网站访问确认脚本

限制指定主域名及其所有子域名的访问,显示确认页面(无刷新),支持30分钟、今日内和本次会话不再提示,受限列表存储在 GM_Value 中,已确认页面显示倒计时,支持通过菜单添加当前域名到限制列表

// ==UserScript==
// @name         网站访问确认脚本
// @namespace    https://github.com/liucong2013/userscript-site-access-check
// @version      1.8
// @icon         
// @description  限制指定主域名及其所有子域名的访问,显示确认页面(无刷新),支持30分钟、今日内和本次会话不再提示,受限列表存储在 GM_Value 中,已确认页面显示倒计时,支持通过菜单添加当前域名到限制列表
// @author       lc cong
// @match        *://*/*
// @grant        GM_setValue
// @grant        GM_getValue
// @grant        GM_deleteValue
// @grant        GM_addStyle
// @grant        GM_registerMenuCommand
// @run-at       document_start
// @noframes
// @supportURL   https://raw.githubusercontent.com/liucong2013/userscript-site-access-check/refs/heads/main/README.md
// ==/UserScript==

(function() {
    'use strict';

    if (window.top !== window.self) {
        console.log("脚本在 iframe 中运行,退出。");
        return;
    }

    // --- 配置 ---
    const RESTRICTED_DOMAINS_KEY = 'my_restricted_domains';
    const LOCAL_CONFIRM_KEY_PREFIX = 'confirmed_access_';

    // --- 工具函数 ---

    function getRestrictedBaseDomains() {
        const domainsJson = GM_getValue(RESTRICTED_DOMAINS_KEY, '[]');
        try {
            const domains = JSON.parse(domainsJson);
            return Array.isArray(domains) ? domains : [];
        } catch (e) {
            console.error("解析受限域名列表失败:", e);
            GM_deleteValue(RESTRICTED_DOMAINS_KEY);
            return [];
        }
    }

    function setRestrictedBaseDomains(domainsArray) {
        GM_setValue(RESTRICTED_DOMAINS_KEY, JSON.stringify(domainsArray));
    }

    function getBaseDomain(hostname) {
        if (!hostname) return '';
        const parts = hostname.split('.');
        if (parts.length <= 2) return hostname;
        const secondLast = parts[parts.length - 2];
        const commonTldParts = ['co', 'com', 'org', 'net', 'gov', 'edu', 'ac'];
        if (commonTldParts.includes(secondLast) && parts.length >= 3) {
            return parts.slice(-3).join('.');
        }
        return parts.slice(-2).join('.');
    }

    function findMatchingRestrictedDomain(hostname, restrictedDomains) {
        return restrictedDomains.find(baseDomain =>
            hostname === baseDomain || hostname.endsWith('.' + baseDomain)
        );
    }

    function getEndOfTodayTimestamp() {
        const now = new Date();
        return new Date(now.getFullYear(), now.getMonth(), now.getDate(), 23, 59, 59, 999).getTime();
    }

    function isLocalConfirmedAndNotExpired(domainKey) {
        const storedData = GM_getValue(LOCAL_CONFIRM_KEY_PREFIX + domainKey, null);
        if (!storedData) return false;
        try {
            const { timestamp, expiryType } = JSON.parse(storedData);
            const now = Date.now();
            if (expiryType === '30min') return now < timestamp + 30 * 60 * 1000;
            if (expiryType === '5min') return now < timestamp + 5 * 60 * 1000;
            if (expiryType === 'today') {
                const todayStart = new Date().setHours(0, 0, 0, 0);
                return timestamp >= todayStart && now < getEndOfTodayTimestamp();
            }
            return false;
        } catch (e) {
            console.error("解析确认状态失败:", e);
            GM_deleteValue(LOCAL_CONFIRM_KEY_PREFIX + domainKey);
            return false;
        }
    }

    function setLocalConfirmed(domainKey, expiryType) {
        const confirmInfo = { timestamp: Date.now(), expiryType };
        GM_setValue(LOCAL_CONFIRM_KEY_PREFIX + domainKey, JSON.stringify(confirmInfo));
        return confirmInfo;
    }

    function showToast(message) {
        const toast = document.createElement('div');
        toast.textContent = message;
        GM_addStyle(`
            #gm-toast { position: fixed; top: 20px; left: 50%; transform: translateX(-50%); background-color: #333; color: white; padding: 10px 20px; border-radius: 5px; z-index: 999999; font-size: 14px; }
        `);
        toast.id = 'gm-toast';
        document.body.appendChild(toast);
        setTimeout(() => toast.remove(), 3000);
    }

    function showRestrictionOverlay(hostname, domainToConfirm) {
        const overlay = document.createElement('div');
        overlay.id = 'restriction-overlay';

        GM_addStyle(`
            #restriction-overlay { position: fixed; top: 0; left: 0; width: 100%; height: 100%; background-color: rgba(0, 0, 0, 0.7); z-index: 9999998; display: flex; justify-content: center; align-items: center; font-family: sans-serif; }
            .restriction-container { background-color: #fff; padding: 30px; border-radius: 8px; box-shadow: 0 2px 10px rgba(0, 0, 0, 0.1); text-align: center; max-width: 500px; }
            .restriction-container h1 { color: #d9534f; margin-bottom: 20px; }
            .restriction-container p { color: #555; margin-bottom: 30px; line-height: 1.6; }
            .button-container button { background-color: #5cb85c; color: white; border: none; padding: 10px 20px; border-radius: 5px; cursor: pointer; font-size: 15px; margin: 5px; transition: background-color 0.3s ease; }
            .button-container button:hover { background-color: #4cae4c; }
        `);

        overlay.innerHTML = `
            <div class="restriction-container">
                <h1>⚠️ 访问受限</h1>
                <p>您正尝试访问的网站 <strong>${hostname}</strong> 已被标记为受限。请确认您希望继续访问,并选择本次确认的有效时长。</p>
                <div class="button-container">
                    <button data-expiry="30min">30分钟内不再提示</button>
                    <button data-expiry="today">今天内不再提示</button>
                    <button data-expiry="5min">允许访问5分钟</button>
                </div>
            </div>
        `;

        overlay.addEventListener('click', (e) => {
            if (e.target.tagName === 'BUTTON') {
                const expiryType = e.target.dataset.expiry;
                const confirmInfo = setLocalConfirmed(domainToConfirm, expiryType);
                overlay.remove();
                showCountdown(hostname, confirmInfo.expiryType, confirmInfo.timestamp);
            }
        });

        // 等待DOM加载完成再插入
        if (document.body) {
            document.body.appendChild(overlay);
        } else {
            document.addEventListener('DOMContentLoaded', () => document.body.appendChild(overlay));
        }
    }

    function showCountdown(hostname, expiryType, timestamp) {
        if (!document.body) {
            window.addEventListener('DOMContentLoaded', () => showCountdown(hostname, expiryType, timestamp));
            return;
        }
        GM_addStyle(`
            #restriction-countdown { position: fixed; bottom: 20px; right: 10px; background-color: rgba(255, 255, 255, 0.9); border: 1px solid #ccc; padding: 5px 10px; border-radius: 4px; font-size: 12px; z-index: 9999999; box-shadow: 0 1px 4px rgba(0,0,0,0.1); color: black; cursor: pointer; }
        `);
        const countdownDiv = document.createElement('div');
        countdownDiv.id = 'restriction-countdown';
        document.body.appendChild(countdownDiv);

        let intervalId = setInterval(updateCountdown, 1000);
        countdownDiv.addEventListener('dblclick', () => {
            clearInterval(intervalId);
            countdownDiv.remove();
        });

        function formatRemainingTime(ms) {
            if (ms <= 0) return '0s';
            const totalSeconds = Math.floor(ms / 1000);
            const hours = Math.floor(totalSeconds / 3600);
            const minutes = Math.floor((totalSeconds % 3600) / 60);
            const seconds = totalSeconds % 60;
            let parts = [];
            if (hours > 0) parts.push(`${hours}h`);
            if (minutes > 0) parts.push(`${minutes}m`);
            if (seconds > 0 || parts.length === 0) parts.push(`${seconds}s`);
            return parts.join(' ');
        }

        function updateCountdown() {
            const now = Date.now();
            let remainingTime, expiryLabel;

            if (expiryType === '30min') {
                remainingTime = (timestamp + 30 * 60 * 1000) - now;
                expiryLabel = '剩余';
            } else if (expiryType === '5min') {
                remainingTime = (timestamp + 5 * 60 * 1000) - now;
                expiryLabel = '剩余';
            } else if (expiryType === 'today') {
                remainingTime = getEndOfTodayTimestamp() - now;
                expiryLabel = '今日剩余';
            }

            if (remainingTime <= 0) {
                countdownDiv.textContent = `❌ 确认已过期`;
                clearInterval(intervalId);
                return;
            }
            countdownDiv.textContent = `⏳ ${expiryLabel}: ${formatRemainingTime(remainingTime)}`;
        }
        updateCountdown();
    }

    function addCurrentDomainToRestrictedList() {
        const currentHostname = window.location.hostname;
        if (!currentHostname || currentHostname === 'localhost') {
            showToast("无法获取当前域名或不支持本地地址。");
            return;
        }
        const baseDomain = getBaseDomain(currentHostname);
        const restrictedDomains = getRestrictedBaseDomains();
        if (restrictedDomains.includes(baseDomain)) {
            showToast(`主域名 "${baseDomain}" 已在限制列表中。`);
            return;
        }
        restrictedDomains.push(baseDomain);
        setRestrictedBaseDomains(restrictedDomains);
        showToast(`主域名 "${baseDomain}" 已添加到限制列表。`);
        // 动态更新菜单
        registerMenus(baseDomain);
        // 显示限制(如果当前页面就是刚添加的)
        if (!document.getElementById('restriction-overlay')) {
             showRestrictionOverlay(currentHostname, baseDomain);
        }
    }

    function registerMenus(matchingDomain) {
        // 清除旧菜单,防止重复
        if (window.registeredMenuCommands) {
            window.registeredMenuCommands.forEach(GM_unregisterMenuCommand);
        }
        window.registeredMenuCommands = [];

        let cmd1 = GM_registerMenuCommand("➕ 添加当前主域名到限制列表", addCurrentDomainToRestrictedList);
        window.registeredMenuCommands.push(cmd1);

        if (matchingDomain) {
            let cmd2 = GM_registerMenuCommand(`🗑️ 将主域名 "${matchingDomain}" 从限制列表移除`, () => {
                const currentRestricted = getRestrictedBaseDomains();
                const idx = currentRestricted.indexOf(matchingDomain);
                if (idx !== -1) {
                    currentRestricted.splice(idx, 1);
                    setRestrictedBaseDomains(currentRestricted);
                    showToast(`主域名 "${matchingDomain}" 已从限制列表移除。`);
                    // 动态更新菜单
                    registerMenus(null);
                } else {
                    showToast("错误:在列表中找不到匹配的域名。");
                }
            });
            window.registeredMenuCommands.push(cmd2);
        }
    }


    // --- 主逻辑 ---
    const currentHostname = window.location.hostname;
    const restrictedDomains = getRestrictedBaseDomains();
    const matchingDomain = findMatchingRestrictedDomain(currentHostname, restrictedDomains);

    // 始终注册菜单,以便动态更新
    registerMenus(matchingDomain);

    if (matchingDomain) {
        const confirmationKey = matchingDomain;
        if (!isLocalConfirmedAndNotExpired(confirmationKey)) {
            console.log(`访问 ${currentHostname} (规则: ${matchingDomain}) 受限,显示确认页面。`);
            // 使用 run-at: document_start 时,需要等待DOM ready
            if (document.readyState === 'loading') {
                document.addEventListener('DOMContentLoaded', () => showRestrictionOverlay(currentHostname, confirmationKey));
            } else {
                showRestrictionOverlay(currentHostname, confirmationKey);
            }
        } else {
            console.log(`访问 ${currentHostname} (规则: ${matchingDomain}) 已放行。`);
            const localConfirmedData = GM_getValue(LOCAL_CONFIRM_KEY_PREFIX + confirmationKey, null);
            if (localConfirmedData) {
                try {
                    const { expiryType, timestamp } = JSON.parse(localConfirmedData);
                    showCountdown(currentHostname, expiryType, timestamp);
                } catch(e) { /* ignore */ }
            }
        }
    }
})();