- // ==UserScript==
- // @name 网页加速器
- // @namespace https://github.com/Gao-Lezhe/instantpage
- // @version 1.0.3
- // @author YouXiaoHou、高乐喆(Greasy Fork镜像用户名:lezhe)
- // @description 自动帮你加速网页中的超链接,加快打开网页的速度,实测符合条件的网页打开速度减少50%以上。
- // @license AGPL
- // @supportURL https://github.com/Gao-Lezhe/instantpage
- // @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==
-
- (function () {
- 'use strict';
-
- let util = {
- getValue(name) {
- return GM_getValue(name);
- },
-
- setValue(name, value) {
- GM_setValue(name, value);
- },
-
- include(str, arr) {
- str = str.replace(/[-_]/ig, '');
- for (let i = 0, l = arr.length; i < l; i++) {
- let val = arr[i];
- 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);
- },
-
- 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,})(?=[\/#?]|$)/,
- }
- };
-
- let main = {
- initValue() {
- let value = [{
- name: 'setting_success_times',
- value: 0
- }, {
- name: 'allow_external_links',
- value: true
- }, {
- name: 'allow_query_links',
- value: true
- }, {
- name: 'enable_store_link',
- 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) => {
- util.getValue(v.name) === undefined && util.setValue(v.name, v.value);
- });
- },
-
- registerMenuCommand() {
- GM_registerMenuCommand('🚀 已加速:' + util.getValue('setting_success_times') + '次', () => {
- Swal.fire({
- showCancelButton: true,
- title: '确定要重置加速次数吗?',
- icon: 'warning',
- confirmButtonText: '确定',
- cancelButtonText: '取消',
- customClass: {
- popup: 'instant-popup',
- },
- }).then((res) => {
- if (res.isConfirmed) {
- util.setValue('setting_success_times', 0);
- history.go(0);
- }
- });
- });
- GM_registerMenuCommand('⚙️ 设置', () => {
- let dom = `<div style="font-size: 1em;">
- <label class="instant-setting-label">加速外部链接<input type="checkbox" id="S-External" ${util.getValue('allow_external_links') ? 'checked' : ''} class="instant-setting-checkbox"></label>
- <label class="instant-setting-label"><span>加速含参数链接 <a href="https://www.youxiaohou.com/tool/install-instantpage.html#%E9%85%8D%E7%BD%AE%E8%AF%B4%E6%98%8E">详见</a></span><input type="checkbox" id="S-Query" ${util.getValue('allow_query_links') ? 'checked' : ''}
- class="instant-setting-checkbox"></label>
- <label class="instant-setting-label">加速扩展商店链接<input type="checkbox" id="S-Store" ${util.getValue('enable_store_link') ? 'checked' : ''} class="instant-setting-checkbox"></label>
- <label class="instant-setting-label">加速链接在当前页打开<input type="checkbox" id="S-Target" ${util.getValue('enable_target_self') ? 'checked' : ''} class="instant-setting-checkbox"></label>
- <label class="instant-setting-label">加速动画效果<input type="checkbox" id="S-Animate" ${util.getValue('enable_animation') ? 'checked' : ''}
- class="instant-setting-checkbox"></label>
- <label class="instant-setting-label">链接预读延时(毫秒)<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">排除下列网址 <textarea placeholder="列表中的域名将不开启加速器,一行一个,例如:www.baidu.com" id="S-Exclude" class="instant-setting-textarea">${util.getValue('exclude_list')}</textarea></label>
- <label class="instant-setting-label-col">排除下列关键词 <textarea placeholder="链接中含关键词将不开启加速器,一行一个,例如:logout" id="S-Exclude-Word" class="instant-setting-textarea">${util.getValue('exclude_keyword')}</textarea></label>
- </div>`;
- Swal.fire({
- title: '加速器配置',
- html: dom,
- showCloseButton: true,
- confirmButtonText: '保存',
- footer: '<div style="text-align: center;font-size: 1em;">点击查看 <a href="https://www.youxiaohou.com/tool/install-instantpage.html" target="_blank">使用说明</a>,助手免费开源,Powered by <a href="https://www.youxiaohou.com">油小猴</a></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-Store').addEventListener('change', (e) => {
- util.setValue('enable_store_link', 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);
- });
- });
- },
-
- //在排除名单里
- inExcludeList() {
- let exclude = util.getValue('exclude_list').split('\n');
- let host = location.host;
- return exclude.includes(host);
- },
-
- //加速主代码
- instantPage() {
- if (window.instantLoaded) return;
- let mouseoverTimer;
- let lastTouchTimestamp;
- const prefetches = new Set();
- const prefetchElement = document.createElement('link');
- const isSupported = prefetchElement.relList && prefetchElement.relList.supports && prefetchElement.relList.supports('prefetch')
- && window.IntersectionObserver && 'isIntersecting' in IntersectionObserverEntry.prototype;
- const isOnline = () => window.navigator.onLine;
- 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 mousedownShortcut = 'instantMousedownShortcut' in document.body.dataset;
- const DELAY_TO_NOT_BE_CONSIDERED_A_TOUCH_INITIATED_ACTION = 1111;
- const enableAnimation = util.getValue('enable_animation');
- const enableTargetSelf = util.getValue('enable_target_self');
- const enableStoreLink = util.getValue('enable_store_link');
- window.instantLoaded = true;
- const excludeKeyword = util.getValue('exclude_keyword').split('\n');
-
- let delayOnHover = util.getValue('delay_on_hover');
- let useMousedown = false;
- let useMousedownOnly = false;
- let useViewport = false;
-
- if ('instantIntensity' in document.body.dataset) {
- const intensity = document.body.dataset.instantIntensity;
-
- if (intensity.substr(0, 'mousedown'.length) === 'mousedown') {
- useMousedown = true;
- if (intensity === 'mousedown-only') {
- useMousedownOnly = true;
- }
- } else if (intensity.substr(0, 'viewport'.length) === 'viewport') {
- if (!(navigator.connection && (navigator.connection.saveData || (navigator.connection.effectiveType && navigator.connection.effectiveType.includes('2g'))))) {
- if (intensity === "viewport") {
- if (document.documentElement.clientWidth * document.documentElement.clientHeight < 450000) {
- useViewport = true;
- }
- } else if (intensity === "viewport-all") {
- useViewport = true;
- }
- }
- } else {
- const milliseconds = parseInt(intensity);
- if (!Number.isNaN(milliseconds)) {
- delayOnHover = milliseconds;
- }
- }
- }
-
- if (isSupported) {
- const eventListenersOptions = {
- capture: true,
- passive: true,
- };
-
- if (!useMousedownOnly) {
- document.addEventListener('touchstart', touchstartListener, eventListenersOptions);
- }
-
- if (!useMousedown) {
- document.addEventListener('mouseover', mouseoverListener, eventListenersOptions);
- } else if (!mousedownShortcut) {
- document.addEventListener('mousedown', mousedownListener, eventListenersOptions);
- }
-
- if (mousedownShortcut) {
- document.addEventListener('mousedown', mousedownShortcutListener, eventListenersOptions);
- }
-
-
- if (useViewport) {
- let triggeringFunction;
- if (window.requestIdleCallback) {
- triggeringFunction = (callback) => {
- requestIdleCallback(callback, {
- timeout: 1500,
- });
- };
- } else {
- triggeringFunction = (callback) => {
- callback();
- };
- }
-
- triggeringFunction(() => {
- const intersectionObserver = new IntersectionObserver((entries) => {
- entries.forEach((entry) => {
- if (entry.isIntersecting) {
- const linkElement = entry.target;
- intersectionObserver.unobserve(linkElement);
- preload(linkElement);
- }
- });
- });
-
- document.querySelectorAll('a').forEach((linkElement) => {
- if (isPreloadable(linkElement)) {
- intersectionObserver.observe(linkElement);
- }
- });
- });
- }
- }
-
- function touchstartListener(event) {
- /* Chrome on Android calls mouseover before touchcancel so `lastTouchTimestamp`
- * must be assigned on touchstart to be measured on mouseover. */
- lastTouchTimestamp = performance.now();
-
- const linkElement = event.target.closest('a');
-
- if (!isPreloadable(linkElement)) {
- return;
- }
-
- preload(linkElement);
- }
-
- function mouseoverListener(event) {
- if (performance.now() - lastTouchTimestamp < DELAY_TO_NOT_BE_CONSIDERED_A_TOUCH_INITIATED_ACTION) {
- return;
- }
-
- if (!('closest' in event.target)) {
- // Without this check sometimes an error “event.target.closest is not a function” is thrown, for unknown reasons
- // That error denotes that `event.target` isn’t undefined. My best guess is that it’s the Document.
-
- // Details could be gleaned from throwing such an error:
- //throw new TypeError(`instant.page non-element event target: timeStamp=${~~event.timeStamp}, type=${event.type}, typeof=${typeof event.target}, nodeType=${event.target.nodeType}, nodeName=${event.target.nodeName}, viewport=${innerWidth}x${innerHeight}, coords=${event.clientX}x${event.clientY}, scroll=${scrollX}x${scrollY}`)
- return
- }
-
- const linkElement = event.target.closest('a');
-
- if (!isPreloadable(linkElement)) {
- return;
- }
-
- linkElement.addEventListener('mouseout', mouseoutListener, {passive: true});
-
- mouseoverTimer = setTimeout(() => {
- preload(linkElement);
- mouseoverTimer = undefined;
- }, delayOnHover);
- }
-
- function mousedownListener(event) {
- const linkElement = event.target.closest('a');
-
- if (!isPreloadable(linkElement)) {
- return;
- }
-
- preload(linkElement);
- }
-
- function mouseoutListener(event) {
- if (event.relatedTarget && event.target.closest('a') === event.relatedTarget.closest('a')) {
- return;
- }
-
- if (mouseoverTimer) {
- clearTimeout(mouseoverTimer);
- mouseoverTimer = undefined;
- }
- }
-
- function mousedownShortcutListener(event) {
- if (performance.now() - lastTouchTimestamp < DELAY_TO_NOT_BE_CONSIDERED_A_TOUCH_INITIATED_ACTION) {
- return;
- }
-
- const linkElement = event.target.closest('a');
-
- if (event.which > 1 || event.metaKey || event.ctrlKey) {
- return;
- }
-
- if (!linkElement) {
- return;
- }
-
- linkElement.addEventListener('click', function (event) {
- if (event.detail === 1337) {
- return;
- }
-
- event.preventDefault();
- }, {capture: true, passive: false, once: true});
-
- const customEvent = new MouseEvent('click', {
- view: window,
- bubbles: true,
- cancelable: true,
- detail: 1337
- });
- linkElement.dispatchEvent(customEvent);
- }
-
- function isPreloadable(linkElement) {
- if (!linkElement || !linkElement.href) {
- return;
- }
-
- if (util.include(linkElement.href, excludeKeyword)) {
- if (!util.reg.chrome.test(linkElement.href) &&
- !util.reg.chromeNew.test(linkElement.href) &&
- !util.reg.edge.test(linkElement.href) &&
- !util.reg.edge.test(linkElement.href) &&
- !util.reg.microsoft.test(linkElement.href)) {
- return;
- }
- }
-
- if (useWhitelist && !('instant' in linkElement.dataset)) {
- return;
- }
-
- if (!allowExternalLinks && linkElement.origin !== location.origin && !('instant' in linkElement.dataset)) {
- return;
- }
-
- if (!['http:', 'https:'].includes(linkElement.protocol)) {
- return;
- }
-
- if (linkElement.protocol === 'http:' && location.protocol === 'https:') {
- if (linkElement.href.indexOf('http://www.baidu.com/link?url') === 0) {
- linkElement.href = linkElement.href.replace('http', 'https');
- } else {
- return;
- }
- }
- //下载文件不加速
- if (/\.[a-zA-Z0-9]{0,5}$/i.test(linkElement.href)) {
- //排除域名,网站扩展名
- if (!/(com|cn|top|ltd|net|tech|shop|vip|xyz|wang|cloud|online|site|love|art|xin|store|fun|cc|website|press|space|beer|luxe|video|ren|group|fit|yoga|org|pro|ink|biz|info|design|link|work|mobi|kim|pub|name|tv|co|asia|red|live|wiki|gov|life|world|run|show|city|gold|today|plus|cool|icu|company|chat|zone|fans|law|host|center|club|email|fund|social|team|guru|htm|html|php|asp|jsp)$/i.test(linkElement.href)) {
- return;
- }
- }
-
- if (!allowQueryString && linkElement.search && !('instant' in linkElement.dataset)) {
- return;
- }
-
- if (linkElement.hash && linkElement.pathname + linkElement.search === location.pathname + location.search) {
- return;
- }
-
- if (linkElement.dataset.filename || linkElement.dataset.noInstant) {
- return;
- }
-
- return true;
- }
-
- function preload(linkElement) {
- let url = linkElement.href;
-
- if (!isOnline()) {
- return;
- }
-
- if (prefetches.has(url)) {
- return;
- }
-
- if (enableStoreLink) {
- if (util.reg.chromeNew.test(url)) {
- linkElement.href = url.replace("chromewebstore.google.com", "chrome.crxsoso.com/webstore");
- }
- if (util.reg.edge.test(url)) {
- linkElement.href = url.replace("microsoftedge.microsoft.com", "microsoftedge.crxsoso.com");
- }
- if (util.reg.firefox.test(url)) {
- linkElement.href = url.replace("addons.mozilla.org", "addons.crxsoso.com");
- }
- }
-
- const prefetcher = document.createElement('link');
- prefetcher.rel = 'prefetch';
- prefetcher.href = url;
- document.head.appendChild(prefetcher);
-
- prefetches.add(url);
-
- if (enableAnimation) {
- linkElement.classList.add("link-instanted");
- }
- if (enableTargetSelf) {
- linkElement.target = '_self';
- }
-
- util.setValue('setting_success_times', util.getValue('setting_success_times') + 1);
- }
- },
-
- 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);
- }
-
- 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});
- },
-
- init() {
- this.initValue();
- this.addPluginStyle();
- this.registerMenuCommand();
- if (this.inExcludeList()) return;
- this.instantPage();
- }
- };
- main.init();
- })();