网站访问确认脚本

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

您需要先安裝使用者腳本管理器擴展,如 TampermonkeyGreasemonkeyViolentmonkey 之後才能安裝該腳本。

You will need to install an extension such as Tampermonkey to install this script.

您需要先安裝使用者腳本管理器擴充功能,如 TampermonkeyViolentmonkey 後才能安裝該腳本。

您需要先安裝使用者腳本管理器擴充功能,如 TampermonkeyUserscripts 後才能安裝該腳本。

你需要先安裝一款使用者腳本管理器擴展,比如 Tampermonkey,才能安裝此腳本

您需要先安裝使用者腳本管理器擴充功能後才能安裝該腳本。

(我已經安裝了使用者腳本管理器,讓我安裝!)

你需要先安裝一款使用者樣式管理器擴展,比如 Stylus,才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展,比如 Stylus,才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展,比如 Stylus,才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展後才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展後才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展後才能安裝此樣式

(我已經安裝了使用者樣式管理器,讓我安裝!)

// ==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 */ }
            }
        }
    }
})();