Greasy Fork

Greasy Fork is available in English.

⭐网页瞬间加载/跳过进度条直接加载网页⭐

任何链接内容跳过进度条秒加载(已经成功预加载的链接文本会显示成功交互),测试网址:https://www.xmwav.com/

当前为 2024-05-16 提交的版本,查看 最新版本

// ==UserScript==
// @name         ⭐网页瞬间加载/跳过进度条直接加载网页⭐
// @namespace    fenda
// @version      1.0.11
// @description  任何链接内容跳过进度条秒加载(已经成功预加载的链接文本会显示成功交互),测试网址:https://www.xmwav.com/
// @author       fenda
// @icon         
// @match        *://*/*
// @grant        GM_getValue
// @grant        GM_setValue
// @grant        GM_registerMenuCommand
// @run-at       document-end
// @license      MPL-2.0
// ==/UserScript==

(function () {
    'use strict';

    var indexedDB = window.indexedDB || window.mozIndexedDB || window.webkitIndexedDB || window.msIndexedDB;
    var dbVersion = 1;
    var dbName = 'preloadedPagesDB';
    var dbStoreName = 'preloadedPages';

    var maxConcurrentPreloads = GM_getValue('maxConcurrentPreloads', 15);
    var dataCleanupInterval = GM_getValue('dataCleanupInterval', 1) * 3600000;
    var maxContentSize = GM_getValue('maxContentSize', 5) * 1024 * 1024;
    var maxStorageItems = GM_getValue('maxStorageItems', 300);
    var visualLinkReminderStyle = GM_getValue('visualLinkReminderStyle', 'underline');
    var returnShortcutKey = GM_getValue('returnShortcutKey', 'Shift+R').toLowerCase();
    var directShortcutKey = GM_getValue('directShortcutKey', 'Shift+E').toLowerCase();
    var forwardShortcutKey = GM_getValue('forwardShortcutKey', 'Shift+F').toLowerCase();
    var isWhitelistModeEnabled = GM_getValue('whitelistMode', false);
    var isBlacklistModeEnabled = GM_getValue('blacklistMode', true);
    var defaultBlacklistDomains = [
        'www.bilibili.com',
        'www.bing.com',
        'www.huya.com',
        'www.vimeo.com',
        'www.tiktok.com',
        'www.twitch.tv',
        'www.youtube.com',
        'www.dailymotion.com',
        'www.liveleak.com',
        'www.metacafe.com',
        'www.youku.com',
        'www.iqiyi.com',
        'www.netflix.com',
        'www.hulu.com',
        'www.primevideo.com'
    ];

    if (typeof GM_getValue === 'function') {
        var storedBlacklistDomains = GM_getValue('blacklistDomains', []);
        var blacklistDomains = [...new Set([...storedBlacklistDomains, ...defaultBlacklistDomains])]; //将检索到的黑名单与默认黑名单合并,确保没有重复项
        GM_setValue('blacklistDomains', blacklistDomains); //保存更新后的黑名单

        var whitelistDomains = GM_getValue('whitelistDomains', []); // 假设以后会在某个地方使用它
    }


    var currentPreloads = 0;
    var preloadQueue = [];
    var preloadSet = new Set();
    var abortControllers = {};
    var db;
    var dbReady = false;
    var clickedLinks = [];
    var currentPreviewIndex = -1;
    var shouldPreloadMapping = {};
    var dbWriteQueue = [];
    var dbWriteInProgress = false;


    //版本更新展示
    function checkForUpdates() {
        var lastCheckedTime = GM_getValue('lastCheckedTime', 0);
        var currentTime = Date.now();

        if (currentTime - lastCheckedTime >= 3600000) {
            GM_setValue('lastCheckedTime', currentTime);
            var updateURL = "https://update.greasyfork.icu/scripts/493851/%E2%AD%90%E7%BD%91%E9%A1%B5%E7%9E%AC%E9%97%B4%E5%8A%A0%E8%BD%BD%E8%B7%B3%E8%BF%87%E8%BF%9B%E5%BA%A6%E6%9D%A1%E7%9B%B4%E6%8E%A5%E5%8A%A0%E8%BD%BD%E7%BD%91%E9%A1%B5%E2%AD%90.meta.js";

            fetch(updateURL).then(function (response) {
                response.text().then(function (text) {
                    var latestVersion = text.match(/@version\s+([^\n]+)/)[1];
                    if (latestVersion) {
                        GM_setValue('latestVersion', latestVersion);
                    }
                });
            }).catch(function (error) {
                console.error('An error occurred while checking for updates:', error);
            });
        }
    }
    checkForUpdates();


    var settingsPanelHTML = `
        <div id="userScriptSettingsPanel" style="
            width: 300px;
            height: auto;
            position: fixed;
            top: 50%;
            left: 50%;
            transform: translate(-50%, -50%);
            background: #f2f2f2;
            box-shadow: 0 4px 8px 0 rgba(0,0,0,0.4), 0 6px 20px 0 rgba(0,0,0,0.19);
            border-radius: 15px;
            padding: 20px;
            box-sizing: border-box;
            z-index: 10000;
            display: none;
            font-family: Arial, sans-serif;">
            <h2 style="
                text-align: center;
                font-size: 20px;
                font-weight: bold;
                color: #333;
                margin-bottom: 20px;
                text-shadow: 1px 1px 2px #888;">脚本设置</h2>
            <div style="margin-bottom: 10px; display: flex; align-items: center;">
                <label for="maxConcurrentPreloads" style="flex-grow: 1; margin-right: 10px;">最大并发预加载数:</label>
                <input type="number" id="maxConcurrentPreloads" value="${GM_getValue('maxConcurrentPreloads', maxConcurrentPreloads)}" min="1" max="999" placeholder="默认:15" style="flex-shrink: 2; width: 100px;" />
            </div>
            <div style="margin-bottom: 10px; display: flex; align-items: center;">
                <label for="dataCleanupInterval" style="flex-grow: 1; margin-right: 10px;">数据清理间隔(小时):</label>
                <input type="number" id="dataCleanupInterval" value="${GM_getValue('dataCleanupInterval', dataCleanupInterval / 3600000)}" min="1" placeholder="默认:1小时" style="flex-shrink: 2; width: 100px;" />
            </div>
            <div style="margin-bottom: 10px; display: flex; align-items: center;">
                <label for="maxContentSize" style="flex-grow: 1; margin-right: 10px;">单页最大内存限制(MB):</label>
                <input type="number" id="maxContentSize" value="${GM_getValue('maxContentSize', maxContentSize / 1024 / 1024)}" min="1" placeholder="默认:5MB" style="flex-shrink: 2; width: 100px;" />
            </div>
            <div style="margin-bottom: 10px; display: flex; align-items: center;">
                <label for="maxStorageItems" style="flex-grow: 1; margin-right: 10px;">最大链接存储项数:</label>
                <input type="number" id="maxStorageItems" value="${GM_getValue('maxStorageItems', maxStorageItems)}" min="10" placeholder="默认:300项" style="flex-shrink: 2; width: 100px;" />
            </div>
            <div style="margin-bottom: 10px; display: flex; align-items: center;">
                <label for="visualLinkReminderStyle" style="flex-grow: 1; margin-right: 10px;">预览加载成功样式:</label>
                <select id="visualLinkReminderStyle">
                    <option value="bullet" ${visualLinkReminderStyle === 'bullet' ? 'selected' : ''}>小圆点</option>
                    <option value="underline" ${visualLinkReminderStyle === 'underline' ? 'selected' : ''}>下划线</option>
                    <option value="highlight" ${visualLinkReminderStyle === 'highlight' ? 'selected' : ''}>高亮显示(黄色)</option>
                    <option value="changeColor" ${visualLinkReminderStyle === 'changeColor' ? 'selected' : ''}>改变颜色(品红)</option>
                    <option value="bold" ${visualLinkReminderStyle === 'bold' ? 'selected' : ''}>文字加粗</option>
                    <option value="border" ${visualLinkReminderStyle === 'border' ? 'selected' : ''}>边框(红色)</option>
                    <option value="increaseFontSize" ${visualLinkReminderStyle === 'increaseFontSize' ? 'selected' : ''}>增大字号</option>
                </select>
            </div>
            <div style="margin-bottom: 10px; display: flex; align-items: center;">
                <label for="whitelistMode" style="flex-grow: 1; margin-right: 10px;">启用白名单模式:</label>
                <input type="checkbox" id="whitelistMode" ${GM_getValue('whitelistMode', false) ? 'checked' : ''} />
                <button id="inputWhitelist" style="margin-left: 10px;">输入域名</button>
            </div>
            <div style="margin-bottom: 10px; display: flex; align-items: center;">
                <label for="blacklistMode" style="flex-grow: 1; margin-right: 10px;">启用黑名单模式:</label>
                <input type="checkbox" id="blacklistMode" ${GM_getValue('blacklistMode', false) ? 'checked' : ''} />
                <button id="inputBlacklist" style="margin-left: 10px;">输入域名</button>
            </div>
            <div style="margin-bottom: 10px; display: flex; align-items: center;">
                <label for="returnShortcutKey" style="flex-grow: 1; margin-right: 10px;">返回快捷键:</label>
                <input type="text" id="returnShortcutKey" readonly value="${GM_getValue('returnShortcutKey', returnShortcutKey)}" style="flex-shrink: 2; width: 100px; overflow-x: auto; height:30px; max-height:30px;" />
                <button id="returnsetKey" style="margin-left: 10px;">设定</button>
            </div>
            <div style="margin-bottom: 10px; display: flex; align-items: center;">
                <label for="directShortcutKey" style="flex-grow: 1; margin-right: 10px;">跳转快捷键:</label>
                <input type="text" id="directShortcutKey" readonly value="${GM_getValue('directShortcutKey', directShortcutKey)}" style="flex-shrink: 2; width: 100px; overflow-x: auto; height:30px; max-height:30px;" />
                <button id="directsetKey" style="margin-left: 10px;">设定</button>
            </div>
            <div style="margin-bottom: 10px; display: flex; align-items: center;">
                <label for="forwardShortcutKey" style="flex-grow: 1; margin-right: 10px;">前进快捷键:</label>
                <input type="text" id="forwardShortcutKey" readonly value="${GM_getValue('forwardShortcutKey', forwardShortcutKey)}" style="flex-shrink: 2; width: 100px; overflow-x: auto; height:30px; max-height:30px;" />
                <button id="forwardsetKey" style="margin-left: 10px;">设定</button>
            </div>


            <div style="text-align: center; padding-bottom: 30px;"> <!-- 增加了 padding-bottom -->
            <button id="saveSettings" style="
                width: 100%;
                padding: 10px;
                border: none;
                background-color: #4CAF50;
                color: white;
                text-align: center;
                text-decoration: none;
                display: inline-block;
                font-size: 16px;
                margin: 4px 2px;
                cursor: pointer;
                border-radius: 12px;
                box-shadow: 0 2px 5px rgba(0, 0, 0, .2);
                text-shadow: 0px 1px 0px #3d8b3d;">保存设置</button>
            </div>
            <div style="
                position: absolute;
                bottom: 10px;
                left: 20px;
                font-size: 10px;
                color: #666;
            ">
                <strong>当前版本:1.0.11(最新版本:${GM_getValue('latestVersion', '')})</strong>
            </div>
            <button 
                id="fetchHistoryVersions" 
                style="margin-top: 20px;right: 10px; position: absolute; bottom:10px;">更新查看
                <svg xmlns="http://www.w3.org/2000/svg" width="15" height="15" viewBox="0 0 24 24" style="vertical-align: middle;"><g fill-rule="evenodd"><path d="M13.118 2.913c-.46-.921-1.775-.921-2.236 0L2.118 20.441a1.25 1.25 0 0 0 1.118 1.809h17.528a1.25 1.25 0 0 0 1.118-1.809L13.118 2.913Zm-3.578-.67c1.014-2.028 3.906-2.028 4.92 0l8.764 17.527c.914 1.829-.416 3.98-2.46 3.98H3.236c-2.044 0-3.374-2.151-2.46-3.98L9.54 2.242Z"/><path d="M12 6.25a2.54 2.54 0 0 0-2.525 2.82l.78 7.013a.75.75 0 0 0 .745.667h2a.75.75 0 0 0 .745-.667l.78-7.012A2.54 2.54 0 0 0 12 6.25ZM12 17.25a1.75 1.75 0 1 0 0 3.5 1.75 1.75 0 0 0 0-3.5Z"/></g></svg></button>
            </div>
    `;


    var domainInputPanelHTML = `
    <div id="domainInputPanel" style="
        width: 300px;
        height: auto;
        position: fixed;
        top: 40%;
        left: 50%;
        transform: translate(-50%, -50%);
        background: #ececec;
        box-shadow: 0 4px 8px 0 rgba(0,0,0,0.4), 0 6px 20px 0 rgba(0,0,0,0.19);
        border-radius: 10px;
        padding: 15px;
        box-sizing: border-box;
        z-index: 10001;
        display: none;
        font-family: Arial, sans-serif;">
        <h3 style="
            text-align: center;
            font-size: 18px;
            color: #333;
            margin-bottom: 15px;
            text-shadow: 1px 1px 2px #888;">输入域名</h3>
        <textarea id="inputDomainName" placeholder="请输入域名" style="
            width: 100%;
            height: 250px;
            padding: 8px 10px;
            margin-bottom: 10px;
            border: 1px solid #ccc;
            border-radius: 4px;
            box-sizing: border-box;"></textarea>
        <div style="text-align: center;">
            <button id="saveInputDomain" style="
                padding: 8px 20px;
                border: none;
                background-color: #4285F4;
                color: white;
                text-align: center;
                text-decoration: none;
                display: inline-block;
                font-size: 14px;
                margin: 5px;
                cursor: pointer;
                border-radius: 12px;
                box-shadow: 0 2px 5px rgba(0, 0, 0, .2);">保存</button>
            <button id="addCurrentDomain" style="
                padding: 8px 20px;
                border: none;
                background-color: #34A853;
                color: white;
                text-align: center;
                text-decoration: none;
                display: inline-block;
                font-size: 14px;
                margin: 5px;
                cursor: pointer;
                border-radius: 12px;
                box-shadow: 0 2px 5px rgba(0, 0, 0, .2);">添加当前域名</button>
        </div>
    </div>
    `;
    // 向页面的body末尾追加刚才创建的设置面板
    document.body.insertAdjacentHTML('beforeend', domainInputPanelHTML);
    document.body.insertAdjacentHTML('beforeend', settingsPanelHTML);
    document.getElementById('fetchHistoryVersions').addEventListener('click', fetchAndDisplayVersionHistory);

    async function fetchAndDisplayVersionHistory() {
        const url = 'http://greasyfork.icu/zh-CN/scripts/493851-%E7%BD%91%E9%A1%B5%E7%9E%AC%E9%97%B4%E5%8A%A0%E8%BD%BD-%E8%B7%B3%E8%BF%87%E8%BF%9B%E5%BA%A6%E6%9D%A1%E7%9B%B4%E6%8E%A5%E5%8A%A0%E8%BD%BD%E7%BD%91%E9%A1%B5/versions';
        try {
            const response = await fetch(url);
            const text = await response.text();
            const parser = new DOMParser();
            const doc = parser.parseFromString(text, 'text/html');
            const versionHistory = doc.querySelector('.history_versions').innerHTML;
            
            displayVersionHistoryPanel(versionHistory);
        } catch (error) {
            console.error('Failed to fetch version history', error);
        }
    }
    function displayVersionHistoryPanel(versionHistory) {
        let versionHistoryPanel = document.getElementById('versionHistoryPanel');
        if (!versionHistoryPanel) {
            versionHistoryPanel = document.createElement('div');
            versionHistoryPanel.id = 'versionHistoryPanel';
            document.body.appendChild(versionHistoryPanel);
        } else {
            // 确保面板是可见的
            versionHistoryPanel.style.display = 'block';
        }
        versionHistoryPanel.innerHTML = `
            <div id="versionHistoryContent" style="position: relative; width: 300px; height: 300px;overflow: auto; position: fixed; top: 50%; left: 50%; transform: translate(-50%, -50%); background: #f2f2f2; box-shadow: 0 4px 8px 0 rgba(0,0,0,0.4), 0 6px 20px 0 rgba(0,0,0,0.19); border-radius: 15px; padding: 20px; box-sizing: border-box; z-index: 10000; display: block; font-family: Arial, sans-serif;">
                <button id="closeVersionHistory" style="position: absolute; top: 10px; right: 10px; background: none; border: none; cursor: pointer;">
                    <svg xmlns="http://www.w3.org/2000/svg" height="24" fill="currentColor" viewBox="0 0 334 334"><path d="M 320 48 Q 334 31 320 14 Q 303 0 286 14 L 167 133 L 48 14 Q 31 0 14 14 Q 0 31 14 48 L 133 167 L 14 286 Q 0 303 14 320 Q 31 334 48 320 L 167 201 L 286 320 Q 303 334 320 320 Q 334 303 320 286 L 201 167 L 320 48 L 320 48 Z" /></svg>
                </button>
                <h2 style="text-align: center; font-size: 20px; font-weight: bold; color: #333; margin-bottom: 20px; text-shadow: 1px 1px 2px #888;">更新日志</h2>
                ${versionHistory}
            </div>
        `;
    
        document.getElementById('closeVersionHistory').addEventListener('click', function() {
            versionHistoryPanel.style.display = 'none';
        });
    }



    // 找到设置面板中的元素
    var domainInputPanel = document.getElementById('domainInputPanel');
    var settingsPanel = document.getElementById('userScriptSettingsPanel');
    var saveSettingsButton = document.getElementById('saveSettings');
    var inputWhitelist = document.getElementById('inputWhitelist');
    var inputBlacklist = document.getElementById('inputBlacklist');
    var inputDomainName = document.getElementById('inputDomainName');
    var saveInputDomainButton = document.getElementById('saveInputDomain');
    var addCurrentDomainButton = document.getElementById('addCurrentDomain');
    var whitelistModeCheckbox = document.getElementById('whitelistMode');
    whitelistModeCheckbox.checked = GM_getValue('whitelistMode', false);

    var blacklistModeCheckbox = document.getElementById('blacklistMode');
    blacklistModeCheckbox.checked = GM_getValue('blacklistMode', true);
    // 为白名单和黑名单复选框添加事件监听器,以便当状态更改时使用GM_setValue保存状态
    whitelistModeCheckbox.addEventListener('change', function () {
        GM_setValue('whitelistMode', whitelistModeCheckbox.checked);
    });

    blacklistModeCheckbox.addEventListener('change', function () {
        GM_setValue('blacklistMode', blacklistModeCheckbox.checked);
    });

    inputWhitelist.addEventListener('click', function () {
        toggleDomainInputPanel();
        var whitelistDomains = GM_getValue('whitelistDomains', []).join('\n');
        inputDomainName.placeholder = '请输入白名单域名';
        inputDomainName.value = whitelistDomains; //显示保存的白名单域名。
        inputDomainName.dataset.mode = 'whitelist';
    });

    inputBlacklist.addEventListener('click', function () {
        toggleDomainInputPanel();
        var blacklistDomains = GM_getValue('blacklistDomains', []).join('\n');
        inputDomainName.placeholder = '请输入黑名单域名';
        inputDomainName.value = blacklistDomains; // 显示已保存的黑名单域名。
        inputDomainName.dataset.mode = 'blacklist';
    });

    saveInputDomainButton.addEventListener('click', function () {
        var mode = inputDomainName.dataset.mode;
        var domainInputContent = inputDomainName.value.trim();
        var domains = domainInputContent ? domainInputContent.split('\n') : [];
        domains = domains.map(function (domain) {
            return domain.trim();
        }).filter(function (domain) {
            return domain !== '';
        });

        GM_setValue(mode + 'Domains', domains);
        toggleDomainInputPanel();
        showAlert('域名已保存!');
    });


    addCurrentDomainButton.addEventListener('click', function () {
        var domain = window.location.hostname;
        var mode = inputDomainName.dataset.mode;
        var domains = GM_getValue(mode + 'Domains', []);
        if (domains.indexOf(domain) === -1) {
            domains.push(domain);
            GM_setValue(mode + 'Domains', domains);
            toggleDomainInputPanel();
            showAlert('域名已保存!');
        } else {
            showAlert('域名已存在!');
        }
    });
    // 保存按钮的事件处理程序

    saveSettingsButton.addEventListener('click', function () {
        toggleSettingsPanel();
        // 向用户显示确认消息
        showAlert("设置已保存!所有设置都即刻生效!");
        // 获取各输入字段的值,并为非有效值提供默认设置
        var maxConcurrentPreloadsInput = document.getElementById('maxConcurrentPreloads');
        var newMaxConcurrentPreloads = parseInt(maxConcurrentPreloadsInput.value, 10) || 25; // 如果值无效,提供默认值 25

        var dataCleanupIntervalInput = document.getElementById('dataCleanupInterval');
        var newDataCleanupInterval = parseInt(dataCleanupIntervalInput.value, 10) || 1; // 提供默认值 1 小时,把小时转换成毫秒

        var maxContentSizeInput = document.getElementById('maxContentSize');
        var newMaxContentSize = parseInt(maxContentSizeInput.value, 10) || 5; // 提供默认值 5MB

        var maxStorageItemsInput = document.getElementById('maxStorageItems');
        var newMaxStorageItems = parseInt(maxStorageItemsInput.value, 10) || 300; // 提供默认值 300 项

        var visualLinkReminderStyleInput = document.getElementById('visualLinkReminderStyle');
        var newVisualLinkReminderStyle = visualLinkReminderStyleInput.value || 'underline';


        // 使用 GM_setValue 函数保存设置
        GM_setValue('maxConcurrentPreloads', newMaxConcurrentPreloads);
        GM_setValue('dataCleanupInterval', newDataCleanupInterval);
        GM_setValue('maxContentSize', newMaxContentSize);
        GM_setValue('maxStorageItems', newMaxStorageItems);
        GM_setValue('visualLinkReminderStyle', newVisualLinkReminderStyle);
        GM_setValue('returnShortcutKey', document.querySelector('#returnShortcutKey').value);
        GM_setValue('directShortcutKey', document.querySelector('#directShortcutKey').value);
        GM_setValue('forwardShortcutKey', document.querySelector('#forwardShortcutKey').value);

        // 更新脚本内部变量
        maxConcurrentPreloads = newMaxConcurrentPreloads;
        dataCleanupInterval = newDataCleanupInterval * 3600000;
        maxContentSize = newMaxContentSize * 1024 * 1024;
        maxStorageItems = newMaxStorageItems;
        visualLinkReminderStyle = newVisualLinkReminderStyle;

        // 调整预加载队列和预加载指示器颜色
        cleanPreloadQueue();
    });

    function toggleSettingsPanel() {
        if (settingsPanel.style.display === 'none') {
            settingsPanel.style.display = 'block';
        } else {
            settingsPanel.style.display = 'none';
        }
    }
    function toggleDomainInputPanel() {
        if (domainInputPanel.style.display === 'none') {
            domainInputPanel.style.display = 'block';
        } else {
            domainInputPanel.style.display = 'none';
        }
    }

    GM_registerMenuCommand("显示/隐藏参数与功能设置", function () {
        toggleSettingsPanel();
    });

    function addAVisualLinkReminder(element) {
        // Get the URL from the link element
        var href = element.href;
        try {
            var linkURL = new URL(href);
            var currentOrigin = window.location.origin;
            if (linkURL.origin === currentOrigin) {
                if (element.dataset.preloaded) {
                    switch (visualLinkReminderStyle) {
                        case 'bullet':
                            if (!element.firstChild || element.firstChild.textContent !== '• ') {
                                var bullet = document.createTextNode('• ');
                                element.insertBefore(bullet, element.firstChild);
                            }
                            break;
                        case 'underline':
                            if (!element.style.textDecoration) {
                                element.style.textDecoration = 'underline';
                                element.style.textDecorationSkipInk = 'none';
                            }
                            break;
                        case 'highlight':
                            if (!element.style.backgroundColor) {
                                element.style.backgroundColor = 'yellow';
                            }
                            break;
                        case 'changeColor':
                            if (!element.style.color) {
                                element.style.color = '#FF00FF'; // Magenta color
                            }
                            break;
                        case 'bold':
                            if (element.style.fontWeight !== 'bold') {
                                element.style.fontWeight = 'bold';
                            }
                            break;
                        case 'border':
                            if (!element.style.border) {
                                element.style.border = '2px solid red';
                                element.style.padding = '2px';
                                element.style.borderRadius = '4px';
                            }
                            break;
                        case 'increaseFontSize':
                            if (element.style.fontSize !== 'larger') {
                                element.style.fontSize = 'larger';
                            }
                            break;
                        default:
                            // If no style is selected, default to no decoration.
                            break;
                    }
                }
            }
        } catch (e) {
            console.error('Error adding visual link reminder:', e);
        }
    }

    function addDraggableIcon() {
        var svgHTML = '<svg id="draggableIcon" style="position: fixed; top: 50%; right: -20px; transform: translateY(-50%); cursor: pointer; z-index: 9999999;" width="50" height="50" viewBox="0 0 24 24" xmlns="http://www.w3.org/2000/svg"><path fill="currentColor" d="M3.464 20.536C4.93 22 7.286 22 12 22c4.714 0 7.071 0 8.535-1.465C22 19.072 22 16.714 22 12s0-7.071-1.465-8.536C19.072 2 16.714 2 12 2S4.929 2 3.464 3.464C2 4.93 2 7.286 2 12c0 4.714 0 7.071 1.464 8.535" opacity=".5"/><path fill="currentColor" d="M12.03 9.53a.75.75 0 0 0-1.06-1.06l-3 3a.75.75 0 0 0 0 1.06l3 3a.75.75 0 1 0 1.06-1.06L9.56 12z"/><path fill="currentColor" d="M16.03 9.53a.75.75 0 0 0-1.06-1.06l-3 3a.75.75 0 0 0 0 1.06l3 3a.75.75 0 1 0 1.06-1.06L13.56 12z"/></svg>',
            div = document.createElement('div');
        div.innerHTML = svgHTML;
        document.body.appendChild(div.firstChild);

        function onDrag(e, move) {
            var startX = ('touches' in e ? e.touches[0] : e).clientX - dragIcon.getBoundingClientRect().left,
                startY = ('touches' in e ? e.touches[0] : e).clientY - dragIcon.getBoundingClientRect().top;
            function dragging(ev) {
                var clientX = ('touches' in ev ? ev.touches[0] : ev).clientX,
                    clientY = ('touches' in ev ? ev.touches[0] : ev).clientY;
                dragIcon.style.left = clientX - startX + 'px';
                dragIcon.style.top = clientY - startY + 'px';
            }
            function endDrag() {
                document.removeEventListener(move ? 'mousemove' : 'touchmove', dragging);
                document.removeEventListener(move ? 'mouseup' : 'touchend', endDrag);
                document.body.style.overflow = '';
                dragIcon.style.transition = '';
                GM_setValue('iconPosition', { left: dragIcon.style.left, top: dragIcon.style.top });
            }
            document.addEventListener(move ? 'mousemove' : 'touchmove', dragging);
            document.addEventListener(move ? 'mouseup' : 'touchend', endDrag);
            document.body.style.overflow = 'hidden';
            dragIcon.style.transition = 'none';
            if (!move) {
                dragIcon.dataset.pressTimer = setTimeout(endDrag, 500);
            }
        }

        var dragIcon = document.getElementById('draggableIcon');
        dragIcon.style.display = 'none';
        var savedPosition = GM_getValue('iconPosition');
        savedPosition && (dragIcon.style.left = savedPosition.left, dragIcon.style.top = savedPosition.top);

        dragIcon.ontouchstart = function (e) {
            onDrag(e, false);
        };
        dragIcon.ontouchend = function () {
            clearTimeout(dragIcon.dataset.pressTimer);
        };
        dragIcon.onmousedown = function (e) {
            e.preventDefault();
            onDrag(e, true);
        };
    }

    function addToPreloadQueue(url, element) {
        // 增加了对集合中存在性的检查
        if (isInViewport(element) && !preloadSet.has(url)) {
            preloadSet.add(url);
            preloadQueue.push({ url: url, element: element });
        }
    }

    function blobToBase64(blob) {
        return new Promise((resolve, reject) => {
            const reader = new FileReader();
            reader.onloadend = () => resolve(reader.result);
            reader.onerror = e => reject(e);
            reader.readAsDataURL(blob);
        });
    }

    function cancelPreload(url) {
        if (abortControllers[url]) {
            abortControllers[url].abort();
            delete abortControllers[url];// 删除abortController实例的引用以防内存泄露
            console.log("Preload cancelled for link out of viewport:", url);
            currentPreloads--; // 更新计数
            preloadNext(); // 尝试开始下一个预加载
        }
        // 从预加载队列中移除链接
        preloadQueue = preloadQueue.filter(item => item.url !== url);
        preloadSet.delete(url); // 从预加载集合中移除链接
    }

    function checkAndAddBulletsForPreloadedLinks() {
        var links = document.querySelectorAll('a');
        links.forEach(function (link) {
            if (link.dataset.preloaded) {
                addAVisualLinkReminder(link);
            }
        });
    }


    function cleanPreloadQueue() {
        // 清理已经预加载或者不在视口内的链接
        preloadQueue = preloadQueue.filter(item => {
            if (!(isInViewport(document.querySelector(`a[href="${item.url}"]`)) && !document.querySelector(`a[href="${item.url}"]`).dataset.preloaded)) {
                preloadSet.delete(item.url); // 如果不满足条件,则从集合中删除
                return false;
            }
            return true;
        });
    }

    function cleanAbortControllers() {
        // 获取当前时间
        var now = Date.now();

        // 遍历abortControllers对象的属性
        for (var url in abortControllers) {
            // 如果请求已经很久没有响应,那么我们认为它可能已经失效,需要删除控制器
            // 或如果请求已经被中止,亦应删除
            var controller = abortControllers[url];
            if ((controller.timestamp && now - controller.timestamp > 30000) || controller.signal.aborted) { // 30秒或已中止
                delete abortControllers[url];
            }
        }
    }

    function createElementWithStylesAndAttributes(tag, styles, attributes) {//为一系列创建元素的函数添加样式和属性
        let element = document.createElement(tag);
        if (styles) {
            Object.assign(element.style, styles);
        }
        if (attributes) {
            for (const key in attributes) {
                if (attributes.hasOwnProperty(key)) {
                    if (key in element) {
                        element[key] = attributes[key];
                    } else {
                        element.setAttribute(key, attributes[key]);
                    }
                }
            }
        }
        return element;
    }

    function debounce(func, wait, immediate) {
        var timeout, called = false;
        return function () {
            var context = this, args = arguments;
            var later = function () {
                timeout = null;
                if (!immediate && !called) func.apply(context, args);
                called = false; // 重置调用标志
            };
            var callNow = immediate && !timeout;
            if (callNow) {
                func.apply(context, args);
                called = true; // 立即执行时,设置调用标志
            }
            clearTimeout(timeout);
            timeout = setTimeout(later, wait);
        };
    };

    function deleteOldContentFromDB() {
        const now = Date.now();
        const transaction = db.transaction([dbStoreName], 'readwrite');
        const store = transaction.objectStore(dbStoreName);

        var contentRequest = store.getAll();

        contentRequest.onsuccess = function () {
            var contents = contentRequest.result;
            // 如有必要,根据最后访问时间进行排序
            contents.sort((a, b) => b.timestamp - a.timestamp);

            // 判断记录是否超出最大存储量或者是否过期,然后执行删除
            contents.forEach((content, index) => {
                if (now - content.timestamp > dataCleanupInterval || index >= maxStorageItems) { // 检查每一项是否过期或超出容量限制
                    store.delete(content.url);
                }
            });
        };

        contentRequest.onerror = function (event) {
            console.error("Error fetching contents from IndexedDB", event.target.error);
        };
    }

    function toggleDragIconVisibility(show) {
        var dragIcon = document.getElementById('draggableIcon');
        if (dragIcon) {
            dragIcon.style.display = show ? 'block' : 'none';
        }
    }

    function displayPreloadedContent(base64Content, url) {
        if (base64Content.startsWith('data:')) {
            var binary = atob(base64Content.split(',')[1]);
            var array = new Uint8Array(binary.length);
            for (var i = 0; i < binary.length; i++) {
                array[i] = binary.charCodeAt(i);
            }
            var documentEncoding = document.characterSet || 'UTF-8'; // 获取网页当前编码,如果无法获取,默认为UTF-8
            var blobContent = new Blob([array], { type: `text/html;charset=${documentEncoding}` }); // 使用文档编码设置

            var reader = new FileReader();
            reader.onload = function () {
                var existingFullPageDiv = document.getElementById('fullPageDiv');
                if (existingFullPageDiv) {
                    existingFullPageDiv.parentNode.removeChild(existingFullPageDiv);
                    existingFullPageDiv.remove();
                }

                var fullPageDiv = document.createElement('div');
                fullPageDiv.id = 'fullPageDiv';
                fullPageDiv.style.cssText = `
                    position: fixed;
                    top: 0;
                    left: 0;
                    width: 100%;
                    height: 100%;
                    overflow: auto;
                    z-index: 1000;
                    background: white;
                `;
                fullPageDiv.innerHTML = reader.result;
                toggleDragIconVisibility(true);
                document.body.style.overflow = 'hidden';
                document.body.appendChild(fullPageDiv);
            };
            reader.readAsText(blobContent, documentEncoding); // 使用从文档获取的编码进行解码
        }
    }

    function initDB(success) {
        if (!dbReady) {
            openDB(success);
        } else if (typeof success === 'function') {
            success();
        }
    }

    function isInViewport(element) {
        var rect = element.getBoundingClientRect();
        var inViewport = (
            rect.top >= 0 &&
            rect.left >= 0 &&
            rect.bottom <= (window.innerHeight || document.documentElement.clientHeight) &&
            rect.right <= (window.innerWidth || document.documentElement.clientWidth)
        );

        if (!inViewport && element.dataset.preloaded) {
            cancelPreload(element.href); // 如果链接不在视窗中并且已被标记为预加载,取消它的预加载
            element.dataset.preloaded = false; // 移除预加载标记
        }

        return inViewport;
    }

    function isMobileDevice() {
        return /Android|webOS|iPhone|iPad|iPod|BlackBerry|IEMobile|Opera Mini/i.test(navigator.userAgent);
    }


    function observeDOMChanges() {
        var handledLinks = new Set(); // 用于储存处理过的链接

        var config = { childList: true, subtree: true };
        var callback = function (mutationsList, observer) {
            requestAnimationFrame(function () {
                mutationsList.forEach(function (mutation) {
                    if (mutation.type == 'childList' && mutation.addedNodes.length) {
                        mutation.addedNodes.forEach(function (node) {
                            if (node.nodeType === 1 && node.matches('a[href]')) {
                                var href = node.getAttribute('href');
                                if (!preloadSet.has(href) && !handledLinks.has(href)) {
                                    preloadLink(href, node);
                                    handledLinks.add(href); // 将链接添加到 handledLinks 中进行标记
                                    preloadSet.add(href);// 将链接添加到 preloadSet 中
                                }
                            }
                        });
                    }
                });
                preloadVisibleLinks();
            });
        };

        var observer = new MutationObserver(callback);
        observer.observe(document.body, config);
    }

    var dbInitializationPromise = null;// 用来追踪数据库初始化的Promise
    function openDB(callback) {
        if (dbInitializationPromise) { // 如果存在初始化的Promise,直接返回它并添加回调
            dbInitializationPromise.then(callback).catch(err => console.error('IndexedDB init error:', err));
            return;
        }

        // 新建一个Promise来处理初始化过程
        dbInitializationPromise = new Promise((resolve, reject) => {
            var request = indexedDB.open(dbName, dbVersion);

            request.onupgradeneeded = function (event) {
                var db = event.target.result;
                if (!db.objectStoreNames.contains(dbStoreName)) {
                    db.createObjectStore(dbStoreName, { keyPath: 'url' });
                }
            };

            request.onsuccess = function (event) {
                db = event.target.result;
                dbReady = true;
                console.log('IndexedDB database opened successfully');
                db.onerror = function (event) {
                    console.error("Database error: " + event.target.error.message);
                };
                resolve(db);
            };

            request.onerror = function (event) {
                console.error('IndexedDB database open error:', event.target.errorCode);
                dbInitializationPromise = null; // 如果Promise失败,我们重置这个变量
                reject(event.target.error);
            };
        });

        // 调用传入的回调函数
        dbInitializationPromise.then(callback).catch(err => console.error('IndexedDB init error:', err));
    }


    function appropriateAsAttributeValue(url) {
        if (url.endsWith('.css')) {
            return 'style';
        } else if (url.endsWith('.js')) {
            return 'script';
        } else if (url.match(/(.jpg|.jpeg|.png|.gif)$/)) {
            return 'image';
        } else if (url.endsWith('.json')) {
            return 'fetch';
        } else {
            return 'fetch';
        }
    }
    function preloadLink(url, element) {
        try {
            var linkURL = new URL(url);
            if (isBlacklistModeEnabled && blacklistDomains.includes(linkURL.hostname)) {
                console.log('Not preloading: Blacklisted domain:', linkURL.hostname);
                return;
            }
            // 跳过与当前页面相同或者跨域的URL
            if (linkURL.hostname !== window.location.hostname || linkURL.href === window.location.href) {
                console.warn('Not preloading: Same page or cross-origin link:', url);
                return;
            }
            if (preloadSet.has(url) || element.dataset.preloaded) {
                console.log('Not preloading: Already preloaded or enqueued for preload:', url);
                return;
            }

            if (currentPreloads < maxConcurrentPreloads) {
                currentPreloads++;
                let options = {
                    cache: "force-cache", // 使用force-cache可以帮助减少不必要的网络请求
                    as: appropriateAsAttributeValue(url) // 根据不同的链接类型,为 'as' 属性设置适当的值
                };

                if (element.rel && (element.rel.includes('noreferrer') || element.rel.includes('noopener'))) {
                    options.referrerPolicy = 'no-referrer';
                }

                // 创建一个新的abortController实例,并将signal传给fetch
                var controller = new AbortController();
                controller.timestamp = Date.now();
                var signal = controller.signal;
                options.signal = signal; // 将signal添加到fetch选项中
                abortControllers[url] = controller;

                fetch(linkURL.href, options).then(function (response) {

                    if (!response.ok) {
                        throw new Error('HTTP error, status = ' + response.status);
                    }

                    return response.blob();
                }).then(function (blob) {
                    saveContentToDB(url, blob);
                }).then(function () {
                    addAVisualLinkReminder(element); // 将此行放置在数据预加载标记之后
                    delete abortControllers[url];
                }).catch(function (error) {
                    if (error.name === 'AbortError') {
                        console.warn('Fetch aborted: ', url);
                    } else {
                        console.error('Preload failed for ', url, ':', error.message);
                    }
                }).finally(function () {
                    currentPreloads--;
                    preloadNext();
                    delete abortControllers[url];
                });
            } else {
                addToPreloadQueue(url, element);
                return;
            }
        } catch (e) {
            console.error('Error preloading link:', e.message);
        }
    }

    function preloadNext() {
        if (preloadQueue.length > 0 && currentPreloads < maxConcurrentPreloads) {
            var nextPreload = preloadQueue.shift();
            preloadLink(nextPreload.url, nextPreload.element);
            delete abortControllers[nextPreload.url];
        }
    }

    function preloadVisibleLinks() {
        if (!isMobileDevice()) {
            // 是移动设备,维持原有的基于视窗滚动的预加载策略
            return;
        }
        var links = document.querySelectorAll('a');

        var visibleLinks = Array.from(links).filter(function (link) {
            var href = link.href;
            return isInViewport(link) && !link.dataset.preloaded && (shouldPreloadMapping[href] !== false);
        });

        visibleLinks.sort(function (a, b) {
            var aRect = a.getBoundingClientRect();
            var bRect = b.getBoundingClientRect();
            return (window.innerHeight - aRect.bottom) - (window.innerHeight - bRect.bottom);
        });

        var preloadLimit = Math.min(visibleLinks.length, maxConcurrentPreloads - currentPreloads);

        for (var i = 0; i < preloadLimit; i++) {
            (function (linkElement) {
                var href = linkElement.href;
                if (shouldPreloadMapping[href] === undefined) {
                    // 如果该链接的预加载状态尚未决定,则发起检查
                    shouldPreload(linkElement.href).then(function (should) {
                        shouldPreloadMapping[href] = should;
                        if (should && isInViewport(linkElement)) {
                            preloadLink(linkElement.href, linkElement);
                            linkElement.dataset.preloaded = true;
                        }
                    });
                } else if (shouldPreloadMapping[href]) {
                    // 如果已经确定需要预加载,则直接预加载,无需重复检查
                    preloadLink(linkElement.href, linkElement);
                    linkElement.dataset.preloaded = true;
                }
            })(visibleLinks[i]);
        }
    }

    function processDBWriteQueue() {
        if (dbWriteInProgress || dbWriteQueue.length === 0) {
            return;
        }

        dbWriteInProgress = true;
        var item = dbWriteQueue.shift();

        blobToBase64(item.blob).then(base64data => {
            var transaction = db.transaction([dbStoreName], 'readwrite');
            var objectStore = transaction.objectStore(dbStoreName);

            // 处理事务完成
            return new Promise((resolve, reject) => {
                var request = objectStore.put({ url: item.url, htmlContent: base64data, timestamp: Date.now() });
                request.onsuccess = () => resolve();
                request.onerror = () => reject(request.error);
            });
        }).then(() => {
            console.log('Page content saved to IndexedDB for', item.url);
            dbWriteInProgress = false;
            processDBWriteQueue(); // 递归处理队列中的下一项
        }).catch(error => {
            console.error('IndexedDB save operation failed for', item.url, error);
            dbWriteQueue.unshift(item); // 发生错误时重新将项目放入队列
            dbWriteInProgress = false;
            setTimeout(processDBWriteQueue, 1000); // 延迟重试
        });
    }

    function readContentFromDB(url, callback) {
        var transaction = db.transaction([dbStoreName], 'readonly');
        var objectStore = transaction.objectStore(dbStoreName);
        var request = objectStore.get(url);

        request.onsuccess = function (event) {
            callback(event.target.result);
        };

        request.onerror = function (event) {
            console.error('IndexedDB read failed for', url);
            callback(null);
        };
    }

    function saveContentToDB(url, blob) {
        if (!dbReady) {
            console.error('IndexedDB is not ready for writing data.');
            return;
        }

        // 检查 blob 的大小是否超过最大内容大小设置
        if (blob.size > maxContentSize) {
            console.log('Content size exceeds the maxContentSize limit. Not saving to IndexedDB');
            return;
        }

        // 添加Promises到队列而不是直接使用FileReader
        dbWriteQueue.push({ url, blob });

        if (!dbWriteInProgress) {
            // 只有当上一个事务不在进行时才开始读取blob数据
            processDBWriteQueue();
        }
    }

    function scheduleNextCleanup() {
        // 删除旧内容后,再次调用此函数以依据当前间隔设定继续调度
        setTimeout(function () {
            deleteOldContentFromDB();
            scheduleNextCleanup();
        }, dataCleanupInterval);
    }

    function setupMouseHoverPreload() {
        document.addEventListener('mouseover', function (event) {
            var target = event.target.closest('a');
            if (target && !target.dataset.preloaded) {

                // 判断链接是否指向图片,如果是,就不进行预加载处理
                var href = target.getAttribute('href');

                // 更新正则表达式来匹配 Greasy Fork 的特定图片链接模式
                if (href.match(/\.(jpeg|jpg|gif|png|webp)$/i) ||
                    href.includes("active_storage/blobs/redirect")) {
                    console.log('Skip preloading for image link', href);
                    return; // 如果链接指向图片,直接返回,不设置预加载
                }

                // 鼠标悬停65毫秒以上就启动预加载
                target.dataset.hoverTimeout = setTimeout(function () {
                    shouldPreload(target.href).then(function (shouldPreloadResult) {
                        if (shouldPreloadResult && !target.dataset.preloaded) {
                            preloadLink(target.href, target);
                            target.dataset.preloaded = true; // 设置链接已预加载
                            addAVisualLinkReminder(target); // 添加小圆球指示器
                        }
                    });
                }, 65);
            }
        });

        document.addEventListener('mouseout', function (event) {
            var target = event.target.closest('a');
            if (target && target.dataset.hoverTimeout) {
                // 当鼠标移开时清除定时器
                clearTimeout(target.dataset.hoverTimeout);
                target.dataset.hoverTimeout = null;
            }
        });
    }

    function showAlert(message) {
        let alertBox = createElementWithStylesAndAttributes("div", {
            position: "fixed",
            bottom: "10px",
            left: "50%",
            transform: "translateX(-50%)",
            backgroundColor: "#f2f2f2",
            padding: "20px",
            borderRadius: "15px",
            boxShadow: "0 4px 8px 0 rgba(0,0,0,0.4), 0 6px 20px 0 rgba(0,0,0,0.19)",
            zIndex: "10000",
        });
        let alertText = createElementWithStylesAndAttributes("div", {
            margin: "0",
            fontSize: "16px",
            fontWeight: "bold",
            textAlign: "center",
            color: "#333",
            textShadow: "1px 1px 2px rgba(128, 128, 128, 0.5)",
        }, {
            innerText: message
        });
        alertBox.appendChild(alertText);
        document.body.appendChild(alertBox);
        setTimeout(function () {
            document.body.removeChild(alertBox);
        }, 1500);
    }

    function shouldPreload(url) {

        var domain = new URL(url).hostname;

        // 检查域名是否在白名单中
        if (isWhitelistModeEnabled && !whitelistDomains.includes(domain)) {
            return Promise.resolve(false);
        }

        // 如果启用了黑名单模式且 URL 在黑名单上,则不预载
        if (isBlacklistModeEnabled && blacklistDomains.includes(domain)) {
            return Promise.resolve(false);
        }
        // 定义一个请求来获取页面的内容
        return fetch(url, { method: 'GET', mode: 'no-cors' })
            .then(function (response) {
                return response.text();
            })
            .then(function (html) {
                // 检查页面是否包含跳转脚本
                var redirectRegex = /window\.location\.href\s*=\s*['"]([^'"]+)['"]/;
                var match = redirectRegex.exec(html);
                if (match && match[1]) {
                    // 页面有跳转,打印消息并返回 false
                    console.log('The page has a redirect script. Skipping preload for:', url);
                    return false;
                }
                // 页面没有跳转,返回 true
                return true;
            })
            .catch(function (error) {
                console.error('Error fetching page for preload check:', error);
                return false;
            });
    }

    // 确保数据库准备就绪后再调用 preloadVisibleLinks
    function startLinkPreloading() {
        initDB(preloadVisibleLinks);
    }

    function handleBackNavigation() {
        if (currentPreviewIndex > 0) {
            currentPreviewIndex -= 1;
            var prevURL = clickedLinks[currentPreviewIndex];

            readContentFromDB(prevURL, function (data) {
                if (data) {
                    displayPreloadedContent(data.htmlContent, prevURL);
                } else {
                    location.href = prevURL;
                }
            });
        } else {
            var fullPageDiv = document.getElementById('fullPageDiv');

            if (fullPageDiv) {
                fullPageDiv.remove();
                toggleDragIconVisibility(false);
                document.body.style.overflow = '';
            }

            clickedLinks = [];
            currentPreviewIndex = -1;
        }
    }

    function handleForwardNavigation() {
        if (currentPreviewIndex < clickedLinks.length - 1) {
            currentPreviewIndex += 1;
            var nextURL = clickedLinks[currentPreviewIndex];

            readContentFromDB(nextURL, function (data) {
                if (data) {
                    displayPreloadedContent(data.htmlContent, nextURL);
                } else {
                    location.href = nextURL;
                }
            });
        }
    }

    function navigateToURL() {
        if (clickedLinks.length > 0 && currentPreviewIndex >= 0) {
            var currentURL = clickedLinks[currentPreviewIndex];
            window.location.href = currentURL;
        }
    }


    let touchStartX = 0;
    let touchStartY = 0;
    let iconVisible = false;
    const svgBack = '<svg id="sliderBIcon" style="position: fixed; top: 50%; left: 100%; transform: translateY(-50%); cursor: pointer; z-index: 9999999;" width="100" height="50" viewBox="0 0 24 24" xmlns="http://www.w3.org/2000/svg"><path d="M14.53 7.53a.75.75 0 0 0-1.06-1.06l-5 5a.75.75 0 0 0 0 1.06l5 5a.75.75 0 1 0 1.06-1.06L10.06 12l4.47-4.47Z"/><path fill-rule="evenodd" d="M12 1.25C6.063 1.25 1.25 6.063 1.25 12S6.063 22.75 12 22.75 22.75 17.937 22.75 12 17.937 1.25 12 1.25ZM2.75 12a9.25 9.25 0 1 1 18.5 0 9.25 9.25 0 0 1-18.5 0Z"/></svg>';
    const svgForward = '<svg id="sliderFIcon" style="position: fixed; top: 50%; right: 100%; transform: translateY(-50%); cursor: pointer; z-index: 9999999;" width="100" height="50" viewBox="0 0 24 24" xmlns="http://www.w3.org/2000/svg"><path d="M10.53 6.47a.75.75 0 1 0-1.06 1.06L13.94 12l-4.47 4.47a.75.75 0 1 0 1.06 1.06l5-5a.75.75 0 0 0 0-1.06l-5-5Z"/><path fill-rule="evenodd" d="M12 1.25C6.063 1.25 1.25 6.063 1.25 12S6.063 22.75 12 22.75 22.75 17.937 22.75 12 17.937 1.25 12 1.25ZM2.75 12a9.25 9.25 0 1 1 18.5 0 9.25 9.25 0 0 1-18.5 0Z"/></svg>';
    const divBack = document.createElement('div');
    const divForward = document.createElement('div');
    divBack.innerHTML = svgBack;
    divForward.innerHTML = svgForward;
    document.body.appendChild(divBack);
    document.body.appendChild(divForward);

    document.addEventListener('touchstart', function (e) {
        touchStartX = e.changedTouches[0].screenX;
        touchStartY = e.changedTouches[0].screenY;
    });

    document.addEventListener('touchmove', function (e) {
        let touchMoveX = e.changedTouches[0].screenX;
        let touchMoveY = e.changedTouches[0].screenY;
        let moveRightDistance = touchMoveX - touchStartX;
        let moveLeftDistance = touchStartX - touchMoveX;
        let moveYDistance = Math.abs(touchMoveY - touchStartY);
        if (moveYDistance > 100) { // 如果Y轴上滑动距离超过100px, 就直接返回
            iconVisible = false;
            divForward.firstChild.style.right = '100%';
            divBack.firstChild.style.left = '100%';
            return;
        }
        if (moveRightDistance > 50 && !iconVisible) {
            iconVisible = true;
            divBack.firstChild.style.left = '-20px';
        } else if (moveRightDistance <= 50 && iconVisible) {
            iconVisible = false;
            divBack.firstChild.style.left = '100%';
        }
        if (iconVisible && moveRightDistance <= 100) {
            divBack.firstChild.style.left = `${0.8 * (moveRightDistance - 50)}px`;
        } else if (moveRightDistance > 100) {
            divBack.firstChild.style.left = '-20px';
        }

        if (moveLeftDistance > 50 && !iconVisible) {
            iconVisible = true;
            divForward.firstChild.style.right = '-20px';
        } else if (moveLeftDistance <= 50 && iconVisible) {
            iconVisible = false;
            divForward.firstChild.style.right = '100%';
        }
        if (iconVisible && moveLeftDistance <= 100) {
            divForward.firstChild.style.right = `${0.8 * (moveLeftDistance - 50)}px`;
        } else if (moveLeftDistance > 100) {
            divForward.firstChild.style.right = '-20px';
        }
    });

    document.addEventListener('touchend', function (e) {
        let touchEndX = e.changedTouches[0].screenX;
        let touchEndY = e.changedTouches[0].screenY;
        let touchDistance = touchEndX - touchStartX;
        let touchYDistance = Math.abs(touchEndY - touchStartY);

        if (touchYDistance > 100) {
            return;
        }
        if (touchDistance > 150) {
            handleBackNavigation();
        }
        if (touchDistance < -150) {
            handleForwardNavigation();
        }

        iconVisible = false;
        divBack.firstChild.style.left = '100%';
        divForward.firstChild.style.right = '100%';
    });



    document.addEventListener('keydown', function (e) {
        var returnShortcutKeyString = GM_getValue('returnShortcutKey', 'Shift+R').toLowerCase(); // 默认值 'Shift+R',并统一转为小写
        var returnShortcutKeys = returnShortcutKeyString.split('+').map(key => key.toLowerCase()); // 将快捷键分割并转为小写

        var directShortcutKeyString = GM_getValue('directShortcutKey', 'Shift+E').toLowerCase(); // 默认值 'Shift+e',并统一转为小写
        var directShortcutKeys = directShortcutKeyString.split('+').map(key => key.toLowerCase()); // 将快捷键分割并转为小写

        var forwardShortcutKey = GM_getValue('forwardShortcutKey', 'Shift+F').toLowerCase(); // 默认值 'ArrowRight',并统一转为小写
        var forwardShortcutKeys = forwardShortcutKey.split('+').map(key => key.toLowerCase()); // 将快捷键分割并转为小写
        // 记录当前按下的键,考虑到各种特殊键
        var keyPressed = {
            'alt': e.altKey,
            'shift': e.shiftKey,
            'control': e.ctrlKey,
            'meta': e.metaKey // 对于 Mac 的 Command 键
        };
        keyPressed[e.key.toLowerCase()] = true; // 按下的普通键也加入到 keyPressed 对象中,并转为小写

        var returnKeysPressed = returnShortcutKeys.every(key => keyPressed[key]);
        var directKeysPressed = directShortcutKeys.every(key => keyPressed[key]);
        var forwardKeysPressed = forwardShortcutKeys.every(key => keyPressed[key]);
        if (returnKeysPressed) {
            handleBackNavigation(); // 对应的操作函数
        }
        if (directKeysPressed) {
            navigateToURL(); // 对应的操作函数
        }
        if (forwardKeysPressed) {
            handleForwardNavigation(); // 对应的操作函数
        }
    });

    var debouncedScrollHandler = debounce(function () {
        cleanPreloadQueue();
        cleanAbortControllers();
        preloadVisibleLinks();
        checkAndAddBulletsForPreloadedLinks(); // 添加小圆点的检查逻辑
    }, 10);

    var debouncedScrollHandlerPC = debounce(function () {
        cleanPreloadQueue();
        cleanAbortControllers();
        checkAndAddBulletsForPreloadedLinks(); // 添加小圆点的检查逻辑
    }, 10);



    document.addEventListener('click', function (event) {
        var target = event.target.closest('a');
        if (target && target.href) {
            event.preventDefault();
            if (event.ctrlKey && !event.shiftKey) {
                // 用户按下了Ctrl键并点击了链接 - 在后台新标签页中打开
                window.open(target.href);
            } else if (event.shiftKey) {
                // 用户按下了Shift键并点击了链接 - 在前台新标签页中打开
                window.open(target.href, '_blank');
            } else {
                // 正常点击 - 不打开新标签页,遵循预加载逻辑以下或直接导航到链接地址
                if (!clickedLinks.includes(target.href)) {
                    clickedLinks.push(target.href);
                }
                currentPreviewIndex = clickedLinks.indexOf(target.href);


                if (target.dataset.preloaded) {
                    readContentFromDB(target.href, function (data) {
                        if (data) {
                            displayPreloadedContent(data.htmlContent, currentPreviewIndex);
                        } else {
                            location.href = target.href;
                        }
                    });
                } else {
                    location.href = target.href;
                }
            }
        }
    });

    observeDOMChanges();
    openDB();
    startLinkPreloading();
    initDB(function () {
        deleteOldContentFromDB();
        scheduleNextCleanup();
    });
    if (isMobileDevice()) {
        window.addEventListener('touchend', debouncedScrollHandler);
        // 是移动设备,维持原有的基于视窗滚动的预加载策略
    } else {
        // 非移动设备,使用基于鼠标悬停预加载的策略
        window.addEventListener('scroll', debouncedScrollHandlerPC);
        setupMouseHoverPreload();
        addDraggableIcon();
        document.getElementById('draggableIcon').onclick = handleBackNavigation;
    }


    document.querySelectorAll('#returnsetKey, #directsetKey').forEach(function (element) {
        element.addEventListener('click', function () {
            var keySequence = [];
            var targetInputElementId = element.id === 'returnsetKey' ? 'returnShortcutKey' : 'directShortcutKey'; // 根据点击的按钮id来决定需要改变哪个输入元素
            var targetInputElement = document.querySelector('#' + targetInputElementId);

            targetInputElement.value = '请按下快捷键...';
            targetInputElement.disabled = false;

            var keyDownEvent = function (event) {
                event.preventDefault();
                var key = event.key.toLowerCase();
                if (!keySequence.includes(key)) {
                    keySequence.push(key);
                    targetInputElement.value = keySequence.join("+").toUpperCase();
                }
            };

            var keyUpEvent = function () {
                document.removeEventListener('keydown', keyDownEvent);
                targetInputElement.disabled = true;
                GM_setValue(targetInputElementId, targetInputElement.value);
                document.removeEventListener('keyup', keyUpEvent);
            };

            document.addEventListener('keydown', keyDownEvent);
            document.addEventListener('keyup', keyUpEvent);
        });
    });

})();