// ==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();
})();