Greasy Fork is available in English.

Web Page Accelerator

Automatically accelerates hyperlinks on web pages to improve loading speed. Integrates the latest instant.page v5.2.0 features with multi-language support (default: English) and removes store link redirection functionality.

// ==UserScript==
// @name              Web Page Accelerator
// @namespace      instant.page
// @version           v1.0.5.2.0
// @author            OB_BUFF
// @description       Automatically accelerates hyperlinks on web pages to improve loading speed. Integrates the latest instant.page v5.2.0 features with multi-language support (default: English) and removes store link redirection functionality.
// @license           GPL-v3
// @require           https://registry.npmmirror.com/sweetalert2/10.16.6/files/dist/sweetalert2.min.js
// @resource          swalStyle https://registry.npmmirror.com/sweetalert2/10.16.6/files/dist/sweetalert2.min.css
// @match             *://*/*
// @noframes
// @run-at            document-idle
// @grant             GM_openInTab
// @grant             GM_setValue
// @grant             GM_getValue
// @grant             GM_registerMenuCommand
// @grant             GM_getResourceText
// @icon              
// ==/UserScript==

/* -------------------------------
   Multi-language support (default: English)
   You can extend the translations object to support more languages.
---------------------------------- */
const translations = {
    en: {
        acceleratedCount: "Accelerated: ",
        times: " times",
        resetPrompt: "Are you sure you want to reset the acceleration count?",
        confirm: "OK",
        cancel: "Cancel",
        settingsTitle: "Accelerator Settings",
        accelerateExternal: "Accelerate external links",
        accelerateParams: "Accelerate links with parameters",
        openInSameTab: "Open links in the same tab",
        animationEffect: "Animation effect",
        prefetchDelay: "Prefetch delay (ms)",
        excludeURLs: "Exclude the following URLs (one per line)",
        excludeKeywords: "Exclude the following keywords (one per line)",
        save: "Save",
        usageFooter: "Click here to view the usage instructions. This assistant is free and open-source."
    }
};
const lang = 'en'; // Default language

