您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
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();