Greasy Fork

来自缓存

Greasy Fork is available in English.

98助手

98tang 隐藏已访问链接,支持拖拽UI、隐藏/透明度/显示切换

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Greasemonkey 油猴子Violentmonkey 暴力猴,才能安装此脚本。

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

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

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

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

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

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

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

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

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

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

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

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

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

// ==UserScript==
// @name         98助手
// @namespace    duang_duang
// @version      2.0.2
// @description  98tang 隐藏已访问链接,支持拖拽UI、隐藏/透明度/显示切换
// @author       q
// @match        *://*/*
// @grant        GM_setValue
// @grant        GM_getValue
// @license MIT
// @icon         
// ==/UserScript==

(function () {
    'use strict';

    // 配置
    const DB_NAME = '98tang_visited_db';
    const STORE_NAME = 'visited_links';
    const DB_VERSION = 1;
    const STORAGE_KEY_POS = '98tang_ui_pos';
    const STORAGE_KEY_MODE = '98tang_ui_mode'; // 'opacity', 'hidden', 'show'
    const STORAGE_KEY_MINIMIZED = '98tang_ui_minimized';

    // UI 相关变量
    const panelId = 'visited-link-panel';
    const visitedClass = 'visited-item';
    let currentMode = localStorage.getItem(STORAGE_KEY_MODE) || 'opacity';
    let hiddenCount = 0;

    // IndexDB 工具类
    const dbTools = {
        db: null,
        init: function () {
            return new Promise((resolve, reject) => {
                const request = indexedDB.open(DB_NAME, DB_VERSION);

                request.onerror = (event) => {
                    console.error("Database error: " + event.target.errorCode);
                    reject(event);
                };

                request.onupgradeneeded = (event) => {
                    const db = event.target.result;
                    if (!db.objectStoreNames.contains(STORE_NAME)) {
                        db.createObjectStore(STORE_NAME, { keyPath: "id" });
                    }
                };

                request.onsuccess = (event) => {
                    this.db = event.target.result;
                    resolve(this.db);
                };
            });
        },
        add: function (id) {
            return new Promise((resolve, reject) => {
                if (!this.db) return reject("DB not initialized");
                const transaction = this.db.transaction([STORE_NAME], "readwrite");
                const store = transaction.objectStore(STORE_NAME);
                const request = store.put({ id: id, timestamp: new Date().getTime() });

                request.onsuccess = () => resolve(true);
                request.onerror = () => resolve(false); // 忽略错误
            });
        },
        has: function (id) {
            return new Promise((resolve, reject) => {
                if (!this.db) return reject("DB not initialized");
                const transaction = this.db.transaction([STORE_NAME], "readonly");
                const store = transaction.objectStore(STORE_NAME);
                const request = store.get(id);

                request.onsuccess = () => resolve(!!request.result);
                request.onerror = () => resolve(false);
            });
        }
    };

    // UI 工具类
    const uiTools = {
        updateCount: function () {
            const el = document.getElementById('visited-count');
            if (el) el.innerText = hiddenCount;
        },
        applyMode: function (mode) {
            currentMode = mode;
            localStorage.setItem(STORAGE_KEY_MODE, mode);
            
            // 移除所有旧模式类
            const classes = document.body.classList;
            for (let i = classes.length - 1; i >= 0; i--) {
                if (classes[i].startsWith('mode-')) {
                    classes.remove(classes[i]);
                }
            }
            document.body.classList.add('mode-' + mode);
        },
        initStyle: function () {
            const style = document.createElement('style');
            style.innerHTML = `
                /* 模式样式 */
                body.mode-hidden .${visitedClass} { display: none !important; }
                body.mode-opacity .${visitedClass} { opacity: 0.3; }
                body.mode-show .${visitedClass} { /* 正常显示 */ }
                body.mode-strikethrough .${visitedClass} a { text-decoration: line-through !important; }
                body.mode-opacity-strike .${visitedClass} { opacity: 0.4; }
                body.mode-opacity-strike .${visitedClass} a { text-decoration: line-through !important; }
                body.mode-yellow .${visitedClass} a { color: #aaaa00 !important; }

                /* 面板样式 */
                #${panelId} {
                    position: fixed; 
                    z-index: 999999;
                    background: rgba(0, 0, 0, 0.75);
                    color: white;
                    padding: 6px;
                    border-radius: 4px;
                    font-size: 12px;
                    font-family: sans-serif;
                    box-shadow: 0 2px 8px rgba(0,0,0,0.3);
                    user-select: none;
                    width: 110px;
                    backdrop-filter: blur(2px);
                    transition: width 0.2s, background 0.2s;
                }
                #${panelId}.minimized {
                    width: auto;
                    padding: 4px 8px;
                    background: rgba(0, 0, 0, 0.5);
                }
                #${panelId}.minimized .panel-content {
                    display: none;
                }
                #${panelId} .panel-header {
                    cursor: move;
                    font-weight: bold;
                    display: flex;
                    justify-content: space-between;
                    align-items: center;
                }
                #${panelId}:not(.minimized) .panel-header {
                    border-bottom: 1px solid rgba(255,255,255,0.2);
                    padding-bottom: 4px;
                    margin-bottom: 4px;
                }
                #${panelId} .toggle-btn {
                    cursor: pointer;
                    font-size: 14px;
                    line-height: 1;
                    opacity: 0.7;
                    padding: 0 2px;
                }
                #${panelId} .toggle-btn:hover {
                    opacity: 1;
                    color: #4CAF50;
                }
                #${panelId} select {
                    background: #333;
                    color: white;
                    border: 1px solid #555;
                    border-radius: 3px;
                    padding: 2px;
                    width: 100%;
                    font-size: 11px;
                    margin-bottom: 2px;
                    cursor: pointer;
                }
                #${panelId} .stat-row {
                    font-size: 10px;
                    color: #ccc;
                    text-align: right;
                }
            `;
            document.head.appendChild(style);
        },
        initPanel: function () {
            if (document.getElementById(panelId)) return;

            const div = document.createElement('div');
            div.id = panelId;
            
            // 恢复位置
            const savedPos = JSON.parse(localStorage.getItem(STORAGE_KEY_POS) || '{"top": "20px", "left": "20px"}');
            div.style.top = savedPos.top;
            div.style.left = savedPos.left;

            // 恢复折叠状态
            const isMinimized = localStorage.getItem(STORAGE_KEY_MINIMIZED) === 'true';
            if (isMinimized) div.classList.add('minimized');

            div.innerHTML = `
                <div class="panel-header" title="双击折叠/展开">
                    <span>98助手</span>
                    <span class="toggle-btn" title="折叠/展开">${isMinimized ? '+' : '-'}</span>
                </div>
                <div class="panel-content">
                    <div>
                        <select id="mode-select">
                            <option value="opacity">透明</option>
                            <option value="hidden">隐藏</option>
                            <option value="strikethrough">删除线</option>
                            <option value="opacity-strike">透明+线</option>
                            <option value="yellow">变黄</option>
                            <option value="show">显示</option>
                        </select>
                    </div>
                    <div class="stat-row">
                        已阅: <span id="visited-count">0</span>
                    </div>
                </div>
            `;
            document.body.appendChild(div);

            // 绑定事件
            const select = div.querySelector('#mode-select');
            select.value = currentMode;
            select.addEventListener('change', (e) => {
                this.applyMode(e.target.value);
            });
            
            // 折叠逻辑
            const toggleBtn = div.querySelector('.toggle-btn');
            const header = div.querySelector('.panel-header');
            
            const toggleMin = () => {
                div.classList.toggle('minimized');
                const isMin = div.classList.contains('minimized');
                toggleBtn.innerText = isMin ? '+' : '-';
                localStorage.setItem(STORAGE_KEY_MINIMIZED, isMin);
            };

            toggleBtn.addEventListener('click', (e) => {
                e.stopPropagation(); // 防止触发拖拽
                toggleMin();
            });
            
            header.addEventListener('dblclick', toggleMin);

            // 初始化当前模式
            this.applyMode(currentMode);
            this.makeDraggable(div);
        },
        makeDraggable: function (element) {
            const header = element.querySelector('.panel-header');
            let isDragging = false;
            let startX, startY, initialLeft, initialTop;

            header.addEventListener('mousedown', (e) => {
                if (e.target.classList.contains('toggle-btn')) return;
                
                isDragging = true;
                startX = e.clientX;
                startY = e.clientY;
                
                const rect = element.getBoundingClientRect();
                initialLeft = rect.left;
                initialTop = rect.top;
                
                header.style.cursor = 'grabbing';
            });

            document.addEventListener('mousemove', (e) => {
                if (!isDragging) return;
                
                const dx = e.clientX - startX;
                const dy = e.clientY - startY;
                
                const newLeft = initialLeft + dx;
                const newTop = initialTop + dy;

                element.style.left = `${newLeft}px`;
                element.style.top = `${newTop}px`;
            });

            document.addEventListener('mouseup', () => {
                if (isDragging) {
                    isDragging = false;
                    header.style.cursor = 'move';
                    // 保存位置
                    localStorage.setItem(STORAGE_KEY_POS, JSON.stringify({
                        top: element.style.top,
                        left: element.style.left
                    }));
                }
            });
        }
    };

    // 业务逻辑
    const core = {
        getIdByUrl: (url, key, regex) => {
            try {
                let uri = new URL(url);
                if (uri.href.indexOf(key) > -1) {
                    const match = uri.href.match(regex);
                    if (match) {
                        return match[1];
                    }
                }
            } catch (e) {
                // 忽略无效URL
            }
            return null;
        },
        getId: function (url) {
            if (!url) {
                url = document.URL;
            }
            var id = this.getIdByUrl(url, "tid=", /tid=(\d+)/);
            return id;
        },
        // 隐藏列表中的已读项
        hideVisited: async function () {
            // 如果是详情页,不执行隐藏,但要记录ID
            if (document.URL.indexOf("mod=viewthread") > -1 || document.URL.indexOf("tid=") > -1) {
                let id = this.getId(document.URL);
                if (id) {
                    await dbTools.add(id);
                    console.log("已记录当前页面 ID:", id);
                }
                return;
            }

            // 列表页才初始化完整面板
            uiTools.initStyle();
            uiTools.initPanel();

            // 列表页处理逻辑
            if (document.URL.indexOf("forum.php") > -1) {
                // 首页/板块列表
                const rows = document.querySelectorAll("table tr, #waterfall li");
                
                for (let item of rows) {
                    let a = item.querySelector("a.xst") || item.querySelector("a.z") || (item.querySelectorAll("a")[1]);
                    
                    // 排除一些非帖子链接
                    if (!a || a.href.indexOf("tid=") === -1) continue;

                    let id = this.getId(a.href);
                    if (id) {
                        // 绑定点击事件,点击即记录
                        a.addEventListener("mousedown", () => {
                            dbTools.add(id);
                            item.classList.add(visitedClass); // 立即标记
                            hiddenCount++;
                            uiTools.updateCount();
                        });

                        // 检查是否已读
                        const hasVisited = await dbTools.has(id);
                        if (hasVisited) {
                            item.classList.add(visitedClass);
                            hiddenCount++;
                        }
                    }
                }
                uiTools.updateCount();
            } 
        },
        start: async function () {
            await dbTools.init();
            await this.hideVisited();
            console.log("98tang 隐藏插件(带UI版)启动完成");
        }
    };

    // 启动
    setTimeout(() => {
        // 检查标题是否包含“98堂”
        if (document.title.indexOf("98堂") === -1) {
             console.log("标题不包含98堂,插件不运行");
             return; 
        }
        core.start();
    }, 500);

})();