// -------------------------------
// Utility functions
// -------------------------------
let util = {
    getValue(name) {
        return GM_getValue(name);
    },
    setValue(name, value) {
        GM_setValue(name, value);
    },
    // Check if the given string (after removing '-' and '_') contains any of the keywords (case-insensitive)
    include(str, arr) {
        str = str.replace(/[-_]/ig, '');
        for (let i = 0, l = arr.length; i < l; i++) {
            let val = arr[i].trim();
            if (val !== '' && str.toLowerCase().indexOf(val.toLowerCase()) > -1) {
                return true;
            }
        }
        return false;
    },
    addStyle(id, tag, css) {
        tag = tag || 'style';
        let doc = document, styleDom = doc.getElementById(id);
        if (styleDom) return;
        let style = doc.createElement(tag);
        style.rel = 'stylesheet';
        style.id = id;
        tag === 'style' ? style.innerHTML = css : style.href = css;
        doc.head.appendChild(style);
    },
    // Common regex patterns
    reg: {
        chrome: /^https?:\/\/chrome\.google\.com\/webstore\/.+?\/([a-z]{32})(?=[\/#?]|$)/,
        chromeNew: /^https?:\/\/chromewebstore\.google\.com\/.+?\/([a-z]{32})(?=[\/#?]|$)/,
        edge: /^https?:\/\/microsoftedge\.microsoft\.com\/addons\/.+?\/([a-z]{32})(?=[\/#?]|$)/,
        firefox: /^https?:\/\/(reviewers\.)?(addons\.mozilla\.org|addons(?:-dev)?\.allizom\.org)\/.*?(?:addon|review)\/([^/<>"'?#]+)/,
        microsoft: /^https?:\/\/(?:apps|www)\.microsoft\.com\/(?:store|p)\/.+?\/([a-zA-Z\d]{10,})(?=[\/#?]|$)/,
    }
};

// -------------------------------
// Main logic
// -------------------------------
let main = {
    // Initialize configuration values (store values using GM storage)
    initValue() {
        // Note: The "store link" redirection option has been removed.
        let value = [{
            name: 'setting_success_times',
            value: 0
        }, {
            name: 'allow_external_links',
            value: true
        }, {
            name: 'allow_query_links',
            value: true
        }, {
            name: 'enable_target_self',
            value: false
        }, {
            name: 'enable_animation',
            value: false
        }, {
            name: 'delay_on_hover',
            value: 65
        }, {
            name: 'exclude_list',
            value: ''
        }, {
            name: 'exclude_keyword',
            value: 'login\nlogout\nregister\nsignin\nsignup\nsignout\npay\ncreate\nedit\ndownload\ndel\nreset\nsubmit\ndoubleclick\ngoogleads\nexit'
        }];
        value.forEach((v) => {
            if (util.getValue(v.name) === undefined) {
                util.setValue(v.name, v.value);
            }
        });
    },

    // Register menu commands for the settings panel and reset function
    registerMenuCommand() {
        GM_registerMenuCommand(
            "🚀 " + translations[lang].acceleratedCount + util.getValue('setting_success_times') + translations[lang].times,
            () => {
                Swal.fire({
                    showCancelButton: true,
                    title: translations[lang].resetPrompt,
                    icon: 'warning',
                    confirmButtonText: translations[lang].confirm,
                    cancelButtonText: translations[lang].cancel,
                    customClass: {
                        popup: 'instant-popup',
                    },
                }).then((res) => {
                    if (res.isConfirmed) {
                        util.setValue('setting_success_times', 0);
                        history.go(0);
                    }
                });
            }
        );
        let dom = `<div style="font-size: 1em;">
                      <label class="instant-setting-label">${translations[lang].accelerateExternal}<input type="checkbox" id="S-External" ${util.getValue('allow_external_links') ? 'checked' : ''} class="instant-setting-checkbox"></label>
                      <label class="instant-setting-label"><span>${translations[lang].accelerateParams} (<a href="https://www.youxiaohou.com/tool/install-instantpage.html#%E9%85%8D%E7%BD%AE%E8%AF%B4%E6%98%8E" target="_blank">Details</a>)</span><input type="checkbox" id="S-Query" ${util.getValue('allow_query_links') ? 'checked' : ''} class="instant-setting-checkbox"></label>
                      <label class="instant-setting-label">${translations[lang].openInSameTab}<input type="checkbox" id="S-Target" ${util.getValue('enable_target_self') ? 'checked' : ''} class="instant-setting-checkbox"></label>
                      <label class="instant-setting-label">${translations[lang].animationEffect}<input type="checkbox" id="S-Animate" ${util.getValue('enable_animation') ? 'checked' : ''} class="instant-setting-checkbox"></label>
                      <label class="instant-setting-label">${translations[lang].prefetchDelay}<input type="number" min="65" id="S-Delay" value="${util.getValue('delay_on_hover')}" class="instant-setting-input"></label>
                      <label class="instant-setting-label-col">${translations[lang].excludeURLs}<textarea placeholder="One per line, e.g., www.example.com" id="S-Exclude" class="instant-setting-textarea">${util.getValue('exclude_list')}</textarea></label>
                      <label class="instant-setting-label-col">${translations[lang].excludeKeywords}<textarea placeholder="One per line, e.g., logout" id="S-Exclude-Word" class="instant-setting-textarea">${util.getValue('exclude_keyword')}</textarea></label>
                    </div>`;
        GM_registerMenuCommand(translations[lang].settingsTitle, () => {
            Swal.fire({
                title: translations[lang].settingsTitle,
                html: dom,
                showCloseButton: true,
                confirmButtonText: translations[lang].save,
                footer: `<div style="text-align: center;font-size: 1em;">${translations[lang].usageFooter}</div>`,
                customClass: {
                    popup: 'instant-popup',
                },
            }).then((res) => {
                if (res.isConfirmed) {
                    history.go(0);
                }
            });
            document.getElementById('S-External').addEventListener('change', (e) => {
                util.setValue('allow_external_links', e.currentTarget.checked);
            });
            document.getElementById('S-Query').addEventListener('change', (e) => {
                util.setValue('allow_query_links', e.currentTarget.checked);
            });
            document.getElementById('S-Target').addEventListener('change', (e) => {
                util.setValue('enable_target_self', e.currentTarget.checked);
            });
            document.getElementById('S-Animate').addEventListener('change', (e) => {
                util.setValue('enable_animation', e.currentTarget.checked);
            });
            document.getElementById('S-Delay').addEventListener('change', (e) => {
                util.setValue('delay_on_hover', e.currentTarget.value);
            });
            document.getElementById('S-Exclude').addEventListener('change', (e) => {
                util.setValue('exclude_list', e.currentTarget.value);
            });
            document.getElementById('S-Exclude-Word').addEventListener('change', (e) => {
                util.setValue('exclude_keyword', e.currentTarget.value);
            });
        });
    },

    // Check if the current host is in the exclude list
    inExcludeList() {
        let exclude = util.getValue('exclude_list').split('\n').map(s => s.trim());
        let host = location.host;
        return exclude.includes(host);
    },

    // -------------------------------
    // Main prefetch logic integrating instant.page v5.2.0 features
    // -------------------------------
    instantPage() {
        if (window.instantLoaded) return;
        window.instantLoaded = true;

        // Configuration options (some from GM storage, some from data attributes)
        const allowQueryString = ('instantAllowQueryString' in document.body.dataset) || util.getValue('allow_query_links');
        const allowExternalLinks = ('instantAllowExternalLinks' in document.body.dataset) || util.getValue('allow_external_links');
        const _useWhitelist = ('instantWhitelist' in document.body.dataset);
        const enableAnimation = util.getValue('enable_animation');
        const enableTargetSelf = util.getValue('enable_target_self');
        const excludeKeyword = util.getValue('exclude_keyword').split('\n');
        let delayOnHover = parseInt(util.getValue('delay_on_hover'));

        // Internal variables similar to instant.page v5.2.0
        let _chromiumMajorVersionInUserAgent = null;
        let _speculationRulesType = 'none';
        let _delayOnHover = delayOnHover; // in milliseconds
        let _lastTouchstartEvent = null;
        let _mouseoverTimer = null;
        let _preloadedList = new Set();

        // Browser support check: ensure <link rel="prefetch"> is supported
        let supportChecksRelList = document.createElement('link').relList;
        if (!(supportChecksRelList && supportChecksRelList.supports && supportChecksRelList.supports('prefetch'))) {
            return;
        }
        const chromium100Check = ('throwIfAborted' in AbortSignal.prototype); // Chromium 100+
        const firefox115AndSafari17_0Check = supportChecksRelList.supports('modulepreload'); // Firefox 115+, Safari 17.0+
        const safari15_4AndFirefox116Check = (Intl.PluralRules && 'selectRange' in Intl.PluralRules.prototype);
        const firefox115AndSafari15_4Check = firefox115AndSafari17_0Check || safari15_4AndFirefox116Check;
        const isBrowserSupported = chromium100Check && firefox115AndSafari15_4Check;
        if (!isBrowserSupported) return;

        // If the page sets data-instantVaryAccept (e.g. Shopify), check Chromium version
        if (document.body.dataset.instantVaryAccept) {
            const chromiumUserAgentIndex = navigator.userAgent.indexOf('Chrome/');
            if (chromiumUserAgentIndex > -1) {
                _chromiumMajorVersionInUserAgent = parseInt(navigator.userAgent.substring(chromiumUserAgentIndex + 'Chrome/'.length));
            }
            if (_chromiumMajorVersionInUserAgent && _chromiumMajorVersionInUserAgent < 110) {
                return;
            }
        }

        // Set speculation rules if supported (<script type="speculationrules">)
        if (HTMLScriptElement.supports && HTMLScriptElement.supports('speculationrules')) {
            const speculationRulesConfig = document.body.dataset.instantSpecrules;
            if (speculationRulesConfig === 'prerender') {
                _speculationRulesType = 'prerender';
            } else if (speculationRulesConfig !== 'no') {
                _speculationRulesType = 'prefetch';
            }
        }

        // Determine trigger method based on data-instantIntensity (default is mouseover)
        let useMousedown = false;
        let useMousedownOnly = false;
        let useViewport = false;
        const mousedownShortcut = ('instantMousedownShortcut' in document.body.dataset);
        if ('instantIntensity' in document.body.dataset) {
            const intensity = document.body.dataset.instantIntensity;
            if (intensity === 'mousedown' && !mousedownShortcut) {
                useMousedown = true;
            }
            if (intensity === 'mousedown-only' && !mousedownShortcut) {
                useMousedown = true;
                useMousedownOnly = true;
            }
            if (intensity === 'viewport' || intensity === 'viewport-all') {
                const isOnSmallScreen = document.documentElement.clientWidth * document.documentElement.clientHeight < 450000;
                const isConnectionAdequate = !(navigator.connection && (navigator.connection.saveData ||
                    (navigator.connection.effectiveType && navigator.connection.effectiveType.includes('2g'))));
                if (isOnSmallScreen && isConnectionAdequate) {
                    useViewport = true;
                }
                if (intensity === 'viewport-all') {
                    useViewport = true;
                }
            }
            const intensityAsInteger = parseInt(intensity);
            if (!isNaN(intensityAsInteger)) {
                _delayOnHover = intensityAsInteger;
            }
        }

        const eventListenersOptions = {
            capture: true,
            passive: true,
        };

        // Register event listeners based on trigger method
        if (useMousedownOnly) {
            document.addEventListener('touchstart', touchstartEmptyListener, eventListenersOptions);
        } else {
            document.addEventListener('touchstart', touchstartListener, eventListenersOptions);
        }
        if (!useMousedown) {
            document.addEventListener('mouseover', mouseoverListener, eventListenersOptions);
        }
        if (useMousedown) {
            document.addEventListener('mousedown', mousedownListener, eventListenersOptions);
        }
        if (mousedownShortcut) {
            document.addEventListener('mousedown', mousedownShortcutListener, eventListenersOptions);
        }

        // If viewport prefetch is enabled, use IntersectionObserver to preload links in view
        if (useViewport) {
            const requestIdleCallbackOrFallback = window.requestIdleCallback || function(callback) { callback(); };
            requestIdleCallbackOrFallback(function () {
                const intersectionObserver = new IntersectionObserver((entries) => {
                    entries.forEach((entry) => {
                        if (entry.isIntersecting) {
                            const anchor = entry.target;
                            intersectionObserver.unobserve(anchor);
                            preload(anchor);
                        }
                    });
                });
                document.querySelectorAll('a').forEach((anchor) => {
                    if (isPreloadable(anchor)) {
                        intersectionObserver.observe(anchor);
                    }
                });
            }, { timeout: 1500 });
        }

        // -------------------------------
        // Event handler functions
        // -------------------------------
        function touchstartListener(event) {
            _lastTouchstartEvent = event;
            const anchor = event.target.closest('a');
            if (!isPreloadable(anchor)) return;
            preload(anchor, 'high');
        }
        function touchstartEmptyListener(event) {
            _lastTouchstartEvent = event;
        }
        function mouseoverListener(event) {
            if (isEventLikelyTriggeredByTouch(event)) return;
            if (!('closest' in event.target)) return;
            const anchor = event.target.closest('a');
            if (!isPreloadable(anchor)) return;
            anchor.addEventListener('mouseout', mouseoutListener, { passive: true });
            _mouseoverTimer = setTimeout(() => {
                preload(anchor, 'high');
                _mouseoverTimer = null;
            }, _delayOnHover);
        }
        function mousedownListener(event) {
            if (isEventLikelyTriggeredByTouch(event)) return;
            const anchor = event.target.closest('a');
            if (!isPreloadable(anchor)) return;
            preload(anchor, 'high');
        }
        function mouseoutListener(event) {
            if (event.relatedTarget && event.target.closest('a') === event.relatedTarget.closest('a')) return;
            if (_mouseoverTimer) {
                clearTimeout(_mouseoverTimer);
                _mouseoverTimer = null;
            }
        }
        function mousedownShortcutListener(event) {
            if (isEventLikelyTriggeredByTouch(event)) return;
            const anchor = event.target.closest('a');
            if (event.which > 1 || event.metaKey || event.ctrlKey) return;
            if (!anchor) return;
            anchor.addEventListener('click', function (e) {
                if (e.detail === 1337) return;
                e.preventDefault();
            }, { capture: true, passive: false, once: true });
            const customEvent = new MouseEvent('click', {
                view: window,
                bubbles: true,
                cancelable: false,
                detail: 1337
            });
            anchor.dispatchEvent(customEvent);
        }
        // Check if a mouse event is likely triggered by a preceding touch event (avoid duplicate prefetch on touch devices)
        function isEventLikelyTriggeredByTouch(event) {
            if (!_lastTouchstartEvent || !event) return false;
            if (event.target !== _lastTouchstartEvent.target) return false;
            const now = event.timeStamp;
            const duration = now - _lastTouchstartEvent.timeStamp;
            const MAX_DURATION = 2500; // ms
            return duration < MAX_DURATION;
        }
        // Determine whether the link element is preloadable based on various criteria
        function isPreloadable(anchor) {
            if (!anchor || !anchor.href) return false;
            if (_useWhitelist && !('instant' in anchor.dataset)) return false;
            if (anchor.origin !== location.origin) {
                const allowed = allowExternalLinks || ('instant' in anchor.dataset);
                if (!allowed) return false;
            }
            if (!['http:', 'https:'].includes(anchor.protocol)) return false;
            if (anchor.protocol === 'http:' && location.protocol === 'https:') return false;
            if (!allowQueryString && anchor.search && !('instant' in anchor.dataset)) return false;
            if (anchor.hash && (anchor.pathname + anchor.search === location.pathname + location.search)) return false;
            if ('noInstant' in anchor.dataset) return false;
            // Exclude links containing any of the specified keywords
            if (util.include(anchor.href, excludeKeyword)) return false;
            return true;
        }
        // Perform prefetch if the link has not been prefetched yet.
        // Choose between speculation rules (if supported) and <link rel="prefetch">
        function preload(anchor, fetchPriority = 'auto') {
            const url = anchor.href;
            if (_preloadedList.has(url)) return;
            if (_speculationRulesType !== 'none') {
                preloadUsingSpeculationRules(url);
            } else {
                preloadUsingLinkElement(url, fetchPriority);
            }
            _preloadedList.add(url);
            if (enableAnimation) {
                anchor.classList.add("link-instanted");
            }
            if (enableTargetSelf) {
                anchor.target = '_self';
            }
            util.setValue('setting_success_times', util.getValue('setting_success_times') + 1);
        }
        // Prefetch using <script type="speculationrules">
        function preloadUsingSpeculationRules(url) {
            const script = document.createElement('script');
            script.type = 'speculationrules';
            script.textContent = JSON.stringify({
                [_speculationRulesType]: [{
                    source: 'list',
                    urls: [url]
                }]
            });
            document.head.appendChild(script);
        }
        // Prefetch using <link rel="prefetch">
        function preloadUsingLinkElement(url, fetchPriority = 'auto') {
            const link = document.createElement('link');
            link.rel = 'prefetch';
            link.href = url;
            link.fetchPriority = fetchPriority;
            link.as = 'document';
            document.head.appendChild(link);
        }
    },

    // Add plugin styles to the page
    addPluginStyle() {
        let style = `
            .instant-popup { font-size: 14px !important; }
            .instant-setting-label { display: flex; align-items: center; justify-content: space-between; padding-top: 15px; }
            .instant-setting-label-col { display: flex; align-items: flex-start; padding-top: 15px; flex-direction: column; }
            .instant-setting-checkbox { width: 16px; height: 16px; }
            .instant-setting-textarea { width: 100%; margin: 14px 0 0; height: 60px; resize: none; border: 1px solid #bbb; box-sizing: border-box; padding: 5px 10px; border-radius: 5px; color: #666; line-height: 1.2; }
            .instant-setting-input { border: 1px solid #bbb; box-sizing: border-box; padding: 5px 10px; border-radius: 5px; width: 100px; }
            @keyframes instantAnminate { from { opacity: 1; } 50% { opacity: 0.4; } to { opacity: 0.9; } }
            .link-instanted { animation: instantAnminate 0.6s 1; animation-fill-mode: forwards; }
            .link-instanted * { animation: instantAnminate 0.6s 1; animation-fill-mode: forwards; }
        `;
        if (document.head) {
            util.addStyle('swal-pub-style', 'style', GM_getResourceText('swalStyle'));
            util.addStyle('instant-style', 'style', style);
        }
        // Observe changes in the head element and re-add styles if needed
        const headObserver = new MutationObserver(() => {
            util.addStyle('swal-pub-style', 'style', GM_getResourceText('swalStyle'));
            util.addStyle('instant-style', 'style', style);
        });
        headObserver.observe(document.head, { childList: true, subtree: true });
    },

    // Initialize the script
    init() {
        this.initValue();
        this.addPluginStyle();
        this.registerMenuCommand();
        if (this.inExcludeList()) return;
        this.instantPage();
    }
};

main.init();