- /* eslint-disable no-multi-spaces */
- /* eslint-disable no-useless-call */
-
- // ==UserScript==
- // @name SingleFile - 单文件保存网页
- // @name:en SingleFile - Webpage downloader
- // @name:en-US SingleFile - Webpage downloader
- // @name:en-UK SingleFile - Webpage downloader
- // @name:zh SingleFile - 单文件保存网页
- // @name:zh-CN SingleFile - 单文件保存网页
- // @name:zh-Hans SingleFile - 单文件保存网页
- // @name:zh-TW SingleFile - 單檔案保存網頁
- // @namespace SingleFile
- // @version 2.2
- // @description 将当前网页保存为一个.html网页文件
- // @description:en Save webpages into one .html file
- // @description:en-US Save webpages into one .html file
- // @description:en-UK Save webpages into one .html file
- // @description:zh 将当前网页保存为一个.html网页文件
- // @description:zh-CN 将当前网页保存为一个.html网页文件
- // @description:zh-Hans 将当前网页保存为一个.html网页文件
- // @description:zh-TW 將當前網頁保存為一個.html網頁檔案
- // @author PY-DNG
- // @license MIT
- // @include *
- // @connect *
- // @icon 
- // @grant GM_xmlhttpRequest
- // @grant GM_registerMenuCommand
- // @grant GM_unregisterMenuCommand
- // @grant GM_info
- // @noframes
- // ==/UserScript==
-
- (function() {
- 'use strict';
-
- // Arguments: level=LogLevel.Info, logContent, asObject=false
- // Needs one call "DoLog();" to get it initialized before using it!
- function DoLog() {
- // Global log levels set
- window.LogLevel = {
- None: 0,
- Error: 1,
- Success: 2,
- Warning: 3,
- Info: 4,
- }
- window.LogLevelMap = {};
- window.LogLevelMap[LogLevel.None] = {prefix: '' , color: 'color:#ffffff'}
- window.LogLevelMap[LogLevel.Error] = {prefix: '[Error]' , color: 'color:#ff0000'}
- window.LogLevelMap[LogLevel.Success] = {prefix: '[Success]' , color: 'color:#00aa00'}
- window.LogLevelMap[LogLevel.Warning] = {prefix: '[Warning]' , color: 'color:#ffa500'}
- window.LogLevelMap[LogLevel.Info] = {prefix: '[Info]' , color: 'color:#888888'}
- window.LogLevelMap[LogLevel.Elements] = {prefix: '[Elements]', color: 'color:#000000'}
-
- // Current log level
- DoLog.logLevel = (unsafeWindow ? unsafeWindow.isPY_DNG : window.isPY_DNG) ? LogLevel.Info : LogLevel.Warning; // Info Warning Success Error
-
- // Log counter
- DoLog.logCount === undefined && (DoLog.logCount = 0);
- if (++DoLog.logCount > 512) {
- console.clear();
- DoLog.logCount = 0;
- }
-
- // Get args
- let level, logContent, asObject;
- switch (arguments.length) {
- case 1:
- level = LogLevel.Info;
- logContent = arguments[0];
- asObject = false;
- break;
- case 2:
- level = arguments[0];
- logContent = arguments[1];
- asObject = false;
- break;
- case 3:
- level = arguments[0];
- logContent = arguments[1];
- asObject = arguments[2];
- break;
- default:
- level = LogLevel.Info;
- logContent = 'DoLog initialized.';
- asObject = false;
- break;
- }
-
- // Log when log level permits
- if (level <= DoLog.logLevel) {
- let msg = '%c' + LogLevelMap[level].prefix;
- let subst = LogLevelMap[level].color;
-
- if (asObject) {
- msg += ' %o';
- } else {
- switch(typeof(logContent)) {
- case 'string': msg += ' %s'; break;
- case 'number': msg += ' %d'; break;
- case 'object': msg += ' %o'; break;
- }
- }
-
- console.log(msg, subst, logContent);
- }
- }
- DoLog();
-
- bypassXB();
- GM_PolyFill('default');
-
- // Inner consts with i18n
- const CONST = {
- Number: {
- Max_XHR: 20,
- MaxUrlLength: 4096
- },
- Text: {
- 'zh-CN': {
- SavePage: '保存此网页',
- Saving: '保存中{A}',
- About: '<!-- Web Page Saved By {SCNM} Ver.{VRSN}, Author {ATNM} -->\n<!-- Page URL: {LINK} -->'
- .replaceAll('{SCNM}', GM_info.script.name)
- .replaceAll('{VRSN}', GM_info.script.version)
- .replaceAll('{ATNM}', GM_info.script.author)
- .replaceAll('{LINK}', location.href)
- },
- 'zh-Hans': {
- SavePage: '保存此网页',
- Saving: '保存中{A}',
- About: '<!-- Web Page Saved By {SCNM} Ver.{VRSN}, Author {ATNM} -->\n<!-- Page URL: {LINK} -->'
- .replaceAll('{SCNM}', GM_info.script.name)
- .replaceAll('{VRSN}', GM_info.script.version)
- .replaceAll('{ATNM}', GM_info.script.author)
- .replaceAll('{LINK}', location.href)
- },
- 'zh': {
- SavePage: '保存此网页',
- Saving: '保存中{A}',
- About: '<!-- Web Page Saved By {SCNM} Ver.{VRSN}, Author {ATNM} -->\n<!-- Page URL: {LINK} -->'
- .replaceAll('{SCNM}', GM_info.script.name)
- .replaceAll('{VRSN}', GM_info.script.version)
- .replaceAll('{ATNM}', GM_info.script.author)
- .replaceAll('{LINK}', location.href)
- },
- 'zh-TW': {
- SavePage: '保存此網頁',
- Saving: '保存中{A}',
- About: '<!-- Web Page Saved By {SCNM} Ver.{VRSN}, Author {ATNM} -->\n<!-- Page URL: {LINK} -->'
- .replaceAll('{SCNM}', GM_info.script.name)
- .replaceAll('{VRSN}', GM_info.script.version)
- .replaceAll('{ATNM}', GM_info.script.author)
- .replaceAll('{LINK}', location.href)
- },
- 'en-US': {
- SavePage: 'Save this webpage',
- Saving: 'Saving, please wait{A}',
- About: '<!-- Web Page Saved By {SCNM} Ver.{VRSN}, Author {ATNM} -->\n<!-- Page URL: {LINK} -->'
- .replaceAll('{SCNM}', GM_info.script.name)
- .replaceAll('{VRSN}', GM_info.script.version)
- .replaceAll('{ATNM}', GM_info.script.author)
- .replaceAll('{LINK}', location.href)
- },
- 'en-UK': {
- SavePage: 'Save this webpage',
- Saving: 'Saving, please wait{A}',
- About: '<!-- Web Page Saved By {SCNM} Ver.{VRSN}, Author {ATNM} -->\n<!-- Page URL: {LINK} -->'
- .replaceAll('{SCNM}', GM_info.script.name)
- .replaceAll('{VRSN}', GM_info.script.version)
- .replaceAll('{ATNM}', GM_info.script.author)
- .replaceAll('{LINK}', location.href)
- },
- 'en': {
- SavePage: 'Save this webpage',
- Saving: 'Saving, please wait{A}',
- About: '<!-- Web Page Saved By {SCNM} Ver.{VRSN}, Author {ATNM} -->\n<!-- Page URL: {LINK} -->'
- .replaceAll('{SCNM}', GM_info.script.name)
- .replaceAll('{VRSN}', GM_info.script.version)
- .replaceAll('{ATNM}', GM_info.script.author)
- .replaceAll('{LINK}', location.href)
- },
- 'default': {
- SavePage: 'Save this webpage',
- Saving: 'Saving, please wait{A}',
- About: '<!-- Web Page Saved By {SCNM} Ver.{VRSN}, Author {ATNM} -->\n<!-- Page URL: {LINK} -->'
- .replaceAll('{SCNM}', GM_info.script.name)
- .replaceAll('{VRSN}', GM_info.script.version)
- .replaceAll('{ATNM}', GM_info.script.author)
- .replaceAll('{LINK}', location.href)
- }
- }
- }
-
- // Get i18n code
- let i18n = navigator.language;
- if (!Object.keys(CONST.Text).includes(i18n)) {i18n = 'default';}
-
- // XHRHOOK
- GMXHRHook(CONST.Number.Max_XHR);
-
- main()
- function main() {
- // GUI
- let button = GM_registerMenuCommand(CONST.Text[i18n].SavePage, onclick);
- const SAnime = new SavingAnime;
- SAnime.model = CONST.Text[i18n].Saving;
- SAnime.callback = function(text) {
- GM_unregisterMenuCommand(button);
- button = GM_registerMenuCommand(text, () => {});
- }
-
- function onclick() {
- SAnime.start();
- Generate_Single_File({
- onfinish: (FinalHTML) => {
- saveTextToFile(FinalHTML, 'SingleFile - {Title} - {Time}.html'.replace('{Title}', document.title).replace('{Time}', getTime('-', '-')));
- GM_unregisterMenuCommand(button);
- SAnime.stop();
- button = GM_registerMenuCommand(CONST.Text[i18n].SavePage, onclick);
- }
- });
- }
-
- function SavingAnime() {
- const SA = this;
- SA.model = '{A}';
- SA.time = 1000;
- SA.index = 0;
- SA.frames = ['... ', ' ... ', ' ...', '. ..', '.. .'];
- SA.callback = (frametext) => {console.log(frametext);};
-
- SA.nextframe = function() {
- SA.index++;
- SA.index > SA.frames.length-1 && (SA.index = 0);
- SA.callback(SA.model.replace('{A}', SA.frames[SA.index]));
- return true;
- };
-
- SA.start = function() {
- if (SA.interval) {return false;}
- SA.index = 0;
- SA.interval = setInterval(SA.nextframe, SA.time);
- return true;
- }
-
- SA.stop = function() {
- if (!SA.interval) {return false;}
- clearInterval(SA.interval);
- SA.interval = 0;
- return true;
- }
- };
- }
-
- function Generate_Single_File(details) {
- // Init DOM
- const html = document.querySelector('html').outerHTML;
- const dom = (new DOMParser()).parseFromString(html, 'text/html');
-
- // Functions
- const _J = (args) => {const a = []; for (let i = 0; i < args.length; i++) {a.push(args[i]);}; return a;};
- const $ = function() {return dom.querySelector.apply(dom, _J(arguments))};
- const $_ = function() {return dom.querySelectorAll.apply(dom, _J(arguments))};
- const $C = function() {return dom.createElement.apply(dom, _J(arguments))};
- const $A = (a,b) => (a.appendChild(b));
- const $I = (a,b) => (b.parentElement ? b.parentElement.insertBefore(a, b) : null);
- const $R = (e) => (e.parentElement ? e.parentElement.removeChild(e) : null);
- const ishttp = (s) => (!/^[^\/:]*:/.test(s) || /^https?:\/\//.test(s));
- const ElmProps = new (function() {
- const props = this.props = {};
- const cssMap = this.cssMap = new Map();
-
- this.getCssPath = function(elm) {
- return cssMap.get(elm) || (cssMap.set(elm, cssPath(elm)), cssMap.get(elm));
- }
-
- this.add = function(elm, type, value) {
- const path = cssPath(elm);
- const EPList = props[path] = props[path] || [];
- const EProp = {};
- EProp.type = type;
- EProp.value = value;
- EPList.push(EProp);
- }
- });
-
- // Hook GM_xmlhttpRequest
- const AM = new AsyncManager();
- AM.onfinish = function() {
- // Add applyProps script
- const script = $C('script');
- script.innerText = "window.addEventListener('load', function(){({FUNC})({PROPS});})"
- .replace('{PROPS}', JSON.stringify(ElmProps.props))
- .replace('{FUNC}', `function(c){const funcs={Canvas:{DataUrl:function(a,b){const img=new Image();const ctx=a.getContext('2d');img.onload=()=>{ctx.drawImage(img,0,0)};img.src=b}},Input:{Value:function(a,b){a.value=b}}};for(const[cssPath,propList]of Object.entries(c)){const elm=document.querySelector(cssPath);for(const prop of propList){const type=prop.type;const value=prop.value;const funcPath=type.split('.');let func=funcs;for(let i=0;i<funcPath.length;i++){func=func[funcPath[i]]}func(elm,value)}}}`);
- $A(dom.head, script);
-
- // Generate html
- const FinalHTML = '{ABOUT}\n\n{HTML}'.replace('{ABOUT}', CONST.Text[i18n].About).replace('{HTML}', dom.querySelector('html').outerHTML)
-
- DoLog(LogLevel.Success, 'Single File Generation Complete.')
- DoLog([dom, FinalHTML]);
- details.onfinish(FinalHTML)
- };
-
- // Change document.characterSet to utf8
- DoLog('SingleFile: Setting charset');
- if (document.characterSet !== 'UTF-8') {
- const meta = $('meta[http-equiv="Content-Type"][content*="charset"]');
- meta && (meta.content = meta.content.replace(/charset\s*=\s*[^;\s]*/i, 'charset=UTF-8'));
- }
-
- // Clear scripts
- DoLog('SingleFile: Clearing scripts');
- for (const script of $_('script')) {
- $R(script);
- }
-
- // Clear inline-scripts
- DoLog('SingleFile: Clearing inline scripts');
- for (const elm of $_('*')) {
- const ISKeys = ['onabort', 'onerror', 'onresize', 'onscroll', 'onunload', 'oncancel', 'oncanplay', 'oncanplaythrough', 'onchange', 'onclick', 'onclose', 'oncuechange', 'ondblclick', 'ondrag', 'ondragend', 'ondragenter', 'ondragexit', 'ondragleave', 'ondragover', 'ondragstart', 'ondrop', 'ondurationchange', 'onemptied', 'onended', 'onerror', 'onfocus', 'oninput', 'oninvalid', 'onkeydown', 'onkeypress', 'onkeyup', 'onload', 'onloadeddata', 'onloadedmetadata', 'onloadstart', 'onmousedown', 'onmouseenter', 'onmouseleave', 'onmousemove', 'onmouseout', 'onmouseover', 'onmouseup', 'onmousewheel', 'onpause', 'onplay', 'onplaying', 'onprogress', 'onratechange', 'onreset', 'onresize', 'onscroll', 'onseeked', 'onseeking', 'onselect', 'onshow', 'onstalled', 'onsubmit', 'onsuspend', 'ontimeupdate', 'ontoggle', 'onvolumechange', 'onwaiting', 'onbegin', 'onend', 'onrepeat'];
- for (const key of ISKeys) {
- elm.removeAttribute(key);
- elm[key] = undefined;
- }
- }
-
- // Clear preload-scripts
- DoLog('SingleFile: Clearing preload scripts');
- for (const link of $_('link[rel*=modulepreload]')) {
- $R(link);
- }
-
- // Remove "Content-Security-Policy" meta header
- DoLog('SingleFile: Removing "Content-Security-Policy" meta headers');
- for (const m of $_('meta[http-equiv="Content-Security-Policy"]')) {
- $R(m);
- }
-
- // Deal styles
- /*
- DoLog('SingleFile: Dealing linked stylesheets');
- for (const link of $_('link[rel="stylesheet"]')) {
- if (!link.href) {continue;}
- const href = link.href;
- AM.add();
- requestText(href, (t, l) => {
- const s = $C('style');
- s.innerText = t;
- $I(s, l);
- $R(l);
- AM.finish();
- }, link);
- }
- */
-
- // Deal Style url(http) links
- DoLog('SingleFile: Dealing style urls');
- for (const link of $_('link[rel*=stylesheet][href]')) {
- dealLinkedStyle(link)
- }
- for (const elm of $_('style')) {
- elm.innerText && dealStyle(elm.innerText, (style, elm) => (elm.innerHTML = style), elm);
- }
-
- // Deal <link>s
- DoLog('SingleFile: Dealing links');
- for (const link of $_('link[href]')) {
- // Only deal http[s] links
- if (!link.href) {continue;}
- if (!ishttp(link.href)) {continue;}
-
- // Only deal links that rel includes one of the following:
- // icon, apple-touch-icon, apple-touch-startup-image, prefetch, preload, prerender, manifest, stylesheet
- // And in the same time NOT includes any of the following:
- // alternate
- let deal = false;
- const accepts = ['icon', 'apple-touch-icon', 'apple-touch-startup-image', 'prefetch', 'preload', 'prerender', 'manifest', 'stylesheet'];
- const excludes = ['alternate']
- const rels = link.rel.split(' ');
- for (const rel of rels) {
- deal = deal || (accepts.includes(rel) && !excludes.includes(rel));
- }
- if (!deal) {continue;}
-
- // Save original href to link.ohref
- link.ohref = link.href;
-
- AM.add();
- requestDataURL(link.href, function(durl, link) {
- link.href = durl;
-
- // Deal style if links to a stylesheet
- if (rels.includes('stylesheet')) {
- dealLinkedStyle(link);
- }
- AM.finish();
- }, link);
- }
-
- // Deal images' and sources' src
- DoLog('SingleFile: Dealing images\' & sources\' src');
- for (const img of $_('img[src], source[src]')) {
- // Get full src
- if (img.src.length > CONST.Number.MaxUrlLength) {continue;}
- if (!img.src) {continue;}
- if (!ishttp(img.src)) {continue;}
- const src = fullurl(img.src);
-
- // Get original img element
- const path = ElmProps.getCssPath(img);
- const oimg = document.querySelector(path);
-
- // Get data url
- let url;
- try {
- if (!oimg.complete) {throw new Error();}
- url = img2url(oimg);
- img.src = url;
- } catch (e) {
- if (img.src) {
- AM.add();
- requestDataURL(src, (url) => {
- img.src = url;
- AM.finish();
- });
- }
- }
- }
-
- // Deal images' and sources' srcset
- DoLog('SingleFile: Dealing images\' & sources\' srcset');
- for (const img of $_('img[srcset], source[srcset]')) {
- // Check if empty
- if (!img.srcset) {continue;}
-
- // Get all srcs list
- const list = img.srcset.split(',');
- for (let i = 0; i < list.length; i++) {
- const srcitem = list[i].trim();
- if (srcitem.length > CONST.Number.MaxUrlLength) {continue;}
- if (!srcitem) {continue}
- const parts = srcitem.replaceAll(/(\s){2,}/g, '$1').split(' ');
- if (!ishttp(parts[0])) {continue};
- const src = fullurl(parts[0]);
-
- list[i] = {
- src: src,
- rest: parts.slice(1, parts.length).join(' '),
- parts: parts,
- dataurl: null,
- string: null
- };
- }
-
- // Get all data urls into list
- const S_AM = new AsyncManager();
- const dlist = [];
- S_AM.onfinish = function() {
- img.srcset = dlist.join(',');
- AM.finish();
- }
- AM.add();
- for (const srcobj of list) {
- S_AM.add();
- requestDataURL(srcobj.src, (url, srcobj) => {
- srcobj.dataurl = url;
- srcobj.string = [srcobj.dataurl, srcobj.rest].join(' ');
- dlist.push(srcobj.string);
- S_AM.finish();
- }, srcobj);
- }
- S_AM.finishEvent = true;
- }
-
- // Deal canvases
- DoLog('SingleFile: Dealing canvases');
- for (const cvs of $_('canvas')) {
- let url;
- try {
- url = img2url(cvs);
- ElmProps.add(cvs, 'Canvas.DataUrl', url);
- } catch (e) {}
- }
-
- // Deal background-images
- DoLog('SingleFile: Dealing background-images');
- for (const elm of $_('*')) {
- const urlReg = /^\s*url\(\s*['"]?([^\(\)'"]+)['"]?\s*\)\s*$/;
- const bgImage = elm.style.backgroundImage;
- if (!bgImage) {continue;}
- if (bgImage.length > CONST.Number.MaxUrlLength) {continue;}
- if (bgImage === 'url("https://images.weserv.nl/?url=https://ae01.alicdn.com/kf/H3bbe45ee0a3841ec9644e1ea9aa157742.jpg")') {debugger;}
- if (bgImage && urlReg.test(bgImage)) {
- // Get full image url
- let url = bgImage.match(urlReg)[1];
- if (/^data:/.test(url)) {continue;}
- url = fullurl(url);
-
- // Get image
- AM.add();
- requestDataURL(url, function(durl, elm) {
- elm.style.backgroundImage = 'url({U})'.replace('{U}', durl);
- AM.finish();
- }, elm);
- }
- }
-
- // Deal input/textarea/progress values
- DoLog('SingleFile: Dealing values');
- for (const elm of $_('input,textarea,progress')) {
- // Query origin element's value
- const cssPath = ElmProps.getCssPath(elm);
- const oelm = document.querySelector(cssPath);
-
- // Add to property map
- oelm.value && ElmProps.add(elm, 'Input.Value', oelm.value);
- }
-
- // Get favicon.ico if no icon found
- DoLog('SingleFile: Dealing favicon.ico');
- if (!$('link[rel*=icon]')) {
- const I_AM = new AsyncManager();
- GM_xmlhttpRequest({
- method: 'GET',
- url: getHost() + 'favicon.ico',
- responseType: 'blob',
- onload: (e) => {
- if (e.status >= 200 && e.status < 300) {
- blobToDataURL(e.response, (durl) => {
- const icon = $C('link');
- icon.rel = 'icon';
- icon.href = durl;
- $A(dom.head, icon);
- });
- }
- I_AM.finish();
- }
- })
- }
-
- // Start generating the finish event
- DoLog('SingleFile: Waiting for async tasks to be finished');
- AM.finishEvent = true;
-
- function dealStyle(style, callback, args=[]) {
- const re = /url\(\s*['"]?([^\(\)'"]+)['"]?\s*\)/;
- const rg = /url\(\s*['"]?([^\(\)'"]+)['"]?\s*\)/g;
- const replace = (durl, urlexp, arg1, arg2, arg3) => {
- // Replace style text
- const durlexp = 'url("{D}")'.replace('{D}', durl);
- style = style.replaceAll(urlexp, durlexp);
-
- // Get args
- const args = [style];
- for (let i = 2; i < arguments.length; i++) {
- args.push(arguments[i]);
- }
- callback.apply(null, args);
- AM.finish();
- };
-
- const all = style.match(rg);
- if (!all) {return;}
- for (const urlexp of all) {
- // Check url
- if (urlexp.length > CONST.Number.MaxUrlLength) {continue;}
- const osrc = urlexp.match(re)[1];
- const baseurl = args instanceof HTMLLinkElement && args.ohref ? args.ohref : location.href;
- if (!ishttp(osrc)) {continue;}
- const src = fullurl(osrc, baseurl);
-
- // Request
- AM.add();
- requestDataURL(src, replace, [urlexp].concat(args));
- }
- }
- function dealLinkedStyle(link) {
- if (!link.href || !/^data:/.test(link.href)) {return;}
- const durl = link.href;
- const blob = dataURLToBlob(durl);
- const reader = new FileReader();
- reader.onload = () => {
- dealStyle(reader.result, (style, link) => {
- const blob = new Blob([style],{type:"text/css"});
- AM.add();
- blobToDataURL(blob, function(durl, link) {
- link.href = durl;
- AM.finish();
- }, link)
- }, link);
- AM.finish();
- }
- AM.add();
- reader.readAsText(blob);
- }
- }
-
- // This function is expected to be used on output html
- function applyProps(props) {
- const funcs = {
- Canvas: {
- DataUrl: function(elm, value) {
- const img = new Image();
- const ctx = elm.getContext('2d');
- img.onload = () => {ctx.drawImage(img, 0, 0);};
- img.src = value;
- }
- },
- Input: {
- Value: function(elm, value) {
- elm.value = value;
- }
- }
- };
-
- for (const [cssPath, propList] of Object.entries(props)) {
- const elm = document.querySelector(cssPath);
- for (const prop of propList) {
- const type = prop.type;
- const value = prop.value;
-
- // Get function
- const funcPath = type.split('.');
- let func = funcs;
- for (let i = 0; i < funcPath.length; i++) {
- func = func[funcPath[i]];
- }
-
- // Call function
- func(elm, value);
- }
- }
- }
-
- function fullurl(url, baseurl=location.href) {
- if (/^\/{2,}/.test(url)) {url = location.protocol + url;}
- if (!/^https?:\/\//.test(url)) {
- const base = baseurl.replace(/(.+\/).*?$/, '$1');;
- const a = document.createElement('a');
- a.href = base + url;
- url = a.href;
- }
- return url;
- }
-
- function cssPath(el) {
- if (!(el instanceof Element)) return;
- var path = [];
- while (el.nodeType === Node.ELEMENT_NODE) {
- var selector = el.nodeName.toLowerCase();
- if (el.id) {
- selector += '#' + el.id;
- path.unshift(selector);
- break;
- } else {
- var sib = el,
- nth = 1;
- while (sib = sib.previousElementSibling) {
- if (sib.nodeName.toLowerCase() == selector) nth++;
- }
- if (nth != 1) selector += ":nth-of-type(" + nth + ")";
- }
- path.unshift(selector);
- el = el.parentNode;
- }
- return path.join(" > ");
- }
-
- function requestText(url, callback, args=[]) {
- GM_xmlhttpRequest({
- method: 'GET',
- url: url,
- responseType: 'text',
- onload: function(response) {
- const text = response.responseText;
- const argvs = [text].concat(args);
- callback.apply(null, argvs);
- }
- })
- }
-
- function requestDataURL(url, callback, args=[]) {
- GM_xmlhttpRequest({
- method: 'GET',
- url: url,
- responseType: 'blob',
- onload: function(response) {
- const blob = response.response;
- blobToDataURL(blob, function(url) {
- const argvs = [url].concat(args);
- callback.apply(null, argvs);
- })
- }
- })
- }
-
- function blobToDataURL(blob, callback, args=[]) {
- const reader = new FileReader();
- reader.onload = function () {
- callback.apply(null, [reader.result].concat(args));
- }
- reader.readAsDataURL(blob);
- }
-
- function dataURLToBlob(dataurl) {
- let arr = dataurl.split(','),
- mime = arr[0].match(/:(.*?);/)[1],
- bstr = atob(arr[1]),
- n = bstr.length,
- u8arr = new Uint8Array(n)
- while (n--) {
- u8arr[n] = bstr.charCodeAt(n)
- }
- return new Blob([u8arr], { type: mime })
- }
-
- function XHRFinisher() {
- const XHRF = this;
-
- // Ongoing xhr count
- this.xhrCount = 0;
-
- // Whether generate finish events
- this.finishEvent = false;
-
- // Original xhr
- this.GM_xmlhttpRequest = GM_xmlhttpRequest;
-
- // xhr provided for outer scope
- GM_xmlhttpRequest = function(details) {
- DoLog('XHRFinisher: Requesting ' + details.url);
-
- // Hook functions that will be called when xhr stops
- details.onload = wrap(details.onload)
- details.ontimeout = wrap(details.ontimeout)
- details.onerror = wrap(details.onerror)
- details.onabort = wrap(details.onabort)
-
- // Count increase
- XHRF.xhrCount++;
-
- // Start xhr
- XHRF.GM_xmlhttpRequest(details);
-
- function wrap(ofunc) {
- return function(e) {
- DoLog('XHRFinisher: Request ' + details.url + ' finish. ' + (XHRF.xhrCount-1).toString() + ' requests rest. ');
- ofunc(e);
- --XHRF.xhrCount === 0 && XHRF.finishEvent && XHRF.onfinish && XHRF.onfinish();
- }
- }
- }
- }
-
- function AsyncManager() {
- const AM = this;
-
- // Ongoing xhr count
- this.taskCount = 0;
-
- // Whether generate finish events
- let finishEvent = false;
- Object.defineProperty(this, 'finishEvent', {
- configurable: true,
- enumerable: true,
- get: () => (finishEvent),
- set: (b) => {
- finishEvent = b;
- b && AM.taskCount === 0 && AM.onfinish && AM.onfinish();
- }
- });
-
- // Add one task
- this.add = () => (++AM.taskCount);
-
- // Finish one task
- this.finish = () => ((--AM.taskCount === 0 && AM.finishEvent && AM.onfinish && AM.onfinish(), AM.taskCount));
- }
-
- function img2url(img) {
- const cvs = document.createElement('canvas');
- const ctx = cvs.getContext('2d');
- cvs.width = img.width;
- cvs.height = img.height;
- ctx.drawImage(img, 0, 0)
- return cvs.toDataURL();
- }
-
- // Get a time text like 1970-01-01 00:00:00
- // if dateSpliter provided false, there will be no date part. The same for timeSpliter.
- function getTime(dateSpliter='-', timeSpliter=':') {
- const d = new Date();
- let fulltime = ''
- fulltime += dateSpliter ? fillNumber(d.getFullYear(), 4) + dateSpliter + fillNumber((d.getMonth() + 1), 2) + dateSpliter + fillNumber(d.getDate(), 2) : '';
- fulltime += dateSpliter && timeSpliter ? ' ' : '';
- fulltime += timeSpliter ? fillNumber(d.getHours(), 2) + timeSpliter + fillNumber(d.getMinutes(), 2) + timeSpliter + fillNumber(d.getSeconds(), 2) : '';
- return fulltime;
- }
-
- // Just stopPropagation and preventDefault
- function destroyEvent(e) {
- if (!e) {return false;};
- if (!e instanceof Event) {return false;};
- e.stopPropagation();
- e.preventDefault();
- }
-
- // GM_XHR HOOK: The number of running GM_XHRs in a time must under maxXHR
- // Returns the abort function to stop the request anyway(no matter it's still waiting, or requesting)
- // (If the request is invalid, such as url === '', will return false and will NOT make this request)
- // If the abort function called on a request that is not running(still waiting or finished), there will be NO onabort event
- // Requires: function delItem(){...} & function uniqueIDMaker(){...}
- function GMXHRHook(maxXHR=5) {
- const GM_XHR = GM_xmlhttpRequest;
- const getID = uniqueIDMaker();
- let todoList = [], ongoingList = [];
- GM_xmlhttpRequest = safeGMxhr;
-
- function safeGMxhr() {
- // Get an id for this request, arrange a request object for it.
- const id = getID();
- const request = {id: id, args: arguments, aborter: null};
-
- // Deal onload function first
- dealEndingEvents(request);
-
- /* DO NOT DO THIS! KEEP ITS ORIGINAL PROPERTIES!
- // Stop invalid requests
- if (!validCheck(request)) {
- return false;
- }
- */
-
- // Judge if we could start the request now or later?
- todoList.push(request);
- checkXHR();
- return makeAbortFunc(id);
-
- // Decrease activeXHRCount while GM_XHR onload;
- function dealEndingEvents(request) {
- const e = request.args[0];
-
- // onload event
- const oriOnload = e.onload;
- e.onload = function() {
- reqFinish(request.id);
- checkXHR();
- oriOnload ? oriOnload.apply(null, arguments) : function() {};
- }
-
- // onerror event
- const oriOnerror = e.onerror;
- e.onerror = function() {
- reqFinish(request.id);
- checkXHR();
- oriOnerror ? oriOnerror.apply(null, arguments) : function() {};
- }
-
- // ontimeout event
- const oriOntimeout = e.ontimeout;
- e.ontimeout = function() {
- reqFinish(request.id);
- checkXHR();
- oriOntimeout ? oriOntimeout.apply(null, arguments) : function() {};
- }
-
- // onabort event
- const oriOnabort = e.onabort;
- e.onabort = function() {
- reqFinish(request.id);
- checkXHR();
- oriOnabort ? oriOnabort.apply(null, arguments) : function() {};
- }
- }
-
- // Check if the request is invalid
- function validCheck(request) {
- const e = request.args[0];
-
- if (!e.url) {
- return false;
- }
-
- return true;
- }
-
- // Call a XHR from todoList and push the request object to ongoingList if called
- function checkXHR() {
- if (ongoingList.length >= maxXHR) {return false;};
- if (todoList.length === 0) {return false;};
- const req = todoList.shift();
- const reqArgs = req.args;
- const aborter = GM_XHR.apply(null, reqArgs);
- req.aborter = aborter;
- ongoingList.push(req);
- return req;
- }
-
- // Make a function that aborts a certain request
- function makeAbortFunc(id) {
- return function() {
- let i;
-
- // Check if the request haven't been called
- for (i = 0; i < todoList.length; i++) {
- const req = todoList[i];
- if (req.id === id) {
- // found this request: haven't been called
- delItem(todoList, i);
- return true;
- }
- }
-
- // Check if the request is running now
- for (i = 0; i < ongoingList.length; i++) {
- const req = todoList[i];
- if (req.id === id) {
- // found this request: running now
- req.aborter();
- reqFinish(id);
- checkXHR();
- }
- }
-
- // Oh no, this request is already finished...
- return false;
- }
- }
-
- // Remove a certain request from ongoingList
- function reqFinish(id) {
- let i;
- for (i = 0; i < ongoingList.length; i++) {
- const req = ongoingList[i];
- if (req.id === id) {
- ongoingList = delItem(ongoingList, i);
- return true;
- }
- }
- return false;
- }
- }
- }
-
- function parseDocument(htmlblob, callback, args=[]) {
- const reader = new FileReader();
- reader.onload = function(e) {
- const htmlText = reader.result;
- const dom = new DOMParser().parseFromString(htmlText, 'text/html');
- args = [dom].concat(args);
- callback.apply(null, args);
- //callback(dom, htmlText);
- }
- reader.readAsText(htmlblob, 'GBK');
- }
-
- // Get a url argument from lacation.href
- // also recieve a function to deal the matched string
- // returns defaultValue if name not found
- // Args: name, dealFunc=(function(a) {return a;}), defaultValue=null
- function getUrlArgv(details) {
- typeof(details) === 'string' && (details = {name: details});
- typeof(details) === 'undefined' && (details = {});
- if (!details.name) {return null;};
-
- const url = details.url ? details.url : location.href;
- const name = details.name ? details.name : '';
- const dealFunc = details.dealFunc ? details.dealFunc : ((a)=>{return a;});
- const defaultValue = details.defaultValue ? details.defaultValue : null;
- const matcher = new RegExp(name + '=([^&]+)');
- const result = url.match(matcher);
- const argv = result ? dealFunc(result[1]) : defaultValue;
-
- return argv;
- }
-
- // Append a style text to document(<head>) with a <style> element
- function addStyle(css, id) {
- const style = document.createElement("style");
- id && (style.id = id);
- style.textContent = css;
- for (const elm of document.querySelectorAll('#'+id)) {
- elm.parentElement && elm.parentElement.removeChild(elm);
- }
- document.head.appendChild(style);
- }
-
- function saveTextToFile(text, name) {
- const blob = new Blob([text],{type:"text/plain;charset=utf-8"});
- const url = URL.createObjectURL(blob);
- const a = document.createElement('a');
- a.href = url;
- a.download = name;
- a.click();
- }
-
- // File download function
- // details looks like the detail of GM_xmlhttpRequest
- // onload function will be called after file saved to disk
- function downloadFile(details) {
- if (!details.url || !details.name) {return false;};
-
- // Configure request object
- const requestObj = {
- url: details.url,
- responseType: 'blob',
- onload: function(e) {
- // Save file
- saveFile(URL.createObjectURL(e.response), details.name);
-
- // onload callback
- details.onload ? details.onload(e) : function() {};
- }
- }
- if (details.onloadstart ) {requestObj.onloadstart = details.onloadstart;};
- if (details.onprogress ) {requestObj.onprogress = details.onprogress;};
- if (details.onerror ) {requestObj.onerror = details.onerror;};
- if (details.onabort ) {requestObj.onabort = details.onabort;};
- if (details.onreadystatechange) {requestObj.onreadystatechange = details.onreadystatechange;};
- if (details.ontimeout ) {requestObj.ontimeout = details.ontimeout;};
-
- // Send request
- GM_xmlhttpRequest(requestObj);
- }
-
- // get '/' splited API array from a url
- function getAPI(url=location.href) {
- return url.replace(/https?:\/\/(.*?\.){1,2}.*?\//, '').replace(/\?.*/, '').match(/[^\/]+?(?=(\/|$))/g);
- }
-
- // get host part from a url(includes '^https://', '/$')
- function getHost(url=location.href) {
- const match = location.href.match(/https?:\/\/[^\/]+\//);
- return match ? match[0] : match;
- }
-
- // Your code here...
- // Bypass xbrowser's useless GM_functions
- function bypassXB() {
- if (typeof(mbrowser) === 'object') {
- window.unsafeWindow = window.GM_setClipboard = window.GM_openInTab = window.GM_xmlhttpRequest = window.GM_getValue = window.GM_setValue = window.GM_listValues = window.GM_deleteValue = undefined;
- }
- }
-
- // GM_Polyfill By PY-DNG
- // 2021.07.18 - 2021.07.19
- // Simply provides the following GM_functions using localStorage, XMLHttpRequest and window.open:
- // Returns object GM_POLYFILLED which has the following properties that shows you which GM_functions are actually polyfilled:
- // GM_setValue, GM_getValue, GM_deleteValue, GM_listValues, GM_xmlhttpRequest, GM_openInTab, GM_setClipboard, unsafeWindow(object)
- // All polyfilled GM_functions are accessable in window object/Global_Scope(only without Tempermonkey Sandboxing environment)
- function GM_PolyFill(name='default') {
- const GM_POLYFILL_KEY_STORAGE = 'GM_STORAGE_POLYFILL';
- let GM_POLYFILL_storage;
- const GM_POLYFILLED = {
- GM_setValue: true,
- GM_getValue: true,
- GM_deleteValue: true,
- GM_listValues: true,
- GM_xmlhttpRequest: true,
- GM_openInTab: true,
- GM_setClipboard: true,
- unsafeWindow: true,
- once: false
- }
-
- // Ignore GM_PolyFill_Once
- window.GM_POLYFILLED && window.GM_POLYFILLED.once && (window.unsafeWindow = window.GM_setClipboard = window.GM_openInTab = window.GM_xmlhttpRequest = window.GM_getValue = window.GM_setValue = window.GM_listValues = window.GM_deleteValue = undefined);
-
- GM_setValue_polyfill();
- GM_getValue_polyfill();
- GM_deleteValue_polyfill();
- GM_listValues_polyfill();
- GM_xmlhttpRequest_polyfill();
- GM_openInTab_polyfill();
- GM_setClipboard_polyfill();
- unsafeWindow_polyfill();
-
- function GM_POLYFILL_getStorage() {
- let gstorage = localStorage.getItem(GM_POLYFILL_KEY_STORAGE);
- gstorage = gstorage ? JSON.parse(gstorage) : {};
- let storage = gstorage[name] ? gstorage[name] : {};
- return storage;
- }
-
- function GM_POLYFILL_saveStorage() {
- let gstorage = localStorage.getItem(GM_POLYFILL_KEY_STORAGE);
- gstorage = gstorage ? JSON.parse(gstorage) : {};
- gstorage[name] = GM_POLYFILL_storage;
- localStorage.setItem(GM_POLYFILL_KEY_STORAGE, JSON.stringify(gstorage));
- }
-
- // GM_setValue
- function GM_setValue_polyfill() {
- typeof (GM_setValue) === 'function' ? GM_POLYFILLED.GM_setValue = false: window.GM_setValue = PF_GM_setValue;;
-
- function PF_GM_setValue(name, value) {
- GM_POLYFILL_storage = GM_POLYFILL_getStorage();
- name = String(name);
- GM_POLYFILL_storage[name] = value;
- GM_POLYFILL_saveStorage();
- }
- }
-
- // GM_getValue
- function GM_getValue_polyfill() {
- typeof (GM_getValue) === 'function' ? GM_POLYFILLED.GM_getValue = false: window.GM_getValue = PF_GM_getValue;
-
- function PF_GM_getValue(name, defaultValue) {
- GM_POLYFILL_storage = GM_POLYFILL_getStorage();
- name = String(name);
- if (GM_POLYFILL_storage.hasOwnProperty(name)) {
- return GM_POLYFILL_storage[name];
- } else {
- return defaultValue;
- }
- }
- }
-
- // GM_deleteValue
- function GM_deleteValue_polyfill() {
- typeof (GM_deleteValue) === 'function' ? GM_POLYFILLED.GM_deleteValue = false: window.GM_deleteValue = PF_GM_deleteValue;
-
- function PF_GM_deleteValue(name) {
- GM_POLYFILL_storage = GM_POLYFILL_getStorage();
- name = String(name);
- if (GM_POLYFILL_storage.hasOwnProperty(name)) {
- delete GM_POLYFILL_storage[name];
- GM_POLYFILL_saveStorage();
- }
- }
- }
-
- // GM_listValues
- function GM_listValues_polyfill() {
- typeof (GM_listValues) === 'function' ? GM_POLYFILLED.GM_listValues = false: window.GM_listValues = PF_GM_listValues;
-
- function PF_GM_listValues() {
- GM_POLYFILL_storage = GM_POLYFILL_getStorage();
- return Object.keys(GM_POLYFILL_storage);
- }
- }
-
- // unsafeWindow
- function unsafeWindow_polyfill() {
- typeof (unsafeWindow) === 'object' ? GM_POLYFILLED.unsafeWindow = false: window.unsafeWindow = window;
- }
-
- // GM_xmlhttpRequest
- // not supported properties of details: synchronous binary nocache revalidate context fetch
- // not supported properties of response(onload arguments[0]): finalUrl
- // ---!IMPORTANT!--- DOES NOT SUPPORT CROSS-ORIGIN REQUESTS!!!!! ---!IMPORTANT!---
- function GM_xmlhttpRequest_polyfill() {
- typeof (GM_xmlhttpRequest) === 'function' ? GM_POLYFILLED.GM_xmlhttpRequest = false: window.GM_xmlhttpRequest = PF_GM_xmlhttpRequest;
-
- // details.synchronous is not supported as Tempermonkey
- function PF_GM_xmlhttpRequest(details) {
- const xhr = new XMLHttpRequest();
-
- // open request
- const openArgs = [details.method, details.url, true];
- if (details.user && details.password) {
- openArgs.push(details.user);
- openArgs.push(details.password);
- }
- xhr.open.apply(xhr, openArgs);
-
- // set headers
- if (details.headers) {
- for (const key of Object.keys(details.headers)) {
- xhr.setRequestHeader(key, details.headers[key]);
- }
- }
- details.cookie ? xhr.setRequestHeader('cookie', details.cookie) : function () {};
- details.anonymous ? xhr.setRequestHeader('cookie', '') : function () {};
-
- // properties
- xhr.timeout = details.timeout;
- xhr.responseType = details.responseType;
- details.overrideMimeType ? xhr.overrideMimeType(details.overrideMimeType) : function () {};
-
- // events
- xhr.onabort = details.onabort;
- xhr.onerror = details.onerror;
- xhr.onloadstart = details.onloadstart;
- xhr.onprogress = details.onprogress;
- xhr.onreadystatechange = details.onreadystatechange;
- xhr.ontimeout = details.ontimeout;
- xhr.onload = function (e) {
- const response = {
- readyState: xhr.readyState,
- status: xhr.status,
- statusText: xhr.statusText,
- responseHeaders: xhr.getAllResponseHeaders(),
- response: xhr.response
- };
- (details.responseType === '' || details.responseType === 'text') ? (response.responseText = xhr.responseText) : function () {};
- (details.responseType === '' || details.responseType === 'document') ? (response.responseXML = xhr.responseXML) : function () {};
- details.onload(response);
- }
-
- // send request
- details.data ? xhr.send(details.data) : xhr.send();
-
- return {
- abort: xhr.abort
- };
- }
- }
-
- // NOTE: options(arg2) is NOT SUPPORTED! if provided, then will just be skipped.
- function GM_openInTab_polyfill() {
- typeof (GM_openInTab) === 'function' ? GM_POLYFILLED.GM_openInTab = false: window.GM_openInTab = PF_GM_openInTab;
-
- function PF_GM_openInTab(url) {
- window.open(url);
- }
- }
-
- // NOTE: needs to be called in an event handler function, and info(arg2) is NOT SUPPORTED!
- function GM_setClipboard_polyfill() {
- typeof (GM_setClipboard) === 'function' ? GM_POLYFILLED.GM_setClipboard = false: window.GM_setClipboard = PF_GM_setClipboard;
-
- function PF_GM_setClipboard(text) {
- // Create a new textarea for copying
- const newInput = document.createElement('textarea');
- document.body.appendChild(newInput);
- newInput.value = text;
- newInput.select();
- document.execCommand('copy');
- document.body.removeChild(newInput);
- }
- }
-
- return GM_POLYFILLED;
- }
-
- // Makes a function that returns a unique ID number each time
- function uniqueIDMaker() {
- let id = 0;
- return makeID;
- function makeID() {
- id++;
- return id;
- }
- }
-
- // Fill number text to certain length with '0'
- function fillNumber(number, length) {
- let str = String(number);
- for (let i = str.length; i < length; i++) {
- str = '0' + str;
- }
- return str;
- }
-
- // Del a item from an array using its index. Returns the array but can NOT modify the original array directly!!
- function delItem(arr, delIndex) {
- arr = arr.slice(0, delIndex).concat(arr.slice(delIndex+1));
- return arr;
- }
- })();