Greasy Fork is available in English.

轻小说文库+ (重制)

轻小说文库+的重置,写好了就作为一个更新直接发上去,所以到时候记得改name、namespace和描述

/* eslint-disable no-multi-spaces */
/* eslint-disable no-loop-func */
/* eslint-disable no-return-assign */

// ==UserScript==
// @name               轻小说文库+ (重制)
// @namespace          Wenku8+_re
// @version            0.1.4
// @description        轻小说文库+的重置,写好了就作为一个更新直接发上去,所以到时候记得改name、namespace和描述
// @author             PY-DNG
// @license            GPL-3.0-or-later
// @match              http*://www.wenku8.net/*
// @match              http*://www.wenku8.cc/*
// @match              http*://wenku8.net/*
// @match              http*://wenku8.cc/*
// @connect            wenku8.com
// @connect            wenku8.net
// @connect            wenku8.cc
// @connect            p.sda1.dev
// @connect            image.sogou.com
// @connect            *
// @require            https://update.greasyfork.org/scripts/456034/1303041/Basic%20Functions%20%28For%20userscripts%29.js
// @require            https://greasyfork.org/scripts/449583-configmanager/code/script.js?version=1326639
// @require            https://greasyfork.org/scripts/471280-url-encoder/code/URL%20Encoder.js?version=1247074
// @require            https://greasyfork.org/scripts/460385-gm-web-hooks/code/script.js?version=1221394
// @require            https://greasyfork.org/scripts/445697-greasy-fork-api/code/Greasy%20Fork%20API.js?version=1055543
// @require            https://fastly.jsdelivr.net/npm/@popperjs/core@2.11.8/dist/umd/popper.min.js
// @require            https://fastly.jsdelivr.net/npm/tippy.js@6.3.7/dist/tippy.umd.min.js
// @require            https://fastly.jsdelivr.net/npm/alertifyjs@1.13.1/build/alertify.min.js
// @require            https://fastly.jsdelivr.net/npm/@shopify/draggable@1.0.0-beta.8/lib/sortable.js
// @require            https://fastly.jsdelivr.net/gh/vkiryukhin/vkBeautify@0a238953acf12ac2f366fd63998514e1ac9db042/vkbeautify.js
// @require            https://fastly.jsdelivr.net/npm/darkmode-js@1.5.7/lib/darkmode-js.min.js
// @require            https://fastly.jsdelivr.net/npm/crypto-js@4.1.1/crypto-js.min.js
// @resource           tippy-css       https://cdn.jsdelivr.net/npm/tippy.js@6.3.7/dist/tippy.css
// @resource           alertify-css    https://fastly.jsdelivr.net/npm/alertifyjs@1.13.1/build/css/alertify.min.css
// @resource           alertify-theme  https://fastly.jsdelivr.net/npm/alertifyjs@1.13.1/build/css/themes/default.min.css
// @icon               
// @grant              GM_getValue
// @grant              GM_setValue
// @grant              GM_listValues
// @grant              GM_deleteValue
// @grant              GM_xmlhttpRequest
// @grant              GM_getResourceText
// @run-at             document-start
// ==/UserScript==

/* global require FuncInfo isWenkuFunction Messager FL_listFunctions FL_getFunction FL_enableFunction FL_disableFunction FL_loadSetting FL_getDebug FL_exportConfig FL_importConfig FL_postMessage FL_recieveMessage */
/* global LogLevel DoLog Err $ $All $CrE $AEL $$CrE addStyle detectDom destroyEvent copyProp copyProps parseArgs escJsStr replaceText getUrlArgv dl_browser dl_GM AsyncManager */
/* global ConfigManager $URL GMXHRHook GMDLHook GreasyFork */
/* global tippy alertify Sortable vkbeautify Darkmode CryptoJS*/

/* 开发进度
[x]: 已完成
[-]: 正在施工
[?]: 等待测试
[o]: 待完善
[ ]: 待开发
[D]: 暂时搁置,后期计划开发
*: 代表相对于旧版有bug修复/优化/新功能

[x] 功能加载器
  [x] 加载功能
  [x] 启用/禁用功能
  [x] 功能存储隔离
  [x] 账号存储隔离
  [o] 功能日志记录
[o] 设置界面*
  [o] 功能设置
    [x] 独立功能管理界面
      [x] 界面框架
      [x] 启用/停用
      [o] 存储管理
        [x] 查看存储
        [x] 修改存储
        [x] 导出存储
        [x] 导入存储
        [x] 恢复出厂设置
    [o] 功能设置界面框架
      [x] 功能设置界面
      [x] 功能设置界面表格对齐
  [x] 脚本设置
    [x] 设置界面
    [x] 检查更新
    [x] 自动检查更新
    [x] 导出配置
    [x] 导入配置
    [x] 导出日志
	[ ] 跨账号配置迁移
  [x] 防止设置界面同时打开多次
[x] wenku8.cc域名支持
[o] 在线阅读解锁
  [ ] 兼容haoa的轻小说文库下载
[x] 下载解锁
  [x] 下载panel
  [x] 下载页面*
  [x] “本书公告”
  [x] 繁体版页面支持
  [x] 不用iframe
[x] 下载增强*
  [x] txt*
  [x] txtfull*
  [x] jar*
  [x] umd*
  [x] 繁体版页面支持
  [x] 修复文库下载带html字符编码问题*
    [x] txt分卷
    [x] txt全本
[x] 全部插图下载
[ ] 多格式下载器
  [ ] 下载器框架
  [ ] txt全本
  [ ] txt分卷
  [ ] epub全本
  [ ] epub分卷
  [ ] 插图
[o] 单章节下载*
  [x] 文本下载
  [o] 图片下载
    [D] 下载失败时,点击最终提示信息框,可显示详细下载状态信息
  [x] 繁体版页面支持
[x] 书架增强
  [x] 书架整合
  [x] 书架重命名
  [x] 自动推书界面整合
    [x] 自动推书界面
    [x] 自动推书功能未启用时不显示其界面
    [x] 提示框优化
      [x] 实时显示自动推书总次数
      [x] 相同书的提示框合并显示
[o] 自动推书
    [o] 点击显示推书详情
[o] 页面美化*
  [x] 通用页面美化*
    [x] 背景图
    [x] 半透明遮罩层
    [x] 遮罩层高斯模糊
    [x] 兼容深色模式,深色模式和浅色模式分别存储两套遮罩颜色、模糊程度、不透明度
  [o] 书评页面美化
    [x] 书评页面内容宽度平齐*
    [D] 书评页面美化后保留美化前scroll状态
  [o] 阅读页美化*
    [x] 阅读页美化正文居中(可选)*
    [x] 双击alertify和SidePanel不再自动滚屏*
    [D] 阅读页美化正文宽度可调*
  [x] 繁体版页面支持
  [D] 页面内调节面板,实时应用修改
[x] 阅读增强
  [x] 更多字体颜色
  [x] 更多字体大小
  [x] 更多字体
  [x] 繁体版页面支持
  [x] 翻页键支持
[x] 去广告
  [x] 繁体版页面支持
[x] 未登录自动跳转首页
[o] 图床*
  [o] 图床加载框架
    [x] 基础框架
    [x] 图床自动选择机制
      [x] 上传时评分排序
      [x] 上传后记录结果
    [x] 图床返回url自动encodeURI
    [D] 用户手动选择图床机制
  [x] 设置图片上传组件
    [x] 选择图片
    [x] 上传图片
    [x] 清空图片
  [o] 图床
    [x] 搜狗图床
    [D] 360图床
    [x] 流浪图床
[x] 稍后再读*
  [x] 首页稍后再读书籍列表*
  [x] 书介绍页加入/移出按钮
  [x] 用户手动顺序调整*
  [x] 繁体版页面支持
  [x] 首页点击X移除稍后再读*
[x] 书籍信息复制
[x] 书籍tag跳转
[o] 侧边栏
  [x] 侧边栏框架
  [o] 侧边栏默认svg*
    [D] svg一直展示到fa显示出来再隐藏
[x] 深色模式*
  [x] 主页
  [x] 书架
  [x] 书页
  [x] 阅读
  [x] 目录
  [x] 评论
    [x] bbcode
  [x] 评论编辑
  [x] 评论列表
  [x] 排行榜
  [x] tags
  [x] 书评吐槽
  [x] 用户页面
    [x] 用户主页
    [x] 用户面板
    [x] 用户编辑
    [x] 修改密码
    [x] 设置头像
    [x] 用户好友
    [x] 用户链接
    [x] 短消息
      [x] 收件箱
      [x] 发件箱
      [x] 写新消息
  [x] 登录页面
  [x] 跳转api页面
  [x] 设置界面
  [x] UBBEditor menu
  [x] 右上角提示框
  [x] 文库dialog弹窗
  [x] 其他页面
    [x] https://www.wenku8.net/other/jubao.php
    [x] https://www.wenku8.net/other/help.php
  [x] tippy适配
  [x] 页面美化适配
[D] 在线阅读历史记录
[o] 账号切换
  [x] 顶栏账号切换
  [x] 账号自动录入
  [x] 账号管理
[o] 书评草稿
  [x] 书评草稿自动保存和加载
  [x] 书评草稿管理
[o] 书评增强
  [x] 引用
    [x] 引用
    [x] 仅引用楼号
  [D] @
  [x] 链接图片增强*
    [x] 支持https链接
    [x] 自动encodeuri
    [x] 格式检查
  [x] 本地图片插入
    [x] 界面点击上传
    [x] 拖动上传
    [x] 粘贴上传
    [x] 插入图片保持光标位置*
  [x] 自适应高度的编辑器*
  [x] 书评回复标题
  [o] 书评下载保存
    [x] 基本书评下载
    [x] BBCode书评下载
    [x] json格式导出*
    [D] 保存书评为带图片的格式*
  [o] 书评收藏*
    [x] 收藏/取消收藏
    [o] 首页展示
      [o] 防止文本溢出
    [x] 拖动调整顺序
    [ ] “更多”按钮和管理界面
    [ ] 备注
    [x] 默认收藏:文库导航姬
    [ ] 默认收藏:脚本反馈贴
  [x] 换页不刷新*
  [x] 页面内书评编辑
    [x] UBBEditor编辑器支持*
    [x] 应用UBBEditor增强功能*
  [x] 链接跳转
  [D] 旧版页面[img]*.(jpg|png)[/img]加载*
  [x] 繁体版页面支持
  [x] 自动刷新*
  [x] 页面回复后仍在当前页面*
    [x] 后台提交回复并根据文库返回页面热更新当前页面
      [x] 返回没有重定向的错误页面,显示错误信息并保留编辑器内容
      [x] 返回带重定向的书评编辑成功页面,更新页面内容到当前page
      [x] 返回书评第一页,更新页面内容到last page
  [x] Ctrl+Enter提交表单
    [x] Ctrl+Enter提交表单
    [x] Command+Enter提交表单
    [x] 去除Windows+Enter提交表单
    [x] 按钮提示文本
  [x] 表单格式检查
    [x] 不要空回复
    [x] 内容长度≥7
[ ] tag搜索
[x] 脚本执行时间提前
[-] X浏览器支持
[ ] 旧版配置自动导入
[ ] 回调代码Promise化
*/

(function __MAIN__() {
	'use strict';

	// Constances
	const CONST = {
		TextAllLang: {
			DEFAULT: 'zh-CN',
			'zh-CN': {
				Loader: {
					CheckerError: 'Checker Error',
					CheckerInvalid: 'Checker Invalid'
				}
			}
		},
		Config_Ruleset: {
			'version-key': 'config-version',
			'ignores': ["LOCAL-CDN"],
			'defaultValues': {
				//'config-key': {},
				funcs: {},
				debug: []
			},
			'updaters': {
				/*'config-key': [
					function() {
						// This function contains updater for config['config-key'] from v0 to v1
					},
					function() {
						// This function contains updater for config['config-key'] from v1 to v2
					}
				]*/
			},
			'globalUpdaters': [
				function(config) {
					//config
				}
			]
		}
	};

	// Init language
	const i18n = Object.keys(CONST.TextAllLang).includes(navigator.language) ? navigator.language : CONST.TextAllLang.DEFAULT;
	CONST.Text = CONST.TextAllLang[i18n];

	// Functions
	const Functions = [
		// Test
		{
			name: '测试函数',
			description: '用于调试功能加载器以及脚本环境是否正常',
			id: 'test',
			system: false,
			data: {
				load: false,
				storage: false,
				manager: false,
				makeerror: false,
				errortype: 'setTimeout',
				log: [],
				allowdebug: unsafeWindow.isPY_DNG && unsafeWindow.userscriptDebugging,
				Config_Ruleset: {
					'version-key': 'config-version',
					'ignores': ["LOCAL-CDN"],
					'defaultValues': {},
					'updaters': {
						/*'config-key': [
							function() {
								// This function contains updater for config['config-key'] from v0 to v1
							},
							function() {
								// This function contains updater for config['config-key'] from v1 to v2
							}
						]*/
					}
				}
			},
			checker: {
				type: 'switch',
				value: true
			},
			func: function() {
				//['message', 'success', 'warning', 'error'].forEach(type => alertify.notify(type, type, 0)); // alertify notifier test

				// Function load
				if (FuncInfo.data.load) {
					DoLog(LogLevel.Success, '测试函数启动了!');
				}

				// Basic storage
				if (FuncInfo.data.storage) {
					GM_setValue('test', true);
					DoLog(`读取test存储项的结果为${GM_getValue('test', 'default text')}`);
					DoLog(`读取unset存储项的结果为${GM_getValue('unset', 'default text')}`);
				}

				// Config Manager
				if (FuncInfo.data.manager) {
					const CM = new ConfigManager(FuncInfo.data.Config_Ruleset);
					const CONFIG = CM.Config;
					GM_setValue('testobj', {prop: 'value'});
					DoLog(`Object.keys(CONFIG).length = ${Object.keys(CONFIG).length}`);
					DoLog(`Object.keys(CONFIG.testobj) = ${Object.keys(CONFIG.testobj)}`);
				}

				// Error dealing
				if (FuncInfo.data.makeerror) {
					switch (FuncInfo.data.errortype) {
						case 'error':
							Err('An error occured!');
							break;
						case 'setTimeout':
							setTimeout(function() {Err('An error occured!')}, 0);
							break;
						case 'setTimeout_arrow':
							setTimeout(() => Err('An error occured!'), 0);
							break;
					}
				}
				if (FuncInfo.data.log && FuncInfo.data.log.length) {
					FuncInfo.data.log.forEach(v => console.log(v));
				}

				// Debug object
				if (FuncInfo.data.allowdebug) {
					Object.defineProperty(unsafeWindow, 'wd', {
						get: function() {
							// Leave a reference here so debugger a will be able to access them
							[require, FuncInfo, isWenkuFunction, Messager, FL_listFunctions, FL_getFunction, FL_enableFunction, FL_disableFunction, FL_loadSetting, FL_postMessage, FL_recieveMessage];
							[LogLevel, DoLog, Err, $, $All, $CrE, $AEL, $$CrE, addStyle, detectDom, destroyEvent, copyProp, copyProps, parseArgs, escJsStr, replaceText, getUrlArgv, dl_browser, dl_GM, AsyncManager];
							[ConfigManager, $URL, GreasyFork];
							[tippy, alertify, Sortable, vkbeautify, Darkmode, CryptoJS];

							debugger;
							return 'wenku8+ debugger: just hit "wd" and return';
						},
						configurable: true,
						enumerable: false,
					});
				}

				return {
					log() {DoLog('Debug function log')},
					number: 1,
					boolean: true,
					'null': null,
					'NaN': NaN,
					'undefined': undefined,
					arr: [0, 1, 2],
					obj: {}
				}
			}
		},

		// Common Style
		{
			name: 'CommonStyle',
			description: '基本组件 - 样式管理器',
			id: 'CommonStyle',
			system: true,
			checker: {
				type: 'switch',
				value: false
			},
			func: function() {
				const Assets = {
					ClassName: {
						Button: 'plus_button',
						Text: 'plus_text',
						TextLight: 'plus_text_light',
						Disabled: 'plus_disabled'
					},
					Color: {
						Text: 'rgb(30, 100, 220)',
						TextLight: 'rgb(70, 150, 230)',
						Button: 'rgb(0, 160, 0)',
						ButtonHover: 'rgb(0, 100, 0)',
						ButtonFocus: 'color: rgb(0, 100, 0)',
						ButtonDisabled: 'rgba(150, 150, 150)',
					}
				};
				// Add assets style
				Assets.ElmStyle = addStyle(
					`.${Assets.ClassName.Text} {color: ${Assets.Color.Text} !important;}` +
					`.${Assets.ClassName.TextLight} {color: ${Assets.Color.TextLight} !important;}` +
					`.${Assets.ClassName.Button} {color: ${Assets.Color.Button} !important; fill: ${Assets.Color.Button} !important; cursor: pointer !important; user-select: none;}` +
					`.${Assets.ClassName.Button}:hover {color: ${Assets.Color.ButtonHover} !important; fill: ${Assets.Color.ButtonHover} !important;}` +
					`.${Assets.ClassName.Button}:focus {${Assets.Color.ButtonFocus} !important; fill: ${Assets.Color.ButtonFocus} !important;}` +
					`.${Assets.ClassName.Button}.${Assets.ClassName.Disabled} {color: ${Assets.Color.ButtonDisabled} !important; fill: ${Assets.Color.ButtonDisabled} !important; cursor: not-allowed !important;}` +
					`.tippy-box[data-theme~="wenku_tip"] {background-color: #f0f7ff;color: black;border: 1px solid #a3bee8;}.tippy-box[data-theme~="wenku_tip"][data-placement^="top"]>.tippy-arrow::before {border-top-color: #a3bee8;}.tippy-box[data-theme~="wenku_tip"][data-placement^="left"]>.tippy-arrow::before {border-left-color: #a3bee8;}.tippy-box[data-theme~="wenku_tip"][data-placement^="right"]>.tippy-arrow::before {border-right-color: #a3bee8;}.tippy-box[data-theme~="wenku_tip"][data-placement^="bottom"]>.tippy-arrow::before {border-bottom-color: #a3bee8;}`
					, 'plus-commonstyle-assets'
				);
				// Add common style
				addStyle(`#dialog{z-index: 1600 !important;} #mask{z-index: 1500 !important;}`, 'plus-commonstyle-style');
				return Assets;
			}
		},

		// FontAwesome
		{
			name: 'FontAwesome',
			description: '基本组件 - FontAwesome(图标)',
			id: 'FontAwesome',
			system: true,
			func: function() {
				const urls = [
					'https://fastly.jsdelivr.net/npm/@fortawesome/fontawesome-free@6.5.1/css/all.min.css',
					'https://cdnjs.cloudflare.com/ajax/libs/font-awesome/6.4.0/css/all.min.css',
					'https://bowercdn.net/c/Font-Awesome-6.4.0/css/all.min.css',
					'https://cdn.jsdelivr.net/npm/@fortawesome/fontawesome-free@6.5.1/css/all.min.css',
				];
				const timeout = 15 * 1000;

				let i = 0, loaded = false, tid;
				const link = $$CrE({
					tagName: 'link',
					props: {
						href: urls[i],
						rel: 'stylesheet'
					},
					listeners: [
						['error', changeSource],
						['load', onload]
					]
				});
				tid = setTimeout(changeSource, timeout);
				document.head.appendChild(link);

				return {link, get loaded() {return loaded;}}

				function changeSource() {
					if (++i < urls.length) {
						clearTimeout(tid);
						tid = setTimeout(changeSource, timeout);
						link.href = urls[i];
					} else {
						Err('FontAwesome loading error');
					}
				}

				function onload(e) {
					clearTimeout(tid);
					loaded = true;
					FL_postMessage('FontAwesomeLoaded');
				}
			}
		},

		// Alertify
		{
			name: 'alertify',
			description: '基本组件 - alertify(对话框/信息栏)',
			id: 'alertify',
			system: true,
			func: function() {
				addStyle(GM_getResourceText('alertify-css'), 'plus-alertify-css');
				addStyle(GM_getResourceText('alertify-theme'), 'plus-alertify-theme');
				alertify.set('notifier','position', 'top-right');
			}
		},

		// Tippy
		{
			name: 'tippy',
			description: '基本组件 - tippy(浮动提示框)',
			id: 'tippy',
			system: true,
			func: function() {
				addStyle(GM_getResourceText('tippy-css'), 'plus-tippy-css');
			}
		},

		// Utils
		{
			name: '常用函数库',
			description: '脚本常用函数库',
			id: 'utils',
			system: true,
			checker: {
				type: 'switch',
				value: false
			},
			func: function() {
				return {getLang, getDocument, parseDocument, downloadText, setPageUrl, htmlEncode, htmlDecode, detectDom, randstr, randint, insertText, openDialog, refreshPage, testChecker, submitForm, formEncode, loadFuncs, encrypt, decrypt, getOS, getTime, deepClone};

				// Get language: 0 for simplyfied chinese and others, 1 for traditional chinese
				function getLang() {
					const match = document.cookie.match(/(; *)?jieqiUserCharset=(.+?)( *;|$)/);
					const nvgtLang = ({'zh-CN': 0, 'zh-TW': 1})[navigator.language] || 0;
					return match && match[2] ? (match[2].toLowerCase() === 'big5' ? 1 : 0) : nvgtLang;
				}

				// Download and parse a url page into a html document(dom).
				// when xhr onload: callback.apply(null, [dom, ...args])
				// Usage: getDocument({url, callback[, onerror][, args]}) | getDocument(url, callback) | getDocument(url, callback, args) | getDocument(url, callback, onerror, args)
				function getDocument() {
					const [url, callback, onerror, args] = parseArgs([...arguments], [
						function(args, defaultValues) {
							const arg = args[0];
							return ['url', 'callback', 'onerror', 'args'].map((prop, i) => arg.hasOwnProperty(prop) ? arg[prop] : defaultValues[i]);
						},
						[1, 2],
						[1, 2, 4],
						[1, 2, 3, 4]
					], ['', function() {}, logError, []]);

					GM_xmlhttpRequest({
						method       : 'GET',
						url          : url,
						responseType : 'blob',
						timeout      : 15 * 1000,
						onload       : function(response) {
							const htmlblob = response.response;
							parseDocument(htmlblob, callback, args);
						},
						onerror      : onerror,
						ontimeout    : onerror
					});

					function logError(e) {
						DoLog(LogLevel.Error, 'getDocument: Request Error');
						DoLog(LogLevel.Error, e, 'error');
					}
				}

				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);
					}
					const charset = ['GBK', 'BIG5'][getLang()];
					reader.readAsText(htmlblob, charset);
				}

				// Save text to textfile
				function downloadText(text, name) {
					if (!text || !name) {return false;};

					// Get blob url
					const blob = new Blob([text],{type:"text/plain;charset=utf-8"});
					const url = URL.createObjectURL(blob);

					// Create <a> and download
					const a = $CrE('a');
					a.href = url;
					a.download = name;
					a.click();
				}

				// Change location.href without reloading using history.pushState/replaceState
				// Usage: setPageUrl(url) | setPageUrl(win, url) | setPageUrl(win, url, push)
				function setPageUrl() {
					const [win, url, push] = parseArgs([...arguments], [
						[2],
						[1,2],
						[1,2,3]
					], [window, '', false]);
					return win.history[push ? 'pushState' : 'replaceState']({modified: true, ...history.state}, '', url);
				}

				// Encode text into html text format
				function htmlEncode(text) {
					const span = $CrE('div');
					span.innerText = text;
					return span.innerHTML;
				}

				// Decode html format text into pure text
				function htmlDecode(text) {
					const span = $CrE('div');
					span.innerHTML = text;
					return span.innerText;
				}

				// Returns a random string
				function randstr(length=16, cases=true, aviod=[]) {
					const all = 'abcdefghijklmnopqrstuvwxyz0123456789' + (cases ? 'ABCDEFGHIJKLMNOPQRSTUVWXYZ' : '');
					while (true) {
						let str = '';
						for (let i = 0; i < length; i++) {
							str += all.charAt(randint(0, all.length-1));
						}
						if (!aviod.includes(str)) {return str;};
					}
				}

				function randint(min, max) {
					return Math.floor(Math.random() * (max - min + 1)) + min;
				}

				// Insert into <textarea>
				function insertText(textarea, text, focus=true) {
					const pre_text = textarea.value.substring(0, textarea.selectionStart);
					const after_text = textarea.value.substring(textarea.selectionEnd);
					const curPos = {
						start: {
							atBegin: textarea.selectionStart === 0,
							atEnd: textarea.selectionStart === textarea.value.length,
							atNewLine: /[\r\n]/.test(textarea.value.charAt(textarea.selectionStart-1)), // this excludes the first line
							atWhitespace: /\t/.test(textarea.value.charAt(textarea.selectionStart-1))
						},
						end: {
							atBegin: textarea.selectionEnd === 0,
							atEnd: textarea.selectionEnd === textarea.value.length,
							atNewLine: /[\r\n]/.test(textarea.value.charAt(textarea.selectionEnd)),
						},
					};
					const textWithWhitespace = `${(curPos.start.atBegin || curPos.start.atNewLine || curPos.start.atWhitespace) ? '' : ' '}${text}${((curPos.start.atBegin || curPos.start.atNewLine || curPos.start.atWhitespace) && text.length === 0) ? '' : ' '}`;
					const position = textarea.selectionStart + textWithWhitespace.length;
					textarea.value = `${pre_text}${textWithWhitespace}${after_text}`;

					// Set selection position
					setTimeout(e => {
						const [scrollTop, scrollLeft] = [textarea.scrollTop, textarea.scrollLeft];
						focus && textarea.scrollIntoView();
						focus && textarea.focus();
						textarea.setSelectionRange(position, position);
						[textarea.scrollTop, textarea.scrollLeft] = [scrollTop, scrollLeft]
					}, 0);
				}

				// My openDialog, modified from original wenku8's openDialog, with onload and mask support, and NO CACHE unless showing images
				function openDialog(url, onload){
					!document.getElementById("mask") && unsafeWindow.showMask();
					const [dialogs, Ajax, displayDialog] = [unsafeWindow.dialogs, unsafeWindow.Ajax, unsafeWindow.displayDialog];
					if(url.match(/\.(gif|jpg|jpeg|png|bmp)$/ig)){
						dialogs[url]='<img src="'+url+'" class="imgdialog" onclick="closeDialog()" style="cursor:pointer" />';
						displayDialog(dialogs[url]);
					}else{
						Ajax.Request(url,{onLoading:function(){dialogs[url]=this.response; displayDialog('Loading...');}, onComplete:function(){dialogs[url]=this.response; displayDialog(this.response);typeof onload === 'function' && onload();}});
					}
				}

				// Refresh page without submitting forms again
				function refreshPage() {
					if (unsafeWindow.top.location.href.includes('#')) {
						unsafeWindow.top.location.reload();
					} else {
						unsafeWindow.top.location.href = unsafeWindow.top.location.href;
					}
				}

				// Check whether current page url matches FuncInfo.checker rule
				// This code is copy and modified from FunctionLoader.check
				function testChecker(checker) {
					if (!checker) {return true;}
					const values = Array.isArray(checker.value) ? checker.value : [checker.value]
					return values.some(value => {
						switch (checker.type) {
							case 'regurl': {
								return !!location.href.match(value);
							}
							case 'func': {
								try {
									return value();
								} catch (err) {
									DoLog(LogLevel.Error, CONST.Text.Loader.CheckerError);
									DoLog(LogLevel.Error, err);
									return false;
								}
							}
							case 'switch': {
								return value;
							}
							case 'starturl': {
								return location.href.startsWith(value);
							}
							case 'startpath': {
								return location.pathname.startsWidth(value);
							}
							default: {
								DoLog(LogLevel.Error, CONST.Text.Loader.CheckerInvalid);
								return false;
							}
						}
					});
				}

				// Ajax submit form
				// onload(blob_response) | onerror(err_instance)
				// The form serializing algorithm inside this function is not fully implemented yet. See comments in code.
				function submitForm(form, onload, onerror) {
					const data = serializeFormData(form);
					GM_xmlhttpRequest({
						method: 'POST',
						url: form.getAttribute('action'),
						headers: {'Content-Type': 'application/x-www-form-urlencoded'},
						responseType: 'blob',
						data: data,
						onload: resp => typeof onload === 'function' && onload(resp),
						onerror: err => typeof onerror === 'function' && onerror(err),
						ontimeout: err => typeof onerror === 'function' && onerror(err),
					});

					// Serialize form with GBK encoding
					// Supports string data only
					// This is not fully implemented. See https://url.spec.whatwg.org/#concept-urlencoded-serializer and https://html.spec.whatwg.org/multipage/form-control-infrastructure.html#form-submission-algorithm for further implementation
					function serializeFormData(form) {
						const data = new FormData(form);
						let string = '';
						for (const [key, value] of data) {
							string += `&${formEncode(key)}=${formEncode(value)}`;
						}
						return string;
					}
				}

				function formEncode(str) {
					return $URL.encode(normalizeLinefeeds(str));

					// Code from https://github.com/jimmywarting/FormData/ (MIT License)
					function normalizeLinefeeds(value) {
						return value.replace(/\r?\n|\r/g, '\r\n')
					}
				}

				// Load all function-objs provided in funcs asynchronously, and merge return values into one return obj
				// funcobj: {[checker], [detectDom], func}
				function loadFuncs(oFuncs) {
					const returnObj = {};

					oFuncs.forEach(oFunc => {
						if (!oFunc.checker || testChecker(oFunc.checker)) {
							if (oFunc.detectDom) {
								detectDom(oFunc.detectDom, e => execute(oFunc));
							} else {
								setTimeout(e => execute(oFunc), 0);
							}
						}
					});

					return returnObj;

					function execute(oFunc) {
						setTimeout(e => {
							const rval = oFunc.func(returnObj) || {};
							copyProps(rval, returnObj);
						}, 0);
					}
				}

				// Encrypt given text with key using AES-GCM (See https://developer.mozilla.org/zh-CN/docs/Web/API/SubtleCrypto/encrypt)
				// Returns: Encrypted string
				function encrypt(text, secret) {
					return CryptoJS.AES.encrypt(text, secret).toString();
				}

				// Decrypt given ArrayBuffer with key and iv using AES-GCM
				// Returns: Promise(text)
				function decrypt(ciphertext, secret) {
					const bytes  = CryptoJS.AES.decrypt(ciphertext, secret);
					const originalText = bytes.toString(CryptoJS.enc.Utf8);
					return originalText;
				}

				function getOS() {
					const info = (navigator.platform || navigator.userAgent).toLowerCase();
					const test = (s) => (info.includes(s));
					const map = {
						'Windows': ['window', 'win32', 'win64', 'win86'],
						'Mac': ['mac', 'os x'],
						'Linux': ['linux']
					}
					for (const [sys, strs] of Object.entries(map)) {
						if (strs.some(test)) {
							return sys;
						}
					}
					return 'Null';
				}

				// 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() {
					const [time, dateSpliter, timeSpliter] = parseArgs([...arguments], [
						[1],
						[2, 3],
						[1, 2, 3]
					], [new Date().getTime(), '-', ':']);

					const d = new Date(time);
					let fulltime = '';
					dateSpliter && (fulltime += [d.getFullYear().toString().padStart(4, '0'), (d.getMonth() + 1).toString().padStart(2, '0'), d.getDate().toString().padStart(2, '0')].join(dateSpliter));
					dateSpliter && timeSpliter && (fulltime += ' ');
					timeSpliter && (fulltime += [d.getHours().toString().padStart(2, '0'), d.getMinutes().toString().padStart(2, '0'), d.getSeconds().toString().padStart(2, '0')].join(timeSpliter));
					return fulltime;
				}

				// Deep copy an object
				function deepClone(obj) {
					/* No need for structuredClone
					if (typeof structuredClone === 'function') {
						return structuredClone(obj);
					}
					*/

					let newObj = typeof obj === 'object' && obj !== null ? (Array.isArray(obj) ? [] : {}) : obj;
					if (obj && typeof obj === "object") {
						for (let key in obj) {
							if (obj.hasOwnProperty(key)) {
								newObj[key] = (obj && typeof obj[key] === 'object') ? deepClone(obj[key]) : obj[key];
							}
						}
					}
					return newObj;
				}
			}
		},

		// Dialog
		{
			name: '对话框',
			description: '基础对话框功能,用于构建对话框式界面',
			id: 'dialog',
			system: true,
			checker: {
				type: 'switch',
				value: false
			},
			func: function() {
				addStyle('.dialog-mask {position: fixed;left: 0;top: 0;width: 100vw;height: 100vh;overflow: hidden;background: #FFFFFF20;z-index: 100;}.dialog-container {position: fixed;left: 0;top: 0;width: 100vw;height: 100vh;overflow: hidden;background: #00000000;z-index: 101;display: flex;align-items: center;justify-content: center;}.dialog-main {border-radius: 5px;background: #FFFFFF;box-shadow: 5px 5px 5px 0px rgba(0, 0, 0, 0.50);padding: 2vh 2vw;width: fit-content;}.plus-darkmode .dialog-main {background: #222222;}', 'plus-dialog');
				class dialog {
					#elements;
					#showing;

					constructor(content) {
						const elements = this.#elements = {};
						elements.mask = $$CrE({
							tagName: 'div',
							classes: 'dialog-mask'
						});
						elements.container = $$CrE({
							tagName: 'div',
							classes: 'dialog-container'
						});
						elements.main = $$CrE({
							tagName: 'div',
							classes: 'dialog-main'
						});
						elements.container.appendChild(elements.main);
						[elements.mask, elements.container].forEach(elm => document.body.appendChild(elm));

						if (typeof content === 'string') {
							elements.main.innerHTML = content;
						} else if (content instanceof HTMLElement) {
							elements.main.appendChild(content);
						}
						content ? this.show() : this.hide();
					}

					show() {
						['mask', 'container'].forEach(name => this.#elements[name].style.removeProperty('display'));
						this.#showing = true;
					}

					hide() {
						['mask', 'container'].forEach(name => this.#elements[name].style.display = 'none');
						this.#showing = false;
					}

					get elements() {
						return Object.assign({}, this.#elements);
					}

					get showing() {
						return this.#showing;
					}

					set showing(visible) {
						if (typeof visible === 'boolean') {
							visible ? this.show() : this.hide();
						}
						return this.#showing;
					}
				}

				return dialog;
			}
		},

		// SidePanel
		{
			name: '侧边工具栏',
			description: '基本组件 - 工具栏按钮管理器',
			id: 'SidePanel',
			system: true,
			func: function() {
				const utils = require('utils');
				const CONST = {
					CSS: {
						Sidepanel: '#sidepanel-panel {background-color: #00000000;z-index: 4000;}.sidepanel-button {font-size: 2vmin;color: #1E64DC;fill: #1E64DC;background-color: #FDFDFD;}.sidepanel-button:hover, .sidepanel-button.low-opacity:hover {opacity: 1;color: #FDFDFD;background-color: #1E64DC;}.sidepanel-button.low-opacity{opacity: 0.4 }.sidepanel-button>:is(i[class^="fa-"],svg) {line-height: 3vmin;width: 3vmin;height: 3vmin;}.sidepanel-button[class*="tooltip"]:hover::after {font-size: 0.9rem;top: calc((5vmin - 25px) / 2);}.sidepanel-button[class*="tooltip"]:hover::before {top: calc((5vmin - 12px) / 2);}.sidepanel-button.accept-pointer{pointer-events:auto;}',
						DefaultSVG: '.default-fa-spin {-webkit-animation-name: default-fa-spin;animation-name: default-fa-spin;-webkit-animation-duration: var(--fa-animation-duration,2s);animation-duration: var(--fa-animation-duration,2s);-webkit-animation-iteration-count: var(--fa-animation-iteration-count,infinite);animation-iteration-count: var(--fa-animation-iteration-count,infinite);-webkit-animation-timing-function: var(--fa-animation-timing,linear);animation-timing-function: var(--fa-animation-timing,linear);}@-webkit-keyframes default-fa-spin {0% {-webkit-transform: rotate(0deg);transform: rotate(0deg) }to {-webkit-transform: rotate(1turn);transform: rotate(1turn) }}@keyframes default-fa-spin {0% {-webkit-transform: rotate(0deg);transform: rotate(0deg) }to {-webkit-transform: rotate(1turn);transform: rotate(1turn) }}',
					},
					SVG: {
						DefaultSVG: '<svg xmlns="http://www.w3.org/2000/svg" height="1em" viewBox="0 0 512 512"><!--! Font Awesome Free 6.4.0 by @fontawesome - https://fontawesome.com License - https://fontawesome.com/license (Commercial License) Copyright 2023 Fonticons, Inc. --><path d="M304 48a48 48 0 1 0 -96 0 48 48 0 1 0 96 0zm0 416a48 48 0 1 0 -96 0 48 48 0 1 0 96 0zM48 304a48 48 0 1 0 0-96 48 48 0 1 0 0 96zm464-48a48 48 0 1 0 -96 0 48 48 0 1 0 96 0zM142.9 437A48 48 0 1 0 75 369.1 48 48 0 1 0 142.9 437zm0-294.2A48 48 0 1 0 75 75a48 48 0 1 0 67.9 67.9zM369.1 437A48 48 0 1 0 437 369.1 48 48 0 1 0 369.1 437z"/></svg>'
					}
				};
				return sideFunctions();

				// Side functions area
				function sideFunctions() {
					const SPanel = new SidePanel();
					SPanel.create();
					SPanel.setPosition('bottom-right');
					addStyle(CONST.CSS.Sidepanel, 'plus-sidepanel-css');
					addStyle(CONST.CSS.DefaultSVG, 'plus-sidepanel-defaultsvg');

					commonButtons();
					return SPanel;

					function commonButtons() {
						// Button show/hide-all-buttons
						const btnShowHide = SPanel.add({
							faicon: 'fa-solid fa-down-left-and-up-right-to-center',
							className: 'accept-pointer',
							tip: '隐藏面板',
							onclick: (function() {
								let hidden = false;
								return (e) => {
									hidden = !hidden;
									btnShowHide.faicon.className = 'fa-solid ' + (hidden ? 'fa-up-right-and-down-left-from-center' : 'fa-down-left-and-up-right-to-center');
									btnShowHide.classList[hidden ? 'add' : 'remove']('low-opacity');
									SPanel.setTooltip(btnShowHide, (hidden ? '显示面板' : '隐藏面板'));
									SPanel.elements.panel.style.pointerEvents = hidden ? 'none' : 'auto';
									for (const button of SPanel.elements.buttons) {
										if (button === btnShowHide) {continue;}
										//button.style.display = hidden ? 'none' : 'block';
										button.style.pointerEvents = hidden ? 'none' : 'auto';
										button.style.opacity = hidden ? '0' : '1';
									}
								};
							}) ()
						});

						// Button scroll-to-bottom
						const btnDown = SPanel.add({
							faicon: 'fa-solid fa-angle-down',
							tip: '转到底部',
							onclick: e => {
								const elms = [document.body.parentElement, $('#content'), $('#contentmain')];

								for (const elm of elms) {
									elm && elm.scrollTo && elm.scrollTo(elm.scrollLeft, elm.scrollHeight);
								}
							}
						});

						// Button scroll-to-top
						const btnUp = SPanel.add({
							faicon: 'fa-solid fa-angle-up',
							tip: '转到顶部',
							onclick: e => {
								const elms = [document.body.parentElement, $('#content'), $('#contentmain')];

								for (const elm of elms) {
									elm && elm.scrollTo && elm.scrollTo(elm.scrollLeft, 0);
								}
							}
						});

						// Darkmode (old version)
						/*
					const btnDarkmode = SPanel.add({
						faicon: 'fa-solid ' + (DMode.isActivated() ? 'fa-sun' : 'fa-moon'),
						tip: '明暗切换',
						onclick: (e) => {
							DMode.toggle();
							btnDarkmode.faicon.className = 'fa-solid ' + (DMode.isActivated() ? 'fa-sun' : 'fa-moon');
						}
					});
					*/

						// Refresh page
						const btnRefresh = SPanel.add({
							faicon: 'fa-solid fa-rotate-right',
							tip: '刷新页面',
							onclick: e => utils.refreshPage()
						});
					}
				}

				// Side-located control panel
				// Requirements: FontAwesome, tippy.js, addStyle
				// Use 'new' keyword
				function SidePanel() {
					// Public SP
					const SP = this;
					const elms = SP.elements = {};

					// Private _SP
					// keys start with '_' shouldn't be modified
					const _SP = {
						_id: {
							css: 'sidepanel-style',
							usercss: 'sidepanel-style-user',
							panel: 'sidepanel-panel'
						},
						_class: {
							button: 'sidepanel-button'
						},
						_directions: ['left', 'right', 'top', 'bottom']
					};

					addStyle('#sidepanel-panel {position: fixed; background-color: #00000000; padding: 0.5vmin; line-height: 3.5vmin; height: auto; display: flex; transition-duration: 0.3s; z-index: 9999999999;} #sidepanel-panel.right {right: 3vmin;} #sidepanel-panel.bottom {bottom: 3vmin; flex-direction: column-reverse;} #sidepanel-panel.left {left: 3vmin;} #sidepanel-panel.top {top: 3vmin; flex-direction: column;} .sidepanel-button {padding: 1vmin; margin: 0.5vmin; font-size: 3.5vmin; border-radius: 10%; text-align: center; color: #00000088; background-color: #FFFFFF88; box-shadow:3px 3px 2px #00000022; user-select: none; transition-duration: inherit;} .sidepanel-button:hover {color: #FFFFFFDD; background-color: #000000DD;}', 'plus-sidepanel');

					SP.create = function() {
						// Create panel
						const panel = elms.panel = $CrE('div');
						panel.id = _SP._id.panel;
						SP.setPosition('bottom-right');
						detectDom('body', e => document.body.appendChild(panel));

						// Prepare buttons
						elms.buttons = [];
					}

					// Insert a button to given index
					// details = {index, text, faicon, id, tip, className, onclick, listeners}, all optional
					// listeners = [..[..args]]. [..args] will be applied as button.addEventListener's args
					// faicon = 'fa-icon-name-classname fa-icon-style-classname', this arg stands for a FontAwesome icon to be inserted inside the botton
					// Returns the button(HTMLDivElement), including button.faicon(HTMLElement/HTMLSpanElement in firefox, <i>) if faicon is set
					SP.insert = function(details) {
						const index = details.index;
						const text = details.text;
						const faicon = details.faicon;
						const id = details.id;
						const tip = details.tip;
						const className = details.className;
						const onclick = details.onclick;
						const listeners = details.listeners || [];

						const button = $CrE('div');
						text && (button.innerHTML = text);
						id && (button.id = id);
						tip && setTooltip(button, tip);
						className && (button.className = className);
						onclick && (button.onclick = onclick);
						if (faicon) {
							// Show default svg before FontAwesome loaded
							button.innerHTML = CONST.SVG.DefaultSVG;
							button.children[0].classList.add('default-fa-spin');

							// Show faicon when FontAwesome loaded
							FL_recieveMessage('FontAwesomeLoaded', function() {
								// Remove default svg
								[...button.children].forEach(child => child.remove());

								// Show faicon
								const i = $CrE('i');
								i.className = faicon;
								button.faicon = i;
								button.appendChild(i);
							}, 'FontAwesome', true);
						}
						for (const listener of listeners) {
							$AEL(button, ...listener);
						}
						button.classList.add(_SP._class.button);

						elms.buttons = insertItem(elms.buttons, button, index);
						index < elms.buttons.length ? elms.panel.insertBefore(button, elms.panel.children[index]) : elms.panel.appendChild(button);
						return button;
					}

					// Append a button
					SP.add = function(details) {
						details.index = elms.buttons.length;
						return SP.insert(details);
					}

					// Remove a button
					SP.remove = function(arg) {
						let index, elm;
						if (arg instanceof HTMLElement) {
							elm = arg;
							index = elms.buttons.indexOf(elm);
						} else if (typeof(arg) === 'number') {
							index = arg;
							elm = elms.buttons[index];
						} else if (typeof(arg) === 'string') {
							elm = $(elms.panel, arg);
							index = elms.buttons.indexOf(elm);
						}

						elms.buttons = delItem(elms.buttons, index);
						elm.remove();
					}

					// Show entire panel
					SP.show = function() {
						elms.panel.style.removeProperty('display');
					}

					// Hide entire panel
					SP.hide = function() {
						elms.panel.style.display = 'none';
					}

					// Sets the display position by texts like 'right-bottom'
					SP.setPosition = function(pos) {
						const poses = _SP.direction = pos.split('-');
						const avails = _SP._directions;

						// Available check
						if (poses.length !== 2) {return false;}
						for (const p of poses) {
							if (!avails.includes(p)) {return false;}
						}

						// remove all others
						for (const p of avails) {
							elms.panel.classList.remove(p);
						}

						// add new pos
						for (const p of poses) {
							elms.panel.classList.add(p);
						}

						// Change tooltips' direction
						elms.buttons && elms.buttons.forEach(setTooltipDirection);
					}

					// Gets the current display position
					SP.getPosition = function() {
						return _SP.direction.join('-');
					}

					SP.setTooltip = setTooltip;
					SP.setTooltipDirection = setTooltipDirection;

					// Append a style text to document(<head>) with a <style> element
					// Replaces existing id-specificed <style>s
					function spAddStyle(css, id) {
						const style = document.createElement("style");
						id && (style.id = id);
						style.textContent = css;
						for (const elm of $All('#'+id)) {
							elm.parentElement && elm.parentElement.removeChild(elm);
						}
						document.head.appendChild(style);
					}

					// Set a tooltip to the element
					function setTooltip(elm, text, direction='auto') {
						elm._tippy ? elm._tippy.setContent(text) : tippy(elm, {
							content: text,
							theme: 'wenku_tip',
							arrow: true,
							hideOnClick: false
						});

						setTooltipDirection(elm, direction);
					}

					function setTooltipDirection(elm, direction='auto') {
						direction === 'auto' && (direction = _SP.direction.includes('left') ? 'right' : 'left');
						if (!_SP._directions.includes(direction)) {Err('setTooltip: invalid direction');}

						// Tippy direction
						if (!elm._tippy) {
							DoLog(LogLevel.Error, 'SidePanel.setTooltipDirection: Given elm has no tippy instance(elm._tippy)');
							Err('SidePanel.setTooltipDirection: Given elm has no tippy instance(elm._tippy)');
						}
						elm._tippy.setProps({
							placement: direction
						});
					}

					// Del an item from an array using its index. Returns the array but can NOT modify the original array directly!!
					function delItem(arr, index) {
						arr = arr.slice(0, index).concat(arr.slice(index+1));
						return arr;
					}

					// Insert an item into an array using given index. Returns the array but can NOT modify the original array directly!!
					function insertItem(arr, item, index) {
						arr = arr.slice(0, index).concat(item).concat(arr.slice(index));
						return arr;
					}
				}
			}
		},

		// SettingPanel
		{
			name: 'SettingPanel',
			description: '基本组件 - 设置类窗口',
			id: 'SettingPanel',
			system: true,
			checker: {
				type: 'switch',
				value: true
			},
			func: function() {
				const CommonStyle = require('CommonStyle');
				const utils = require('utils');
				const CONST = {
					Text: {
						Saved: '已保存',
						Reset: '已恢复到修改前',
						Boolean: ['否', '是'],
						Operations: '操作',
						Operation: {
							edit: '编辑',
							delete: '删除'
						}
					}
				};
				const SettingOptionElements = {
					'string': {
						createElement: function() {const e = $CrE('input'); e.style.width = 'calc(100% - 6px - 1ex)'; return e;},
						setValue: function(val) {this.element.value = val;},
						getValue: function() {return this.element.value;},
					},
					'number': {
						createElement: function() {const e = $CrE('input'); e.type = 'number'; e.style.width = 'calc(100% - 6px - 1ex)'; return e;},
						setValue: function (val) {this.element.value = val;},
						getValue: function() {return this.element.value;},
					},
					'boolean': {
						createElement: function() {const e = $CrE('input'); e.type = 'checkbox'; return e;},
						setValue: function(val) {this.element.checked = val;},
						getValue: function(data) {return this.element.checked ? (this.hasOwnProperty('data') ? this.data.checked : true) : (data ? this.data.unchecked : false);},
					},
					'select': {
						createElement: function() {const e = $CrE('select'); this.hasOwnProperty('data') && this.data.forEach((d) => {const o = $CrE('option'); o.innerText = d; e.appendChild(o)}); return e;},
						setValue: function(val) {Array.from(this.element.children).find((opt) => (opt.value === val)).selected = true;},
						getValue: function() {return this.element.value;},
					}
				}

				// initialize
				alertify.dialog('setpanel', function factory(){
					return {
						// The dialog startup function
						// This will be called each time the dialog is invoked
						// For example: alertify.myDialog( data );
						main:function(){
							// Split arguments
							let content, header, buttons, onsave, onreset, onclose;
							switch (arguments.length) {
								case 1:
									switch (typeof arguments[0]) {
										case 'string':
											content = arguments[0];
											break;
										case 'object':
											arguments[0].hasOwnProperty('content') && (content = arguments[0].content);
											arguments[0].hasOwnProperty('header')  && (header = arguments[0].header);
											arguments[0].hasOwnProperty('buttons') && (buttons = arguments[0].buttons);
											arguments[0].hasOwnProperty('onsave')  && (onsave = arguments[0].onsave);
											arguments[0].hasOwnProperty('onreset') && (onreset = arguments[0].onreset);
											arguments[0].hasOwnProperty('onclose') && (buttons = arguments[0].onclose);
											break;
										default:
											Err('Arguments invalid', 1);
									}
									break;
								case 2:
									content = arguments[0];
									header = arguments[1];
									break;
								case 3:
									content = arguments[0];
									header = arguments[1];
									buttons = buttons[2];
									break;
							}

							// Prepare dialog
							this.resizeTo('80%', '80%');
							content && this.setContent(content);
							header  && this.setHeader(header);
							onsave  && this.set('onsave', onsave);
							onreset && this.set('onreset', onreset);
							onclose && this.set('onclose', onclose);

							// Choose & show selected button groups
							const btnGroups = {
								// Close button only
								basic: [[1, 0]],

								// Save & reset button
								saver: [[0, 0], [1, 1]]
							};
							const group = btnGroups[buttons || 'basic'];
							const divs = ['auxiliary', 'primary'];
							divs.forEach((div) => {
								Array.from(this.elements.buttons[div].children).forEach((btn) => {
									btn.style.display = 'none';
								});
							});
							group.forEach((button) => {
								this.elements.buttons[divs[button[0]]].children[button[1]].style.display = '';
							});

							return this;
						},
						// The dialog setup function
						// This should return the dialog setup object ( buttons, focus and options overrides ).
						setup:function(){
							return {
								/* buttons collection */
								buttons:[{
									/* button label */
									text: '恢复到修改前',

									/*bind a keyboard key to the button */
									key: undefined,

									/* indicate if closing the dialog should trigger this button action */
									invokeOnClose: false,

									/* custom button class name  */
									className: alertify.defaults.theme.cancel,

									/* custom button attributes  */
									attrs: {},

									/* Defines the button scope, either primary (default) or auxiliary */
									scope:'auxiliary',

									/* The will conatin the button DOMElement once buttons are created */
									element: undefined
								},{
									/* button label */
									text: '关闭',

									/*bind a keyboard key to the button */
									key: undefined,

									/* indicate if closing the dialog should trigger this button action */
									invokeOnClose: true,

									/* custom button class name  */
									className: alertify.defaults.theme.ok,

									/* custom button attributes  */
									attrs: {},

									/* Defines the button scope, either primary (default) or auxiliary */
									scope:'primary',

									/* The will conatin the button DOMElement once buttons are created */
									element: undefined
								},{
									/* button label */
									text: '保存',

									/*bind a keyboard key to the button */
									key: undefined,

									/* indicate if closing the dialog should trigger this button action */
									invokeOnClose: false,

									/* custom button class name  */
									className: alertify.defaults.theme.ok,

									/* custom button attributes  */
									attrs: {},

									/* Defines the button scope, either primary (default) or auxiliary */
									scope:'primary',

									/* The will conatin the button DOMElement once buttons are created */
									element: undefined
								}],

								/* default focus */
								focus:{
									/* the element to receive default focus, has differnt meaning based on value type:
										number:     action button index.
										string:     querySelector to select from dialog body contents.
										function:   when invoked, should return the focus element.
										DOMElement: the focus element.
										object:     an object that implements .focus() and .select() functions.
									*/
									element: 0,

									/* indicates if the element should be selected on focus or not*/
									select: true

								},
								/* dialog options, these override the defaults */
								options: {
									title: 'Setting Panel',
									modal: true,
									basic: false,
									frameless: false,
									pinned: false,
									movable: true,
									moveBounded: false,
									resizable: true,
									autoReset: false,
									closable: true,
									closableByDimmer: true,
									maximizable: false,
									startMaximized: false,
									pinnable: false,
									transition: 'fade',
									padding: true,
									overflow: true,
									/*
									onshow:...,
									onclose:...,
									onfocus:...,
									onmove:...,
									onmoved:...,
									onresize:...,
									onresized:...,
									onmaximize:...,
									onmaximized:...,
									onrestore:...,
									onrestored:...
									*/
								}
							};
						},
						// This will be called once the dialog DOM has been created, just before its added to the document.
						// Its invoked only once.
						build:function(){

							// Do custom DOM manipulation here, accessible via this.elements

							// this.elements.root           ==> Root div
							// this.elements.dimmer         ==> Modal dimmer div
							// this.elements.modal          ==> Modal div (dialog wrapper)
							// this.elements.dialog         ==> Dialog div
							// this.elements.reset          ==> Array containing the tab reset anchor links
							// this.elements.reset[0]       ==> First reset element (button).
							// this.elements.reset[1]       ==> Second reset element (button).
							// this.elements.header         ==> Dialog header div
							// this.elements.body           ==> Dialog body div
							// this.elements.content        ==> Dialog body content div
							// this.elements.footer         ==> Dialog footer div
							// this.elements.resizeHandle   ==> Dialog resize handle div

							// Dialog commands (Pin/Maximize/Close)
							// this.elements.commands           ==> Object containing  dialog command buttons references
							// this.elements.commands.container ==> Root commands div
							// this.elements.commands.pin       ==> Pin command button
							// this.elements.commands.maximize  ==> Maximize command button
							// this.elements.commands.close     ==> Close command button

							// Dialog action buttons (Ok, cancel ... etc)
							// this.elements.buttons                ==>  Object containing  dialog action buttons references
							// this.elements.buttons.primary        ==>  Primary buttons div
							// this.elements.buttons.auxiliary      ==>  Auxiliary buttons div

							// Each created button will be saved with the button definition inside buttons collection
							// this.__internal.buttons[x].element

						},
						// This will be called each time the dialog is shown
						prepare:function(){
							// Do stuff that should be done every time the dialog is shown.
						},
						// This will be called each time an action button is clicked.
						callback:function(closeEvent){
							//The closeEvent has the following properties
							//
							// index: The index of the button triggering the event.
							// button: The button definition object.
							// cancel: When set true, prevent the dialog from closing.
							const myEvent = utils.deepClone(closeEvent);
							switch (closeEvent.index) {
								case 0: {
									// Reset button
									closeEvent.cancel = myEvent.cancel = true;
									myEvent.save = false;
									myEvent.reset = true;
									const onreset = this.get('onreset');
									typeof onreset === 'function' && onreset(myEvent);
									break;
								}
								case 1: {
									// Close button
									// Do something here if need
									break;
								}
								case 2: {
									// Save button
									closeEvent.cancel = myEvent.cancel = true;
									myEvent.save = true;
									myEvent.reset = false;
									const onsave = this.get('onsave');
									typeof onsave === 'function' && onsave(myEvent);
								}
							}
							myEvent.save && this.get('saver').call(this);
							myEvent.reset && this.get('reseter').call(this);
							closeEvent.cancel = myEvent.cancel;
						},
						// To make use of AlertifyJS settings API, group your custom settings into a settings object.
						settings:{
							onsave: function() {},
							onreset: function() {},
							options: [], // SettingOption array
							saver: function() {
								this.get('options').forEach(o => o.save());
							},
							reseter: function() {
								this.get('options').forEach(o => o.reset());
							}
						},
						// AlertifyJS will invoke this each time a settings value gets updated.
						settingUpdated:function(key, oldValue, newValue){
							// Use this to respond to specific setting updates.
							const _this = this;
							['onsave', 'onreset', 'saver', 'reseter'].includes(key) && check('function');
							['options'].includes(key) && check(Array);

							function rollback() {
								_this.set(key, oldValue);
							}

							function check(type) {
								valid(oldValue, type) && !valid(newValue, type) && rollback();
							}

							function valid(value, type) {
								return ({
									'string': () => (typeof value === type),
									'function': () => (value instanceof type)
								})[typeof type]();
							}
						},
						// listen to internal dialog events.
						hooks:{
							// triggered when the dialog is shown, this is seperate from user defined onshow
							onshow: function() {
								this.resizeTo('80%', '80%');
							},
							// triggered when the dialog is closed, this is seperate from user defined onclose
							onclose: function() {
								const onclose = this.get('onclose');
								typeof onclose === 'function' && onclose();
							},
							// triggered when a dialog option gets updated.
							// IMPORTANT: This will not be triggered for dialog custom settings updates ( use settingUpdated instead).
							onupdate: function() {
							}
						}
					}
				}, true);

				return {
					easyStorage,
					easySettings,
					SettingPanel,
					SettingOption,
					optionAvailable,
					isOption,
					registerElement,
				};

				// Storage management panel for structured data like [{prop1, prop2, ...}, ...]
				// details: {String path, String key, [title|panel],
				//     props: {'prop1': {type: 'string'/'number'/'time'/'boolean', name, [editable], [listeners], [styles], [oncreate]}}},
				//     operations: [{type: 'delete'/'edit', [text]}, ...]
				function easyStorage(details, CM) {
					!optionAvailable('storage') && register();

					const CONFIG = CM.Config;
					const panel = details.panel || new SettingPanel({
						buttons: 'saver',
						header: details.title || ''
					}, CM);

					const option = new SettingOption({ type: 'storage', data: { details, panel }, path: details.path }, CM);
					panel.alertifyBox.get('options').push(option);
					panel.appendTable(option.data.table);

					function register() {
						registerElement('storage', {
							createElement() {
								this.data = {...this.data};
								const panel = this.data.panel;
								const details = this.data.details;

								const colCount = Object.keys(details.props).length + details.operations.length;
								const table = new panel.PanelTable();
								this.data.table = table;
								makeHeader(this);

								return table.element;
							},
							setValue(val) {
								this.data.data = val;
								this.data.table.rows.slice().forEach(row => row.remove());
								makeHeader(this);
								makeRows(this, val);
							},
							getValue() {
								return this.data.data;
							}
						});

						function makeHeader(Option) {
							const panel = Option.data.panel;
							const table = Option.data.table;
							const details = Option.data.details;

							const colCount = Object.keys(details.props).length + details.operations.length;
							table.appendRow({
								blocks: [{
									isHeader: true,
									colSpan: colCount,
									innerText: details.title,
								}]
							});
							table.appendRow({
								blocks: [
									...Object.values(details.props).map(p => ({
										innerText: p.name,
										width: p.width || Math.floor(100 / colCount).toString() + '%',
										style: { 'text-align': 'center' }
									})),
									{ innerText: CONST.Text.Operations, colSpan: details.operations.length, style: { 'text-align': 'center' } },
								]
							});
						}

						function makeRows(Option, storageArr) {
							const panel = Option.data.panel;
							const table = Option.data.table;
							const details = Option.data.details;

							for (const item of storageArr) {
								const row = new panel.PanelRow();

								// Properties
								for (const [p, prop] of Object.entries(details.props)) {
									const block = new panel.PanelBlock({
										innerText: {
											string: v => v,
											number: v => v.toString(),
											time: v => utils.getTime(v),
											boolean: v => CONST.Text.Boolean[+v]
										}[prop.type](item[p]),
										style: { 'text-align': 'center', ...(prop.styles || {}) },
										listeners: (prop.listeners || []).map(listener => {
											const lis = [...listener];
											const func = lis[1];
											lis[1] = function() { func.apply(this, [item[details.key], ...arguments]); }
											return lis;
										})
									});
									typeof prop.oncreate === 'function' && prop.oncreate(item[details.key], block);
									row.appendBlock(block);
								}

								// Operations
								details.operations.map(op => $$CrE({
									tagName: 'span',
									props: { innerText: op.text || CONST.Text.Operation[op.type] },
									classes: [CommonStyle.ClassName.Button],
									listeners: [['click', e => {
										const items = Option.data.data;
										const index = items.findIndex(it => it[details.key] === item[details.key]);
										({
											delete: e => {
												items.splice(index, 1);
												row.remove();
												table.element.dispatchEvent(new Event('change'));
											},
											edit: e => { /* Edit function here */ console.log(`edit ${item[details.key]}`); },
											func: () => Option.data.data = op.func(e, items, index)
										})[op.type]();
									}]]
								})).forEach(span => row.appendBlock({ children: [span], style: { 'text-align': 'center' } }));

								table.appendRow(row);
							}
						}
					}
				}

				// Creates a simple SettingPanel in a few code (or append tables to given settingpanel)
				// details: {Array areas, [title, panel]} or area; area: {title, itmes: [{text, path, type}]}
				function easySettings(details, CM) {
					details = details.areas ? details : {areas: [details], title: details.title};
					const panel = details.panel || new SettingPanel({
						buttons: 'saver',
						header: details.title
					}, CM);

					for (const area of details.areas) {
						const table = new panel.PanelTable({
							rows: area.title ? [{
								blocks: [{
									isHeader: true,
									colSpan: 2,
									innerText: area.title,
								}]
							}] : []
						});
						for (const item of area.items) {
							const option = {};
							copyProps(item, option, ['type', 'path', 'checker', 'children']);
							table.appendRow({
								blocks: [{
									innerText: item.text
								}, {
									options: [option]
								}]
							});
						}
						panel.appendTable(table);
					}

					return panel;
				}

				// A table-based setting panel using alertify-js
				// For wenku8++ only version
				// Use 'new' keyword
				// Usage:
				/*
					var panel = new SettingPanel({
						buttons: 0,
						header: '',
						className: '',
						id: '',
						name: '',
						tables: [
							{
								className: '',
								id: '',
								name: '',
								rows: [
									{
										className: '',
										id: '',
										name: '',
										blocks: [
											{
												isHeader: false,
												width: '',
												height: '',
												innerHTML / innerText: ''
												colSpan: 1,
												rowSpan: 1,
												className: '',
												id: '',
												name: '',
												options: [SettingOption, ...]
												children: [HTMLElement, ...]
											},
											...
										]
									},
									...
								]
							},
							...
						]
					});
				*/
				function SettingPanel(details={}, CM=null) {
					const SP = this;
					SP.insertTable = insertTable;
					SP.appendTable = appendTable;
					SP.removeTable = removeTable;
					SP.remove = remove;
					SP.PanelTable = PanelTable;
					SP.PanelRow = PanelRow;
					SP.PanelBlock = PanelBlock;

					// <div> element
					const elm = $CrE('div');
					copyProps(details, elm, ['id', 'name', 'className']);
					elm.classList.add('settingpanel-container');

					// Make alerity box
					const box = SP.alertifyBox = alertify.setpanel({
						onsave: function() {
							alertify.notify(CONST.Text.Saved);
						},
						onreset: function() {
							alertify.notify(CONST.Text.Reset);
						},
						buttons: details.hasOwnProperty('buttons') ? details.buttons : 'basic'
					});
					clearChildNodes(box.elements.content);
					box.elements.content.appendChild(elm);
					box.elements.content.style.overflow = 'auto';
					box.setHeader(details.header);
					box.setting({
						maximizable: true,
						overflow: true
					});
					details.onclose && box.setting('onclose', details.onclose);
					!box.isOpen() && box.show();

					// Configure object
					let css='', usercss='';
					SP.element = elm;
					SP.tables = [];
					SP.length = 0;
					copyProps(details, SP, ['id', 'name']);
					Object.defineProperty(SP, 'css', {
						configurable: false,
						enumerable: true,
						get: function() {
							return css;
						},
						set: function(_css) {
							addStyle(_css, 'settingpanel-css');
							css = _css;
						}
					});
					Object.defineProperty(SP, 'usercss', {
						configurable: false,
						enumerable: true,
						get: function() {
							return usercss;
						},
						set: function(_usercss) {
							addStyle(_usercss, 'settingpanel-usercss');
							usercss = _usercss;
						}
					});
					SP.css = `.settingpanel-table {border-spacing: 0px; border-collapse: collapse; width: 100%; margin: 2em 0;} .settingpanel-block {border: 1px solid ${CommonStyle.Color.Text}; text-align: center; vertical-align: middle; padding: 3px; text-align: left;} .settingpanel-header {font-weight: bold;} td.settingpanel-block:nth-of-type(1):is(:not([colspan]),[colspan="1"]){width: 30%;}`;

					// Create tables
					if (details.tables) {
						for (const table of details.tables) {
							if (table instanceof PanelTable) {
								appendTable(table);
							} else {
								appendTable(new PanelTable(table));
							}
						}
					}

					// Insert a Panel-Row
					// Returns Panel object
					function insertTable(table, index) {
						// Insert table
						!(table instanceof PanelTable) && (table = new PanelTable(table));
						index < SP.length ? elm.insertBefore(table.element, elm.children[index]) : elm.appendChild(table.element);
						insertItem(SP.tables, table, index);
						table.id !== undefined && (SP.children[table.id] = table);
						SP.length++;

						// Set parent
						table.parent = SP;
					}

					// Append a Panel-Row
					// Returns Panel object
					function appendTable(table) {
						return insertTable(table, SP.length);
					}

					// Remove a Panel-Row
					// Returns Panel object
					function removeTable(index) {
						const table = SP.tables[index];
						SP.element.removeChild(table.element);
						removeItem(SP.rows, index);
						return SP;
					}

					// Remove itself from parentElement
					// Returns Panel object
					function remove() {
						SP.element.parentElement && SP.parentElement.removeChild(SP.element);
						return SP;
					}

					// Panel-Table object
					// Use 'new' keyword
					function PanelTable(details={}) {
						const PT = this;
						PT.insertRow = insertRow;
						PT.appendRow = appendRow;
						PT.removeRow = removeRow;
						PT.remove = remove

						// <table> element
						const elm = $CrE('table');
						copyProps(details, elm, ['id', 'name', 'className']);
						elm.classList.add('settingpanel-table');

						// Configure
						PT.element = elm;
						PT.rows = [];
						PT.length = 0;
						copyProps(details, PT, ['id', 'name']);

						// Append rows
						if (details.rows) {
							for (const row of details.rows) {
								if (row instanceof PanelRow) {
									insertRow(row);
								} else {
									insertRow(new PanelRow(row));
								}
							}
						}

						// Insert a Panel-Row
						// Returns Panel-Table object
						function insertRow(row, index) {
							// Insert row
							!(row instanceof PanelRow) && (row = new PanelRow(row));
							index < PT.length ? elm.insertBefore(row.element, elm.children[index]) : elm.appendChild(row.element);
							insertItem(PT.rows, row, index);
							PT.length++;

							// Set parent
							row.parent = PT;
							return PT;
						}

						// Append a Panel-Row
						// Returns Panel-Table object
						function appendRow(row) {
							return insertRow(row, PT.length);
						}

						// Remove a Panel-Row
						// Returns Panel-Table object
						function removeRow(index) {
							const row = PT.rows[index];
							PT.element.removeChild(row.element);
							removeItem(PT.rows, index);
							return PT;
						}

						// Remove itself from parentElement
						// Returns Panel-Table object
						function remove() {
							PT.parent instanceof SettingPanel && PT.parent.removeTable(PT.tables.indexOf(PT));
							return PT;
						}
					}

					// Panel-Row object
					// Use 'new' keyword
					function PanelRow(details={}) {
						const PR = this;
						PR.insertBlock = insertBlock;
						PR.appendBlock = appendBlock;
						PR.removeBlock = removeBlock;
						PR.remove = remove;

						// <tr> element
						const elm = $CrE('tr');
						copyProps(details, elm, ['id', 'name', 'className']);
						elm.classList.add('settingpanel-row');

						// Configure object
						PR.element = elm;
						PR.blocks = [];
						PR.length = 0;
						copyProps(details, PR, ['id', 'name']);

						// Append blocks
						if (details.blocks) {
							for (const block of details.blocks) {
								if (block instanceof PanelBlock) {
									appendBlock(block);
								} else {
									appendBlock(new PanelBlock(block));
								}
							}
						}

						// Insert a Panel-Block
						// Returns Panel-Row object
						function insertBlock(block, index) {
							// Insert block
							!(block instanceof PanelBlock) && (block = new PanelBlock(block));
							index < PR.length ? elm.insertBefore(block.element, elm.children[index]) : elm.appendChild(block.element);
							insertItem(PR.blocks, block, index);
							PR.length++;

							// Set parent
							block.parent = PR;
							return PR;
						};

						// Append a Panel-Block
						// Returns Panel-Row object
						function appendBlock(block) {
							return insertBlock(block, PR.length);
						}

						// Remove a Panel-Block
						// Returns Panel-Row object
						function removeBlock(index) {
							const block = PR.blocks[index];
							PR.element.removeChild(block.element);
							removeItem(PR.blocks, index);
							return PR;
						}

						// Remove itself from parent
						// Returns Panel-Row object
						function remove() {
							PR.parent instanceof PanelTable && PR.parent.removeRow(PR.parent.rows.indexOf(PR));
							return PR;
						}
					}

					// Panel-Block object
					// Use 'new' keyword
					function PanelBlock(details={}) {
						const PB = this;
						PB.remove = remove;

						// <td> element
						const elm = $CrE(details.isHeader ? 'th' : 'td');
						copyProps(details, elm, ['innerText', 'innerHTML', 'colSpan', 'rowSpan', 'id', 'name', 'className']);
						copyProps(details, elm.style, ['width', 'height']);
						details.style && copyProps(details.style, elm.style);
						elm.classList.add('settingpanel-block');
						details.isHeader && elm.classList.add('settingpanel-header');
						details.listeners && details.listeners.forEach(listener => $AEL(elm, ...listener))

						// Configure object
						PB.element = elm;
						copyProps(details, PB, ['id', 'name']);

						// Append to parent if need
						details.parent instanceof PanelRow && (PB.parent = details.parent.appendBlock(PB));

						// Append SettingOptions if exist
						if (details.options) {
							(CM ? details.options : details.options.filter(isOption)).map((o) => (isOption(o) ? o : new SettingOption(o, CM))).forEach(function(option) {
								SP.alertifyBox.get('options').push(option);
								elm.appendChild(option.element);
							});
						}

						// Append child elements if exist
						if (details.children) {
							for (const child of details.children) {
								elm.appendChild(child);
							}
						}

						// Remove itself from parent
						// Returns Panel-Block object
						function remove() {
							PB.parent instanceof PanelRow && PB.parent.removeBlock(PB.parent.blocks.indexOf(PB));
							return PB;
						}
					}

					function insertItem(arr, item, index) {
						arr.splice(index, 0, item);
						return arr;
					}
					function removeItem(arr, index) {
						arr.splice(index, 1);
						return arr;
					}
					function MakeReadonlyObj(val) {
						return isObject(val) ? new Proxy(val, {
							get: function(target, property, receiver) {
								return MakeReadonlyObj(target[property]);
							},
							set: function(target, property, value, receiver) {},
							has: function(target, prop) {}
						}) : val;

						function isObject(value) {
							return ['object', 'function'].includes(typeof value) && value !== null;
						}
					}
				}

				// details = {path='config path', type='config type', data='option data'}
				function SettingOption(details={}, CM) {
					const SO = this;
					SO.save = save;
					SO.reset = reset;

					// Initialize ConfigManager
					!CM && Err('SettingOption requires a ConfigManager instance');
					const CONFIG = CM.Config;

					// Get args
					const options = ['path', 'type', 'checker', 'data', 'autoSave'];
					copyProps(details, SO, options);

					// Get first available type if multiple types provided
					Array.isArray(SO.type) && (SO.type = SO.type.find((t) => (optionAvailable(t))));
					!optionAvailable(SO.type) && Err('Unsupported Panel-Option type: ' + details.type);

					// Create element
					const original_value = CM.getConfig(SO.path);
					const SOE = {
						create: SettingOptionElements[SO.type].createElement.bind(SO),
						get: SettingOptionElements[SO.type].getValue.bind(SO),
						set: v => SettingOptionElements[SO.type].setValue.call(SO, utils.deepClone(v)),
					}
					SO.element = SOE.create();
					SOE.set(original_value);

					// Bind change-checker-saver
					SO.element.addEventListener('change', function(e) {
						if (SO.checker) {
							if (SO.checker(e, SOE.get())) {
								SO.autoSave && save();
							} else {
								// Reset value
								reset();

								// Do some value-invalid reminding here
							}
						} else {
							SO.autoSave && save();
						}
					});

					function save() {
						CM.setConfig(SO.path, SOE.get());
					}

					function reset(save=false) {
						SOE.set(original_value);
						save && CM.setConfig(SO.path, original_value);
					}
				}

				// Check if an settingoption type available
				function optionAvailable(type) {
					return Object.keys(SettingOptionElements).includes(type);
				}

				// Register SettingOption element
				function registerElement(name, obj) {
					const formatOkay = typeof obj.createElement === 'function' && typeof obj.setValue === 'function' && typeof obj.getValue === 'function';
					const noConflict = !SettingOptionElements.hasOwnProperty(name);
					const okay = formatOkay && noConflict;
					okay && (SettingOptionElements[name] = obj);
					return okay;
				}

				function isOption(obj) {
					return obj instanceof SettingOption;
				}

				function clearChildNodes(elm) {[...elm.childNodes].forEach(el => el.remove());}
			}
		},

		// MouseTip
		{
			name: '鼠标提示框',
			description: '兼容与扩充文库自带的鼠标跟随提示框',
			id: 'mousetip',
			system: true,
			func: function() {
				let tipready = tipcheck();
				addStyle('#tips {z-index: 10000;}', 'plus_mousetip');
				tipscroll();
				tipiframe();
				const return_value = {
					get tipready() {return tipready},
					settip: settip,
					showtip: showtip,
					hidetip: hidetip
				};
				return return_value;

				// Check if tipobj is ready, if not, then make it
				function tipcheck() {
					DoLog('checking tipobj...');
					if (typeof unsafeWindow.tipobj === 'object' && unsafeWindow.tipobj !== null) {
						DoLog('tipobj ready');
						return true;
					} else {
						if (typeof(tipinit) === 'function') {
							unsafeWindow.tipinit();
							DoLog('tipinit executed');
							return true;
						} else {
							if (!$('a[href*="/themes/wenku8/theme.js"]')) {
								loadTipScript();
							} else {
								DoLog(LogLevel.Error, 'tip init error: theme script exist but tipobj and tipinit missing');
							}
							return false;
						}
					}

					function loadTipScript() {
						const s = $CrE('script');
						s.src = `https://${location.host}/themes/wenku8/theme.js`;
						document.head.appendChild(s);

						$AEL(s, 'load', e => {
							tipready = true;
							if (document.readyState === 'loading') {
								// Wait for tipinit triggered
								$AEL(unsafeWindow, 'load', tipscroll);
							} else {
								// Too late, call tipinit manually
								unsafeWindow.tipinit();
								tipscroll();
							}
							DoLog('theme script loaded');
						});

						DoLog('theme script appended');
					}
				}

				// New tipobj movement method. Makes sure the tipobj stay close with the mouse.
				function tipscroll() {
					if (!tipready) {return false;}

					DoLog('tipscroll executed. ');
					unsafeWindow.tipobj.style.position = 'fixed';
				}

				function tipiframe() {
					$AEL(unsafeWindow, 'mousemove', tipmoveplus);
					return true;

					function tipmoveplus(e) {
						// Move
						if (unsafeWindow.tipobj) {
							unsafeWindow.tipobj.style.left = e.clientX + unsafeWindow.tipx + 'px';
							unsafeWindow.tipobj.style.top = e.clientY + unsafeWindow.tipy + 'px';
						}

						// Call parent
						if (unsafeWindow !== unsafeWindow.parent) {
							const parent = unsafeWindow.parent;
							const doc = parent.document;
							const iframe = [...$All(doc, 'iframe')].find(ifr => ifr.contentDocument === document);
							const clientRect = iframe.getClientRects()[0];
							const topevt = new CustomEvent('mousemove');
							topevt.clientX = clientRect.left + e.clientX;
							topevt.clientY = clientRect.top + e.clientY;
							parent.dispatchEvent(topevt);
						}
					}
				}

				// show & hide tip when mouse in & out. accepts tip as a string or a function that returns the tip string
				function settip(elm, tip) {
					elm[{'string': 'tiptitle', 'function': 'tipgetter'}[typeof(tip)]] = tip;
					elm.removeEventListener('mouseover', showtip);
					elm.removeEventListener('mouseout', hidetip);
					$AEL(elm, 'mouseover', showtip, {capture: true});
					$AEL(elm, 'mouseout', hidetip, {capture: true});
				}

				function showtip(e) {
					if (!tipready) {return false;}

					switch (typeof e) {
						case 'object': {
							if (e[Symbol.toStringTag] === 'MouseEvent') {
								const elm = e.currentTarget;
								if (elm.tiptitle || elm.tipgetter) {
									const tip = elm.tiptitle || elm.tipgetter();
									tipready && unsafeWindow.tipshow(tip);
								}
							}
							break;
						}
						case 'string': {
							unsafeWindow.tipshow(e);
							break;
						}
					}
				}

				function hidetip() {
					tipready && unsafeWindow.tiphide();
				}
			}
		},

		// AndriodAPI
		{
			name: '安卓API',
			description: '文库安卓API,用于获取网页端没有的资源',
			id: 'AndroidAPI',
			system: true,
			checker: {
				type: 'switch',
				value: false
			},
			func: function() {
				return new AndroidAPI();

				// Android API set
				function AndroidAPI() {
					const AA = this;
					const DParser = new DOMParser();

					const encode = AA.encode = function(str) {
						return '&appver=1.13&request=' + btoa(str) + '&timetoken=' + (new Date().getTime());
					};

					const request = AA.request = function(details) {
						const url = details.url;
						const type = details.type || 'text';
						const callback = details.callback || function() {};
						const args = details.args || [];
						GM_xmlhttpRequest({
							method: 'POST',
							url: 'http://app.wenku8.com/android.php',
							headers: {
								'Content-Type': 'application/x-www-form-urlencoded',
								'User-Agent': 'Dalvik/2.1.0 (Linux; U; Android 7.1.2; unknown Build/NZH54D)'
							},
							data: encode(url),
							onload: function(e) {
								let result;
								switch (type) {
									case 'xml':
										result = DParser.parseFromString(e.responseText, 'text/xml');
										break;
									case 'text':
										result = e.responseText;
										break;
								}
								callback.apply(null, [result].concat(args));
							},
							onerror: function(e) {
								Err('Request error while requesting "' + url + '"');
							}
						});
					};

					// aid, lang, callback, args
					AA.getNovelShortInfo = function(details) {
						const aid = details.aid;
						const lang = details.lang;
						const callback = details.callback || function() {};
						const args = details.args || [];
						const url = 'action=book&do=info&aid=' + aid + '&t=' + lang;
						request({
							url: url,
							callback: callback,
							args: args,
							type: 'xml'
						});
					}

					// aid, lang, callback, args
					AA.getNovelIndex = function(details) {
						const aid = details.aid;
						const lang = details.lang;
						const callback = details.callback || function() {};
						const args = details.args || [];
						const url = 'action=book&do=list&aid=' + aid + '&t=' + lang;
						request({
							url: url,
							callback: callback,
							args: args,
							type: 'xml'
						});
					};

					// aid, cid, lang, callback, args
					AA.getNovelContent = function(details) {
						const aid = details.aid;
						const cid = details.cid;
						const lang = details.lang;
						const callback = details.callback || function() {};
						const args = details.args || [];
						const url = 'action=book&do=text&aid=' + aid + '&cid=' + cid + '&t=' + lang;
						request({
							url: url,
							callback: callback,
							args: args,
							type: 'text'
						});
					};

					// Requires user logged in with AndroidAPI, which is not implemented yet
					AA.getUserInfo = function(details) {
						const callback = details.callback || function() {};
						const args = details.args || [];
						const url = 'action=userinfo';
						request({
							url: url,
							callback: callback,
							args: args,
							type: 'xml'
						});
					}
				}
			}
		},

		// Settings
		{
			name: '设置',
			description: '基本组件 - 设置',
			id: 'settings',
			system: true,
			CONST: {
				Number: {
					GFScriptID: 416310,
				},
				Text: {
					SettingsOfUserscript: '脚本自身设置',
					SettingsOfFunctions: '脚本功能设置',
					ScriptVersion: '版本号',
					CheckUpdate: '检查更新',
					ClickMeToCheck: '点我检查更新',
					ExportDebugInfo: '导出调试包',
					ExportConfig: '导出配置',
					ImportConfig: '导入配置',
					ClickMeToExport: '点我导出',
					ClickMeToImport: '点我导入',
					ImportSuccess: '导入成功',
					ImportFailed: '导入失败,错误码: {ErrorCode}',
					Settings: '设置',
					Function: '功能',
					Operation: '操作',
					ShowSystemFuncs: '显示系统组件',
					HideSystemFuncs: '隐藏系统组件',
					SystemFunc: '[系统组件]',
					NonSystemFunc: '[功能模块]',
					DisableFunction: '停用此功能',
					FunctionDisabled: '功能 {FuncName} 已停用',
					EnableFunction: '启用此功能',
					FunctionEnabled: '功能 {FuncName} 已启用',
					FunctionManagement: '管理此功能',
					FunctionSettings: '功能设置',
					RememberSaving: '修改设置/恢复设置后记得点击保存哦:)',
					NoSetting: '此功能没有设置项!',
					FunctionDetail: '<span class="{CT}">名称:</span><span data-key="name"></span></br> <span class="{CT}">描述:</span><span data-key="description"></span></br> <span class="{CT}">ID:</span><span data-key="id"></span></br> <span class="{CT}">系统组件:</span><span data-key="system" use-false-while-empty></span></br> <span class="{CT}">是否已启用:</span><span data-key="enabled"></span></br> ',
					FunctionDetail_EmptyItem: '无',
					Boolean: {
						'true': '是',
						'false': '否'
					},
					ManageWindow: {
						Header: '功能管理',
						EnableTitle: '启用',
						StorageTitle: '存储',
						Save: '保存',
						Reload: '重新载入',
						Export: '导出配置',
						Import: '导入配置',
						Reset: '恢复出厂配置',
						SystemFunc: '[系统组件]',
						NonSystemFunc: '[功能模块]',
						SysAlert: '系统组件的配置仅供查看',
						StorageAlert: '注意:大多数功能的设置都可以通过[功能设置](而不是这里的[管理此功能])进行直观的修改,需要直接在此处进行的手动修改几乎不存在!直接修改功能模块的配置存储很有可能会导致未知的错误,除非你明确的知道你要改什么,否则就不要修改!</br>确定仍然要进行手动修改?',
						StorageSaved: '配置存储已保存',
						StorageEdit: '启用编辑',
						StorageFormatError: '配置存储格式不正确,请检查修正后再保存',
						StorageResetConfirm: '真的要恢复出厂配置吗?</br>如果恢复出厂配置,所有此功能的用户设置都需要重新手动设置。</br>此功能一般用于排除bug,建议您先导出配置进行备份后再恢复出厂配置,并且推荐您在恢复出厂配置后对相关页面先刷新再使用。'
					},
					CheckingUpdate: `${GM_info.script.name}: 正在检查更新...`,
					UpdateFound: {
						Main: `${GM_info.script.name}: 有新版本啦!`,
						View: '[点击 查看 更新]',
						Install: '[点击 安装 更新]'
					},
					UpdateInfo: {
						Header: `${GM_info.script.name} v{version}`,
						Install: '安装',
						Close: '关闭'
					}
				},
				Faicon: {
					Info: 'fa-solid fa-circle-info'
				},
				CSS: {
					Common: '.plus_system:not(.plus_system_show) {display: none;}',
					Manager: '.plus_func_dialog:not(#a) {height: 80vh;min-width: 80vw;}.plus_func_titleblock {display: flex;}.plus_func_icon {height: 50px;}.plus_func_namearea {flex-grow: 1;position: relative;margin-left: 10px;display: flex;flex-direction: column;}.plus_func_nameline {}.plus_func_nameline>* {margin-right: 5px;}.plus_func_name {font-size: 30px;}.plus_func_version {font-size: 20px;}.plus_func_system {font-size: 10px;}.plus_func_descline {font-size: 10px;position: absolute;bottom: 0;}.plus_func_descline>* {margin-right: 3px;}.plus_func_sysalert {margin-top: 5px;color: #f66a13;background-color: #f5eba4;font-size: 15px;line-height: 23px;text-align: center;}.plus_func_block {margin-top: 5px;border-top: 1px solid lightgrey;display: flex;}.plus_func_blocktitle {padding: 1em;text-align: left;vertical-align: top;display: table-cell;width: calc(50px - 2em);}.plus_func_blockcontent {padding: 1em 0;text-align: left;vertical-align: top;display: table-cell;flex-grow: 1;}.plus_func_block [disabled] {cursor: not-allowed;}.plus_func_storageeditor {width: 100%;height: 300px;resize: vertical;}.plus_func_storagebuttons {margin-top: 5px;}.plus_func_storagebuttons>* {margin-right: 10px;padding: 1px 10px;}.plus_func_editlabel {float: right;margin: unset;padding: unset;}.plus_func_editcheckbox {vertical-align: middle;margin-right: 0.3em;}.plus_func_edittext {vertical-align: middle;}'
				},
				Resouces: {
					DefaultIcon: {
						type: 'image',
						value: '',
						crossOrigin: false,
					},
				},
				Config_Ruleset: {
					'version-key': 'config-version',
					'ignores': ["LOCAL-CDN"],
					'defaultValues': {
						//'config-key': {},
						lastupdate: '0000-00-00',
					},
					'updaters': {
						/*'config-key': [
							function() {
								// This function contains updater for config['config-key'] from v0 to v1
							},
							function() {
								// This function contains updater for config['config-key'] from v1 to v2
							}
						]*/
					}
				}
			},
			func: function() {
				const CommonStyle = require('CommonStyle');
				const SPanel = require('SidePanel');
				const SettingPanel = require('SettingPanel');
				const mousetip = require('mousetip');
				const utils = require('utils');
				const CONST = FuncInfo.CONST;

				const CM = new ConfigManager(CONST.Config_Ruleset);
				const CONFIG = CM.Config;

				const UFManager = new UserFunctionManager();
				SPanel.insert({
					index: 1,
					faicon: 'fa-solid fa-gear',
					tip: CONST.Text.Settings,
					onclick: UFManager.show
				});

				return {checkUpdate, userCheckUpdate};

				// Module manager user interface
				function UserFunctionManager() {
					const UMM = this;
					const moduleSettingFuncs = {};
					UMM.showing = false;

					initWindow();

					UMM.show = show;
					UMM.showFuncSettings = showFuncSettings;

					function show() {
						//box.set('message', 'No implemented yet!').show();
						if (UMM.showing) {return false;}

						const functions = FL_listFunctions();

						// Make panel
						const SetPanel = new SettingPanel.SettingPanel({
							header: CONST.Text.Settings,
							tables: [],
							onclose: e => UMM.showing = false
						});
						UMM.showing = true;

						// Title - Userscript Settings
						SetPanel.element.appendChild($$CrE({
							tagName: 'h1',
							props: { innerText: CONST.Text.SettingsOfUserscript }
						}));

						SetPanel.appendTable({
							rows: [{
								blocks: [{
									innerText: CONST.Text.ScriptVersion
								}, {
									innerText: `v${GM_info.script.version}`
								}]
							}, {
								blocks: [{
									innerText: CONST.Text.CheckUpdate
								}, {
									innerText: CONST.Text.ClickMeToCheck,
									style: { cursor: 'pointer' },
									listeners: [['click', e => userCheckUpdate()]]
								}]
							}, {
								blocks: [{
									innerText: CONST.Text.ExportDebugInfo
								}, {
									innerText: CONST.Text.ClickMeToExport,
									style: { cursor: 'pointer' },
									listeners: [['click', e => {
										utils.downloadText(JSON.stringify({
											version: GM_info.script.version,
											GM_info: GM_info,
											platform: navigator.platform,
											userAgent: navigator.userAgent,
											time: new Date().getTime(),
											debug: FL_getDebug()
										}), `${GM_info.script.name}-debug.json`);
									}]]
								}]
							}, {
								blocks: [{
									innerText: CONST.Text.ExportConfig
								}, {
									innerText: CONST.Text.ClickMeToExport,
									style: { cursor: 'pointer' },
									listeners: [['click', e => FL_exportConfig()]]
								}]
							}, {
								blocks: [{
									innerText: CONST.Text.ImportConfig
								}, {
									innerText: CONST.Text.ClickMeToImport,
									style: { cursor: 'pointer' },
									listeners: [['click', e => FL_importConfig(err => {
										if (!err) {
											// err === 0
											alertify.success(CONST.Text.ImportSuccess);
										} else {
											// err > 0
											alertify.error(replaceText(CONST.Text.ImportFailed, { '{ErrorCode}': err }));
										}
									})]]
								}]
							}]
						});

						// Title - Function Manager
						SetPanel.element.appendChild($$CrE({
							tagName: 'h1',
							props: { innerText: CONST.Text.SettingsOfFunctions }
						}));

						// Make table
						const table = new SetPanel.PanelTable({});

						// Make header
						table.appendRow({
							blocks: [{
								isHeader: true,
								colSpan: 1,
								width: '70%',
								style: {'text-align': 'center'},
								innerText: CONST.Text.Function,
							},{
								isHeader: true,
								colSpan: 3,
								width: '30%',
								style: {'text-align': 'center'},
								innerText: CONST.Text.Operation,
							}]
						});

						// Make module rows
						// get all funcs
						const funcs = functions.map(id => FL_getFunction(id));
						// insert system funcs into the ending
						funcs.filter(f => f.system).forEach(f => funcs.push(funcs.splice(funcs.indexOf(f), 1)[0]));
						// insert those funcs with settings into the beginning
						funcs.filter(f => f.setting).forEach(f => funcs.splice(funcs.findIndex(f => !f.setting), 0, ...funcs.splice(funcs.indexOf(f), 1)));
						for (const func of funcs) {
							const id = func.id;
							/*
							const btnEnable = makeBtn({
								text: CONST.Text.EnableFunction,
								onclick: function (e) {
									FL_enableFunction(id);
									enableBtn(btnDisable);
									disableBtn(btnEnable);
								},
								disabled: func.enabled,
								alt: CONST.Text.FunctionEnabled
							});
							const btnDisable = makeBtn({
								text: CONST.Text.DisableFunction,
								onclick: function(e) {
									FL_disableFunction(id);
									enableBtn(btnEnable);
									disableBtn(btnDisable);
								},
								disabled: !func.enabled,
								alt: CONST.Text.FunctionDisabled
							});
							*/
							const btnManage = makeBtn({
								text: CONST.Text.FunctionManagement,
								onclick: function(e) {
									return manageFunc(id);
								}
							});
							const btnSetting = makeBtn({
								text: CONST.Text.FunctionSettings,
								onclick: function(e) {
									return showFuncSettings(id) ? 0 : 1;
								},
								disabled: typeof func.setting !== 'function',
								alt: [CONST.Text.RememberSaving, CONST.Text.NoSetting]
							});
							const row = new SetPanel.PanelRow({
								blocks: [
									// Function info
									{
										colSpan: 1,
										rowSpan: 1,
										children: [
											(() => {
												const icon = $CrE('i');
												icon.className = CONST.Faicon.Info;
												icon.style.marginRight = '0.5em';
												icon.classList.add(CommonStyle.ClassName.Text);

												tippy(icon, {
													content: makeContent(),
													theme: 'wenku_tip',
													onTrigger: (instance, event) => {
														instance.setContent(makeContent());
													}
												});
												return icon;

												function makeContent() {
													const func = FL_getFunction(id);
													const tip = $CrE('div');
													tip.innerHTML = replaceText(CONST.Text.FunctionDetail, {'{CT}': CommonStyle.ClassName.TextLight, '{CB}': CommonStyle.ClassName.Button});
													[...tip.children].forEach((elm) => {
														const name = elm.dataset.key;
														if (name) {
															const value = func.hasOwnProperty(name) ? func[name] : (elm.hasAttribute('use-false-while-empty') ? false : CONST.Text.FunctionDetail_EmptyItem);
															elm.innerText = ({
																string: (s) => (s),
																boolean: (b) => (CONST.Text.Boolean[b.toString()]),
															})[typeof value](value);
														}
													});

													return tip;
												}
											}) (),
											(() => {
												const span = $CrE('span');
												span.innerText = func.system ? CONST.Text.SystemFunc : CONST.Text.NonSystemFunc;
												span.style.marginRight = '0.5em';
												return span;
											}) (),
											(() => {
												const span = $CrE('span');
												span.innerText = func.name;
												return span;
											}) (),
										],
									},

									/*
									// Enable function
									{
										colSpan: 1,
										rowSpan: 1,
										width: '15%',
										style: {'text-align': 'center'},
										children: [btnEnable]
									},

									// Diable function
									{
										colSpan: 1,
										rowSpan: 1,
										width: '15%',
										style: {'text-align': 'center'},
										children: [btnDisable]
									},
									*/

									// Module management
									{
										colSpan: 1,
										rowSpan: 1,
										width: '15%',
										style: {'text-align': 'center'},
										children: [btnManage],
									},

									// Module settings
									{
										colSpan: 1,
										rowSpan: 1,
										width: '15%',
										style: {'text-align': 'center'},
										children: [btnSetting]
									}
								]
							});

							// Only show non-system functions by default
							func.system && row.element.classList.add('plus_system');

							// Append to table
							table.appendRow(row);
						}

						// row to show/hide system funcs
						const sysBtn = {
							icon: (() => {
								const icon = $$CrE({
									tagName: 'i',
									classes: ['fa-solid', 'fa-angle-down', CommonStyle.ClassName.Text],
									styles: {
										'margin-right': '0.5em'
									}
								});
								return icon;
							}) (),
							span: (() => {
								const span = $$CrE({
									tagName: 'span',
									props: {
										innerText: CONST.Text.ShowSystemFuncs
									}
								});
								return span;
							}) ()
						}
						const sysrow = new SetPanel.PanelRow({
							blocks: [
								{
									colSpan: 3,
									rowSpan: 1,
									children: [sysBtn.icon, sysBtn.span]
								}
							]
						});
						$AEL(sysrow.element, 'click', (function() {
							let show = false;
							return function(e) {
								// Change icon
								const [rm, add] = show ? ['fa-angle-up', 'fa-angle-down'] : ['fa-angle-down', 'fa-angle-up'];
								sysBtn.icon.classList.remove(rm);
								sysBtn.icon.classList.add(add);

								// Change text
								sysBtn.span.innerText = show ? CONST.Text.ShowSystemFuncs : CONST.Text.HideSystemFuncs;

								// Show / Hide
								[...$All('.plus_system')].forEach(tr => {
									(show ? tr.classList.remove : tr.classList.add).call(tr.classList, 'plus_system_show');
								});

								// Reverse flag
								show = !show;
							}
						}) ());
						sysrow.element.classList.add(CommonStyle.ClassName.Button);
						table.appendRow(sysrow);

						// Append table
						SetPanel.appendTable(table);

						function makeBtn(details) {
							// Get arguments
							let text, onclick, disabled, alt;
							text = details.text;
							onclick = details.onclick;
							disabled = details.disabled;
							alt = details.alt;

							const span = $CrE('span');
							span.innerText = text;
							onclick && span.addEventListener('click', _onclick);
							span.classList.add(CommonStyle.ClassName.Button);
							disabled && span.classList.add(CommonStyle.ClassName.Disabled);
							span.disabled = disabled;

							return span;

							function _onclick() {
								const disabled = span.disabled;
								const result = !disabled && onclick ? onclick.call(this, arguments) : 0;
								const alt_content = alt && (Array.isArray(alt) ? alt[result] : alt);
								!disabled && ![null, false].includes(result) && alt_content && alertify.message(alt_content);
							}
						}

						function disableBtn(span) {
							span.disabled = true;
							span.classList.add(CommonStyle.ClassName.Disabled);
						}

						function enableBtn(span) {
							span.disabled = false;
							span.classList.remove(CommonStyle.ClassName.Disabled);
						}
					}

					function showFuncSettings(id) {
						const setter = FL_getFunction(id).setting;
						if (typeof setter === 'function') {
							FL_loadSetting(id);
							return true;
						} else {
							return false;
						}
					}

					function manageFunc(idorfunc) {
						// Make panel
						//alertify.alert('not implemented yet!');
						const [id, func] = isWenkuFunction(idorfunc) ? [idorfunc.id, idorfunc] : [idorfunc, FL_getFunction(idorfunc)];
						alertify.funcmanager(func);
					}

					function initWindow() {
						addStyle(CONST.CSS.Common, 'plus-settings-common');
						addStyle(CONST.CSS.Manager, 'plus-settings-manager');
						/*!alertify.installer && alertify.dialog('installer', function() {
							return{
								main: function(){
									let content, oninstall, oncancel;
									switch (arguments.length) {
										case 1: {
											const val = arguments[0];
											if (typeof val === 'object' && !(val instanceof HTMLElement)) {
												content = val.content;
												oninstall = val.oninstall;
												oncancel = val.oncancel;
											} else if (typeof val === 'string' || val instanceof HTMLElement) {
												[content] = arguments;
											} else {
												Err('Invalid argument type', 1);
											}
											break;
										}
										case 2:
											[content, oninstall] = arguments;
											break;
										case 3:
											[content, oninstall, oncancel] = arguments;
											break;
										default:
											Err('Invalid argument length', 1);
									}
									this.set('content', content);
									this.set('oninstall', oninstall);
									this.set('oncancel', oncancel);
								},
								setup: function(){
									return {
										buttons:[{
											text: CONST.Text.Cancel,
											scope: 'auxiliary',
											action: 'cancel', // This is a custom attribute
											key: 27 // Esc
										}, {
											text: CONST.Text.InstallModule,
											scope: 'primary',
											action: 'install', // This is a custom attribute
											key: 65 // Enter
										}],
										focus: {element: 0}
									};
								},
								prepare: function(){
									this.setContent(this.get('content'));
								},
								callback: function(closeEvent) {
									const listenerName = 'on' + closeEvent.button.action;
									const listener = this.get(listenerName);
									return typeof listener === 'function' ? listener(closeEvent) : true;
								},
								settings: {
									content: undefined,
									oninstall: undefined,
									oncancel: undefined
								}
							}
						}, true);*/
						/* 窗口设计
						参照Tampermonky设置界面,每一个大功能设置分类就一个块,从上到下罗列开来
						--------------------
						||图标|  名称.版本  ||
						||图标|    描述    ||
						--------------------
						| 启用 | [ ]        |
						--------------------
						|     |------------|
						| 存  ||    JS    ||
						| 储  ||    ON    ||
						|     |------------|
						--------------------

						*/
						!alertify.funcmanager && alertify.dialog('funcmanager', function() {
							return {
								// Accept every alertify.funcmanager call
								main: function(func) {
									const elements = this.elements;
									this.func = func;

									// Icon
									const icon = func.icon || {type: 'image', value: GM_info.script.icon};
									switch (icon.type) {
										case 'image': {
											const img = $$CrE({
												tagName: 'img',
												props: {
													src: icon.value,
												},
												classes: 'plus_func_icon'
											});
											[...elements.iconarea.children].forEach(n => n.remove())
											elements.iconarea.appendChild(img);
											break;
										}
										case 'faicon': {
											const i = $$CrE({
												tagName: 'i',
												classes: icon.value
											});
											[...elements.iconarea.children].forEach(n => n.remove())
											elements.iconarea.appendChild(i);
										}
									}

									// Name
									elements.name.innerText = func.name;

									// System
									elements.system.innerText = func.system ? CONST.Text.ManageWindow.SystemFunc : CONST.Text.ManageWindow.NonSystemFunc;

									// Version
									elements.version.innerText = `v${func.version || '0.1'}`;

									// Description
									elements.description.innerText = func.description || `功能:${func.name}`;

									// System alert
									elements.sysalert.style.display = func.system ? '' : 'none';

									// Enable
									const enable = elements.enablecheckbox;
									enable.checked = func.enabled;
									enable.disabled = func.system;

									// Storage
									const editor = elements.storageeditor;
									editor.value = vkbeautify.json(func.storage, 2);
									editor.disabled = true;

									// Stoarge buttons
									elements.storagesave.disabled = true;
									elements.storagereload.disabled = false;
									elements.storageexport.disabled = false;
									elements.storageimport.disabled = true;
									elements.storageedit.checked = false;
									elements.storageedit.disabled = func.system;
									elements.storageeditlabel.style.display = func.system ? 'none' : '';
								},

								// Custom buttons
								setup: function() {
									return {
										buttons: [{
											text: '确认',
											key: 27, // esc, see https://developer.mozilla.org/zh-CN/docs/Web/API/KeyboardEvent/keyCode#%E9%94%AE%E7%A0%81%E5%80%BC
											invokeOnClose: true,
											className: alertify.defaults.theme.ok,
											scope: 'primary'
										}],
										focus: {
											element: 0,
											select: true
										}
									}
								},

								// Build standard function manager dom structure
								build: function() {
									const _this = this;
									const elements = this.elements;
									const container = elements.content;

									// Dialog
									elements.dialog.classList.add('plus_func_dialog');

									// Header
									elements.header.innerText = CONST.Text.ManageWindow.Header;

									// Title area
									const titleblock = elements.titleblock = $$CrE({
										tagName: 'div',
										styles: {
											'border-width': '5px 0 0 0',
										}
									});
									titleblock.classList.add('plus_func_titleblock');
									container.appendChild(titleblock);

									// Icon
									const iconarea = elements.iconarea = $CrE('div');
									titleblock.appendChild(iconarea);

									// Name area
									const namearea = elements.namearea = $CrE('div');
									namearea.classList.add('plus_func_namearea');
									titleblock.appendChild(namearea);

									// Name line
									const nameline = $CrE('div');
									nameline.classList.add('plus_func_nameline');
									namearea.appendChild(nameline);

									// Name
									const name = elements.name = $CrE('span');
									name.classList.add('plus_func_name');
									nameline.appendChild(name);

									// Version
									const version = elements.version = $CrE('span');
									version.classList.add('plus_func_version');
									nameline.appendChild(version);

									// Description line
									const descline = $CrE('div');
									descline.classList.add('plus_func_descline');
									namearea.appendChild(descline);

									// System
									const system = elements.system = $CrE('span');
									system.classList.add('plus_func_system');
									descline.appendChild(system);

									// Description
									const description = elements.description = $CrE('span');
									description.classList.add('plus_func_description');
									descline.appendChild(description);

									// System alert
									const sysalert = elements.sysalert = $CrE('div');
									sysalert.innerText = CONST.Text.ManageWindow.SysAlert;
									sysalert.classList.add('plus_func_sysalert');
									container.appendChild(sysalert);

									// Enable block
									const enableblock = elements.enableblock = $CrE('div');
									enableblock.classList.add('plus_func_block');
									container.appendChild(enableblock);

									// Enable title
									const enabletitle = elements.enabletitle = $$CrE({
										tagName: 'div',
										props: {
											innerText: CONST.Text.ManageWindow.EnableTitle
										},
										classes: 'plus_func_blocktitle'
									});
									enableblock.appendChild(enabletitle);

									// Enable centent
									const enablecontent = elements.enablecontent = $$CrE({
										tagName: 'div',
										classes: 'plus_func_blockcontent'
									});
									enableblock.appendChild(enablecontent);

									// Enable checkbox
									const enablecheckbox = elements.enablecheckbox = $$CrE({
										tagName: 'input',
										props: {
											type: 'checkbox',
										},
										classes: 'plus_func_enablecheckbox',
										listeners: [['change', function(e) {
											_this.func.enabled = enablecheckbox.checked;
											alertify.success(replaceText(enablecheckbox.checked ? CONST.Text.FunctionEnabled : CONST.Text.FunctionDisabled, {'{FuncName}': _this.func.name}));
										}]]
									});
									enablecontent.appendChild(enablecheckbox);

									// Storage block
									const storageblock = elements.storageblock = $CrE('div');
									storageblock.classList.add('plus_func_block');
									container.appendChild(storageblock);

									// Storage title
									const storagetitle = elements.storagetitle = $$CrE({
										tagName: 'div',
										props: {
											innerText: CONST.Text.ManageWindow.StorageTitle
										},
										classes: 'plus_func_blocktitle'
									});
									storageblock.appendChild(storagetitle);

									// Storage content
									const storagecontent = elements.storagecontent = $$CrE({
										tagName: 'div',
										classes: 'plus_func_blockcontent',
										/*listeners: [['click', function(e) {
											// storageeditor cannot listen mouse click events while it's disabled
											if (e.target === storageeditor && storageeditor.disabled && !_this.func.system) {
												alertify.confirm(CONST.Text.ManageWindow.Header, CONST.Text.ManageWindow.StorageAlert, function() {
													storageeditor.disabled = false;
													//storageeditor.focus() // Do not focus directly, no focus will gain!
													setTimeout(e => storageeditor.focus(), 0); // focus with a setTimeout is okay
													storageedit.checked = true;
												}, e => {});
											}
										}]]*/
									});
									storageblock.appendChild(storagecontent);

									// Storage editor
									const storageeditor = elements.storageeditor = $$CrE({
										tagName: 'textarea',
										prop: {
											disabled: true
										},
										classes: 'plus_func_storageeditor'
									});
									storagecontent.appendChild(storageeditor);

									// Storage buttons
									const storagebuttons = elements.storagebuttons = $$CrE({
										tagName: 'div',
										classes: 'plus_func_storagebuttons'
									});
									storagecontent.appendChild(storagebuttons);

									// Storage save button
									const storagesave = elements.storagesave = $$CrE({
										tagName: 'button',
										props: {
											innerText: CONST.Text.ManageWindow.Save
										},
										classes: 'plus_func_storagesave',
										listeners: [['click', function(e) {
											try {
												const storage = JSON.parse(storageeditor.value);
												Object.keys(_this.func.storage).forEach(k => delete _this.func.storage[k])
												copyProps(storage, _this.func.storage);
												alertify.success(CONST.Text.ManageWindow.StorageSaved);
											} catch(err) {
												alertify.error(CONST.Text.ManageWindow.StorageFormatError);
											}
										}]]
									});
									storagebuttons.appendChild(storagesave);

									// Storage reload button
									const storagereload = elements.storagereload = $$CrE({
										tagName: 'button',
										props: {
											innerText: CONST.Text.ManageWindow.Reload
										},
										classes: 'plus_func_storagereload',
										listeners: [['click', function(e) {
											storageeditor.value = vkbeautify.json(_this.func.storage, 2);
										}]]
									});
									storagebuttons.appendChild(storagereload);

									// Storage export button
									const storageexport = elements.storageexport = $$CrE({
										tagName: 'button',
										props: {
											innerText: CONST.Text.ManageWindow.Export
										},
										classes: 'plus_func_storagereload',
										listeners: [['click', function(e) {
											utils.downloadText(JSON.stringify(_this.func.storage), `${GM_info.script.name}-${_this.func.id}-config.json`);
											storageeditor.value = vkbeautify.json(_this.func.storage, 2);
										}]]
									});
									storagebuttons.appendChild(storageexport);

									// Storage import button
									const storageimport = elements.storageimport = $$CrE({
										tagName: 'button',
										props: {
											innerText: CONST.Text.ManageWindow.Import
										},
										classes: 'plus_func_storagereload',
										listeners: [['click', function(e) {
											if (!storageeditor.disabled) {
												const input = $$CrE({
													tagName: 'input',
													props: { type: 'file' },
													listeners: [['change', e => {
														try {
															const file = input.files[0];
															const reader = new FileReader();
															reader.onload = e => {
																_this.func.storage = JSON.parse(reader.result);
																storageeditor.value = vkbeautify.json(_this.func.storage, 2);
															}
															reader.onerror = err => { throw err; };
															reader.readAsText(file);
														} catch(err) {
															DoLog(LogLevel.Warning, 'Import error');
															DoLog(LogLevel.Warning, err, 'warn');
														}
													}]]
												});
												input.click();
											}
											/*
											utils.downloadText(JSON.stringify(_this.func.storage), `${GM_info.script.name}-${_this.func.id}-config.json`);
											storageeditor.value = vkbeautify.json(_this.func.storage, 2);
											*/
										}]]
									});
									storagebuttons.appendChild(storageimport);

									const storagereset = elements.storagereset = $$CrE({
										tagName: 'button',
										props: {
											innerText: CONST.Text.ManageWindow.Reset
										},
										classes: 'plus_func_storagereset',
										listeners: [['click', function(e) {
											alertify.confirm(CONST.Text.ManageWindow.Header, CONST.Text.ManageWindow.StorageResetConfirm, function() {
												_this.func.storage = {};
												storageeditor.value = vkbeautify.json(_this.func.storage, 2);
											}, e => {});
										}]]
									});
									storagebuttons.appendChild(storagereset);

									// Storage edit checkbox
									const storageedit = elements.storageedit = $$CrE({
										tagName: 'input',
										props: {
											type: 'checkbox'
										},
										classes: 'plus_func_editcheckbox',
										listeners: [['change', (function() {
											let firstChecked = false;
											return function(e) {
												if (_this.func.system) {
													storageedit.checked = false;
													return;
												}
												if (storageedit.checked) {
													if (!firstChecked) {
														alertify.confirm(CONST.Text.ManageWindow.Header, CONST.Text.ManageWindow.StorageAlert, function() {
															firstChecked = true;
															enable();
														}, e => { storageedit.checked = false; });
													} else {
														enable();
													}
												} else {
													disable();
												}

												function disable() {
													[storagesave, storageimport, storageeditor].forEach(btn => btn.disabled = true);
												}

												function enable() {
													[storagesave, storageimport, storageeditor].forEach(btn => btn.disabled = false);
													//storageeditor.focus() // Do not focus directly, no focus will gain!
													setTimeout(e => storageeditor.focus(), 0); // focus with a setTimeout is okay
												}
											};
										})()]]
									});
									const storageedittext = elements.storageedittext = $$CrE({
										tagName: 'span',
										classes: 'plus_func_edittext',
										props: { innerText: CONST.Text.ManageWindow.StorageEdit }
									});
									const storageeditlabel = elements.storageeditlabel = $$CrE({
										tagName: 'label',
										classes: 'plus_func_editlabel',
									});
									storageeditlabel.appendChild(storageedit);
									storageeditlabel.appendChild(storageedittext);
									storagebuttons.appendChild(storageeditlabel);
								}
							}
						});
					}
				}

				// Check userscript update and returns a Promise({info, hasUpdate})
				async function checkUpdate(autoInstall=true) {
					const GF = new GreasyFork();

					alertify.message(CONST.Text.CheckingUpdate);
					const info = await GF.get().script().info(CONST.Number.GFScriptID);
					CONFIG.lastupdate = utils.getTime('-', false);

					if (versionNewer(info.version, GM_info.script.version)) {
						autoInstall && info.action('install', { id: CONST.Number.GFScriptID });
						return {info, hasUpdate: true};
					} else {
						return {info, hasUpdate: false};
					}

					function versionNewer(v1, v2) {
						if ([v1, v2].some(v => !/(\d\.?)+/.test(v))) {
							Err('versionNewer: all version strings should only contain number and dot(.), and have no consecutive dots.');
						}

						[v1, v2] = [v1, v2].map(v => v.split('.').map(s => parseInt(s, 10)));
						for (let i = 0; i < Math.min(v1.length, v2.length); i++) {
							if (v1[i] > v2[i]) {
								return true;
							}
						}
						if (v1.length > v2.length) {
							return true;
						}
						return false;
					}
				}

				// Check userscript update and provide update GUI for user if update exists
				// Returns a Promise(statusNumber), statusNumber = -1: User refuse to update; 0: No update found; 1: User clicked 'install' button in GUI
				async function userCheckUpdate() {
					checkUpdate(false).then(update => new Promise((resolve, reject) => {
						const GF = new GreasyFork();
						if (!update.hasUpdate) {
							resolve(0);
						}

						const message = $$CrE({ tagName: 'div', props: { innerText: CONST.Text.UpdateFound.Main } });
						const btnView = $$CrE({
							tagName: 'span',
							props: { innerText: CONST.Text.UpdateFound.View },
							classes: [CommonStyle.ClassName.Button],
							listeners: [['click', view]]
						});
						const btnInstall = $$CrE({
							tagName: 'span',
							props: { innerText: CONST.Text.UpdateFound.Install },
							classes: [CommonStyle.ClassName.Button],
							listeners: [['click', install]]
						});
						[$CrE('br'), btnView, $CrE('br'), btnInstall].forEach(elm => message.appendChild(elm));
						alertify.message(message);

						function view() {
							console.log('autoupdate: view update');

							const dp = new DOMParser();
							const doc = dp.parseFromString(update.info.additionalInfo[0].html, 'text/html');
							const updateInfo = $(doc, 'div[name="update-info"]');
							const box = alertify.confirm(updateInfo ? updateInfo : doc.body, install, e => resolve(-1));
							box.setHeader(replaceText(CONST.Text.UpdateInfo.Header, {'{version}': update.info.version}));
							box.set('labels', {ok: CONST.Text.UpdateInfo.Install, cancel: CONST.Text.UpdateInfo.Close});
							box.set('overflow', true);
						}

						function install() {
							GF.action('install', { id: CONST.Number.GFScriptID });
							resolve(1);
						}
					}));
				}
			},
			alwaysRun: function() {
				const settings = require('settings');
				const CommonStyle = require('CommonStyle');
				const utils = require('utils');
				const GF = new GreasyFork();
				const CONST = FuncInfo.CONST;

				const CM = new ConfigManager(CONST.Config_Ruleset);
				const CONFIG = CM.Config;

				!todayUpdated() && settings.userCheckUpdate();

				function todayUpdated() {
					return utils.getTime('-', false) === CONFIG.lastupdate;
				}
			}
		},

		// WenkuBlockGUI
		{
			name: 'WenkuBlockGUI',
			description: '基本组件 - 提供文库风格的界面元素支持',
			id: 'WenkuBlockGUI',
			system: true,
			checker: {
				type: 'switch',
				value: false
			},
			func: function() {
				const mousetip = require('mousetip');
				const CONST = {
					Text: {
						DefaultTitle: '操作区域'
					},
					CSS: {
						ClassName: {
							List: 'plus_list',
							ListButton: 'plus_list_input',
							ListItem: 'plus_list_item'
						},
						Style: {
							PlusList: '.plus_list>ul {list-style: none; text-align: center; padding: 0px; margin: 0px;} .plus_list {position: absolute; z-index: 999; background-color: #f5f5f5; float: left; clear: both; overflow-y: auto; overflow-x: visible;} .plus_list_input {display: block; list-style: outside none none; margin: 0px; border: 0;} .plus_list_item {border: 0px; cursor: pointer; padding: 0 0.5em; border: 1px solid rgb(204, 204, 204); background-color: buttonface;}',
						}
					}
				};

				Object.keys(CONST.CSS.Style).forEach(key => addStyle(CONST.CSS.Style[key], key));

				return {
					makeStandardBlock,
					makeBookcaseBlock,
					makeIndexCenterToplistBlock,
					makeIndexRightToplistBlock,
					PlusList
				};

				// Create a standard block structure
				function makeStandardBlock() {
					const block = $$CrE({tagName: 'div', classes: ['block']});
					const blocktitle = $$CrE({tagName: 'div', classes: ['blocktitle']});
					const blockcontent = $$CrE({tagName: 'div', classes: ['blockcontent']});
					block.appendChild(blocktitle);
					block.appendChild(blockcontent);
					return {block, blocktitle, blockcontent};
				}

				// Bookcase-left block
				function makeBookcaseBlock(detail) {
					const block = makeStandardBlock();

					// Title
					block.blocktitle.appendChild($$CrE({
						tagName: 'span',
						props: {'innerText': detail.title || ''},
						classes: ['txt']
					}));
					block.blocktitle.appendChild($$CrE({
						tagName: 'span',
						classes: ['txtr']
					}));

					// Content links
					const ul = $$CrE({
						tagName: 'ul',
						classes: ['ulitem']
					});
					for (const objLink of detail.links || []) {
						const li = $CrE('li');
						const a = $$CrE({
							tagName: 'a',
							props: {
								'href': objLink.href || 'javascript: void(0);',
								'innerText': objLink.text || '',
								...objLink.props || {}
							},
							styles:  objLink.styles || {},
							classes: objLink.classes || [],
							listeners: objLink.listeners || []
						});
						objLink.hasOwnProperty('key') && (block[objLink.key] = a);
						li.appendChild(a);
						ul.appendChild(li);
					}
					block.blockcontent.appendChild(ul);

					return block;
				}

				// Index page center books toplist block
				// '本周会员推荐榜'
				function makeIndexCenterToplistBlock(detail) {
					const block = makeStandardBlock();

					// Title
					block.blocktitle.innerText = detail.title || '';

					// Content books
					const container = $$CrE({tagName: 'div', styles: {'height': '155px'}});
					block.blockcontent.appendChild(container);
					block.items = [];
					for (const book of detail.books) {
						const b_container = $$CrE({
							tagName: 'div',
							styles: {
								cssText: 'float: left; text-align: center; width: 95px; height: 155px; overflow: hidden;'
							}
						});
						container.appendChild(b_container);

						const imgLink = $$CrE({
							tagName: 'a',
							props: {
								href: book.url,
								target: '_blank'
							}
						});
						mousetip.settip(imgLink, book.name);
						b_container.appendChild(imgLink);

						const img = $$CrE({
							tagName: 'img',
							props: {src: book.cover},
							attrs: {
								border: 0,
								width: 90,
								height: 127
							}
						});
						imgLink.appendChild(img);

						b_container.appendChild($CrE('br'));

						const nameLink = $$CrE({
							tagName: 'a',
							props: {
								innerText: book.name,
								href: book.url,
								target: '_blank'
							}
						});
						b_container.appendChild(nameLink);

						block.items.push({ ...book, b_container, imgLink, img, nameLink });
					}

					return block;
				}

				// '最受关注'
				function makeIndexRightToplistBlock(detail) {
					const block = makeStandardBlock();

					// Title
					block.blocktitle.appendChild($$CrE({
						tagName: 'span',
						props: {
							innerText: detail.title || '',
						},
						classes: ['txt']
					}));
					block.blocktitle.appendChild($$CrE({
						tagName: 'span',
						classes: ['txtr']
					}));

					// Content
					const ul = $$CrE({
						tagName: 'ul',
						classes: ['ultop']
					});
					block.blockcontent.appendChild(ul);

					block.items = [];
					for (const link of detail.links) {
						const li = $$CrE({
							tagName: 'li',
							styles: {
								overflow: 'hidden'
							}
						});
						const a = $$CrE({
							tagName: 'a',
							props: {
								innerText: link.text,
								href: link.url,
								target: '_blank',
							}
						});
						mousetip.settip(a, link.text);
						li.appendChild(a);
						ul.appendChild(li);
						block.items.push({ ...link, li, a });
					}

					return block;
				}

				// Create a list gui like reviewshow.php##FontSizeTable
				// list = {display: '', id: '', parentElement: <*>, insertBefore: <*>, list: [{value: '', onclick: Function, tip: ''/Function}, ...], visible: bool, onshow: Function(bool shown), onhide: Function(bool hidden)}
				// structure: {div: <div>, ul: <ul>, list: [{li: <li>, button: <input>}, ...], visible: list.visible, show: Function, hide: Function, append: Function({...}), remove: Function(index), clear: Function, onshow: list.onshow, onhide: list.onhide}
				// Use 'new' keyword
				function PlusList(list) {
					const PL = this;

					// Make list
					const div = PL.div = document.createElement('div');
					const ul = PL.ul = document.createElement('ul');
					div.classList.add(CONST.CSS.ClassName.List);
					div.appendChild(ul);
					list.display && (div.style.display = list.display);
					list.id && (div.id = list.id);
					list.parentElement && list.parentElement.insertBefore(div, list.insertBefore ? list.insertBefore : null);

					PL.list = [];
					for (const item of list.list) {
						appendItem(item);
					}

					// Attach properties
					let onshow = list.onshow ? list.onshow : function() {};
					let onhide = list.onhide ? list.onhide : function() {};
					let visible = list.visible;
					let maxheight;
					PL.create = createItem;
					PL.append = appendItem;
					PL.insert = insertItem;
					PL.remove = removeItem;
					PL.clear = removeAll;
					PL.show = showList;
					PL.hide = hideList;
					Object.defineProperty(PL, 'onshow', {
						get: function() {return onshow;},
						set: function(func) {
							onshow = func ? func : function() {};
						},
						configurable: false,
						enumerable: true
					});
					Object.defineProperty(PL, 'onhide', {
						get: function() {return onhide;},
						set: function(func) {
							onhide = func ? func : function() {};
						},
						configurable: false,
						enumerable: true
					});
					Object.defineProperty(PL, 'visible', {
						get: function() {return visible;},
						set: function(bool) {
							if (typeof(bool) !== 'boolean') {return false;};
							visible = bool;
							bool ? showList() : hideList();
						},
						configurable: false,
						enumerable: true
					});
					Object.defineProperty(PL, 'maxheight', {
						get: function() {return maxheight;},
						set: function(num) {
							if (typeof(num) !== 'number') {return false;};
							maxheight = num;
						},
						configurable: false,
						enumerable: true
					});

					// Apply configurations
					div.style.display = list.visible === true ? '' : 'none';

					// Functions
					function appendItem(item) {
						const listitem = createItem(item);
						ul.appendChild(listitem.li);
						PL.list.push(listitem);
						return listitem;
					}

					function insertItem(item, index, insertByNode=false) {
						const listitem = createItem(item);
						const children = insertByNode ? ul.childNodes : ul.children;
						const elmafter = children[index];
						ul.insertBefore(item.li, elmafter);
						inserttoarr(PL.list, listitem, index);
					}

					function createItem(item) {
						const listitem = {
							remove: () => {removeItem(listitem);},
							li: document.createElement('li'),
							button: document.createElement('input')
						};
						const li  = listitem.li;
						const btn = listitem.button;
						btn.type = 'button';
						btn.classList.add(CONST.CSS.ClassName.ListButton);
						li.classList.add(CONST.CSS.ClassName.ListItem);
						item.value && (btn.value = item.value);
						item.onclick && btn.addEventListener('click', item.onclick);
						item.tip && mousetip.settip(li, item.tip);
						item.tip && mousetip.settip(btn, item.tip);
						li.appendChild(btn);
						return listitem;
					}

					function removeItem(itemorindex) {
						// Get index
						let index;
						if (typeof(itemorindex) === 'number') {
							index = itemorindex;
						} else if (typeof(itemorindex) === 'object') {
							index = PL.list.indexOf(itemorindex);
						} else {
							return false;
						}
						if (index < 0 || index >= PL.list.length) {
							return false;
						}

						// Remove
						const li = PL.list[index];
						ul.removeChild(li.li);
						delfromarr(PL.list, index);
						return li;
					}

					function removeAll() {
						const length = PL.list.length;
						for (let i = 0; i < length; i++) {
							removeItem(0);
						}
					}

					function showList() {
						if (visible) {return false;};
						onshow(false);
						div.style.display = '';
						onshow(true);
						visible = true;
					}

					function hideList() {
						if (!visible) {return false;};
						onhide(false);
						div.style.display = 'none';
						mousetip.hidetip();
						onhide(true);
						visible = false;
					}

					// Support functions
					// Del an item from an array by provided index, returns the deleted item. MODIFIES the original array directly!!
					function delfromarr(arr, delIndex) {
						if (delIndex < 0 || delIndex > arr.length-1) {
							return false;
						}
						const deleted = arr[delIndex];
						for (let i = delIndex; i < arr.length-1; i++) {
							arr[i] = arr[i+1];
						}
						arr.pop();
						return deleted;
					}

					// Insert an item to an array by its provided index, returns the item itself. MODIFIES the original array directly!!
					function inserttoarr(arr, item, index) {
						if (index < 0 || index > arr.length-1) {
							return false;
						}
						for (let i = arr.length; i > index; i--) {
							arr[i] = arr[i-1];
						}
						arr[index] = item;
						return item;
					}
				}
			}
		},

		// Imager
		{
			name: '图床',
			description: '提供图床上传图片接口,方便用户上传图片',
			id: 'imager',
			system: true,
			checker: {
				type: 'switch',
				value: true
			},
			STOP: false,
			CONST: {
				Text: {
					/*ImageOnly: '您提供的文件无法被识别为图片',
					ImageUploadError: '图片上传失败',
					ClicktoPaste: '点击需要粘贴图片的位置,或者点击其他位置/esc取消',*/
					StatusText: {
						empty: '拖放图片到这里或者单击选择图片',
						waiting: '等待上传',
						uploading: '正在上传',
						uploaded: '上传成功',
						error: '上传失败'
					},
					ScriptTitle: GM_info.script.name,
					OneImageOnly: '每次请选择仅一张图片',
					ImageOnly: '您提供的文件并没有可识别的图片格式</br>请尝试jpg、jpeg、png等常用格式的图片',
					ConnectRequired_Title: '需要网络权限',
					ConnectRequired: `${GM_info.script.name}在为您上传图片时出现了错误</br>原因:缺乏网络权限</br></br>您需要允许对图床网站的网络访问权限,${GM_info.script.name}才能够使用此图片或者访问图床</br></br>您可以在弹出的Tampermonkey页面中点击“允许一次”或者“临时允许”来临时允许${GM_info.script.name}访问此图片或者访问图床,或者点击“总是允许全部域名”永久允许${GM_info.script.name}访问您提供的图片或图床以避免以后再次弹出确认`,
					UploadError: '图片上传失败',
					CoverText: '拖放上传图片',
					UploaderHoverText: '拖放上传图片 或 单击选择图片',
					ClickToClear: '清除已上传图片'
				},
				Imagers: [{
					id: 'p.sda1.dev',
					upload: {
						request: {
							url: 'https://p.sda1.dev/api_dup$backendid$/v1/upload_noform?ts=$time$&rand=$random$&filename=$filename$&batch_size=1',
							replacers: {
								'$backendid$': () => parseInt(Math.random() * 1000000007) % 10
							},
							data: {
								type: 'file'
							},
							responseType: 'json',
							onerror: (err, callback) => GM_xmlhttpRequest({
								method: 'GET',
								url: 'https://p.sda1.dev/assets/css/thumb.css',
								timeout: 5 * 1000,
								onload: err => callback(true),
								onerror: err => callback(false),
								ontimeout: err => callback(false),
							})
						},
						response: {
							type: 'func',
							func: response => new URL(response.response.data.url).href,
							valid: response => typeof response.response === 'object' && response.response !== null && response.response.data && response.response.data.url
						}
					},
					features: ['custom-filename', 'custom-extention']
				}, {
					id: 'sougou',
					upload: {
						request: {
							url: 'https://image.sogou.com/pic/upload_pic.jsp',
							headers: {
								//'Content-Type': 'multipart/form-data' // NO this header manually! this will overwrite Content-Type with Boundary value missing
							},
							data: {
								type: 'form',
								value: 'pic_path'
							}
						},
						response: {
							type: 'url',
							valid: response => typeof response.response === 'string' && /https?:\/\//.test(response.response),
						}
					},
				}],
				Config_Ruleset: {
					'version-key': 'config-version',
					'ignores': ["LOCAL-CDN"],
					'defaultValues': {
						//'config-key': {},
						stat: {}
					},
					'updaters': {
						/*'config-key': [
							function() {
								// This function contains updater for config['config-key'] from v0 to v1
							},
							function() {
								// This function contains updater for config['config-key'] from v1 to v2
							}
						]*/
					}
				}
			},
			func: function() {
				const CommonStyle = require('CommonStyle');
				const SettingPanel = require('SettingPanel');
				const mousetip = require('mousetip');

				const CONST = FuncInfo.CONST;
				const CM = new ConfigManager(CONST.Config_Ruleset);
				const CONFIG = CM.Config;
				const UPLOADER = new Uploader();

				// <image-selector> custom element
				registerElement();

				SettingPanel.registerElement('image', {
					createElement: function() {
						const uploader = $CrE('image-uploader');
						$AEL(uploader, 'error', err => {
							alertify.error(CONST.Text.UploadError);
						});
						return uploader;
					},
					setValue: function(val) {
						// when val accepted, it means we're reading an uploaded url from storage
						// So val is url
						this.element.url = val;
					},
					getValue: function() {
						return this.element.url;
					},
				});

				return {upload};

				function upload() {
					UPLOADER.upload.apply(this, arguments);
				}

				// Image uploader
				/*
					UPLOADER.upload(file, onload, onerror=e=>{}, retry=3)
						onload => onload(url)
						onerror => onerror()
					UPLOADER.uploadByImager(file, imager, onload, onerror)
						onload => onload(url)
						onerror => onerror()
				*/
				function Uploader() {
					const UP = this;
					UP.upload = upload;
					UP.uploadByImager = uploadByImager;
					init();

					function upload(file, onload, onerror=e=>{}, features=[]) {
						const imager_ids = getSortedImagers(features);
						let index = 0, retry = 3, imagerRetried = imager_ids.map(i => false);
						uploadByImager(file, imager_ids[0], onload, autoRetry);

						function autoRetry(err) {
							const imager = CONST.Imagers.find(i => i.id === imager_ids[index]);
							if (isConnectErr(err)) {
								// @connect not allowed
								onerror();
								return false;
							} else if (imager.upload.request.onerror && !imagerRetried[index]) {
								let timeouted = false, retrying = false;
								imagerRetried[index] = true;

								// Imager has an error handler, handle the error
								// and give it a chance to retry before change to another imager
								imager.upload.request.onerror(err, success => {
									if (success && !timeouted) {
										retrying = true;
										uploadByImager(file, imager_ids[index], onload, autoRetry)
									} else {
										autoRetry();
									}
								});

								// We give imager's error handler 5 seconds to finish its task
								// If it doesn't finish its task in time, just go ahead
								setTimeout(e => {
									if (!retrying) {
										timeouted = true;
										autoRetry();
									}
								}, 5* 1000);
							} else if (--retry > 0 && ++index < imager_ids.length) {
								uploadByImager(file, imager_ids[index], onload, autoRetry);
							} else {
								onerror();
							}
						}
					}

					function uploadByImager(file, imager_id, onload, onerror) {
						const imager = CONST.Imagers.find(i => i.id === imager_id);
						const req = imager.upload.request;
						const res = imager.upload.response;

						const headers = req.headers || {};
						const responseType = req.responseType || 'text';
						const data = ({
							'file': () => file,
							'form': () => {
								const formdata = new FormData();
								formdata.set(req.data.value, file);
								return formdata;
							},
							checker: url => !!url
						})[req.data.type]();

						// Construct request url
						const url = makeUrl(req.url, req.replacers);

						// Request
						GM_xmlhttpRequest({
							method: 'POST',
							url: url,
							headers: {...headers}, // headers should be object, use {...headers} insteadof headers to validate
							responseType, data,
							onerror: handleError,
							ontimeout: handleError,
							onload: dealResponse
						});

						// Response
						function dealResponse(response) {
							if (!res.valid(response)) {
								handleError(response);
								return false;
							}

							let url = ({
								'url': response => response.responseText,
								'func': response => res.func(response)
							})[res.type](response);
							url = new URL(url).href;

							record(imager_id, true);

							onload(url);
						}

						function handleError(err) {
							isConnectErr(err) ?
								alertify.alert(CONST.Text.ConnectRequired_Title, CONST.Text.ConnectRequired) :
								record(imager_id, false);

							onerror.apply(this, arguments);
						}

						function makeUrl(url, _replacers={}) {
							const replacers = {
								'$filename$': () => (encodeURIComponent(file.name)),
								'$random$': () => (Math.random().toString()),
								'$time$': () => ((new Date()).getTime().toString()),
								..._replacers
							};
							const replaceObj = {};
							for (let [mark, func] of Object.entries(replacers)) {
								replaceObj[mark] = func();
							}
							return replaceText(url, replaceObj);;
						}
					}

					function getSortedImagers(features=[]) {
						const stat = CM.getConfig('stat');
						const total = stat.total;
						delete stat.total;

						// filter requested features
						CONST.Imagers.forEach(imager => {
							features.some(f => !imager.features || !imager.features.includes(f)) && delete stat[imager.id];
						});

						return Object.values(stat).map(imager => ({
							id: imager.id,
							score: score(imager)
						})).sort((i1, i2) => (i2.score - i1.score)).map(imager => imager.id);

						function score(imager) {
							// Whether last three uploads successed takes 0.4 point
							// Last three upload takes 0.25, 0.1, 0.05 points in order
							const lastScore = [0.25, 0.1, 0.05].reduce((score, n, index) => {
								return score + n * (imager.last[index] ? 1 : 0);
							}, 0);

							// Whether imager stat record is enough takes 0.2 point
							const numScore = (imager.all / total.all) >= (1 / CONST.Imagers.length) ? 0.2 : 0;

							// Success rate takes 0.4 point
							const sucScore = imager.all ? (imager.success / imager.all) * 0.4 : 0;

							// Sum up all three scores
							return lastScore + numScore + sucScore;
						}
					}

					function record(imager_id, success) {
						if (typeof imager_id !== 'string') {
							Err(`Uploader.record: imager_id should be a string, not ${typeof success}`, 1);
						}
						if (!CONST.Imagers.some(i => i.id === imager_id)) {
							Err(`Uploader.record: imager_id(${imager_id}) not found`, 1);
						}
						if (typeof success !== 'boolean') {
							Err(`Uploader.record: success should be a boolean, not ${typeof success}`, 1);
						}

						[imager_id, 'total'].forEach(key => {
							CONFIG.stat[key].all++;
							CONFIG.stat[key][success ? 'success' : 'fail']++;
						});

						CONFIG.stat[imager_id].last.unshift(success);
						CONFIG.stat[imager_id].last.pop();
					}

					function isConnectErr(err) {
						return typeof err === 'object' && err !== null && typeof err.error === 'string' && err.error.startsWith('Refused to connect to ');
					}

					function init() {
						for (const imager of CONST.Imagers) {
							if (!CONFIG.stat[imager.id]) {
								CONFIG.stat[imager.id] = {
									id: imager.id, // Imager id
									last: [false, false, false],   // Whether last three upload successed
									success: 0,    // Total success count
									fail: 0,       // Total fail count
									all: 0,        // Total uploads count
								};
							}
							if (!CONFIG.stat.total) {
								CONFIG.stat.total = {
									success: 0,    // Total success count for all imagers
									fail: 0,       // Total fail count for all imagers
									all: 0,        // Total uploads count for all imagers
								};
							}
						}
					}
				}

				// New solution
				function registerElement() {
					// ImageUploader custom element
					/* ImageUploader design:
					<image-uploader>
						shadowroot (closed):
							<div id='wrapper'>
								<div id='image'>
									<span id='closebtn'></span>
									<img id='img' src>
									<br id='br'>
									<span id='placeholder' class='empty local uploaded error'>状态提示文本(无图片,未上传,上传成功,上传失败)</span>
								</div>
								<div id='cover'>拖入时提示文本,覆盖在整个element最上层显示</div>
							</div>
							<style>CSS Here</style>
					</image-uploader>

					Methods:
					[x] ImageUploader.prototype.upload:    Upload user-selected image
					[?] ImageUploader.prototype.clear:     Clear user-selected image
					[x] ImageUploader.prototype.load(url): download image blob from url and <set> to ImageUploader.prototype.image

					Properties:
					[x] ImageUploader.prototype.image: <get/set> image file/blob
					[x] ImageUploader.prototype.url: <get/set> uploaded image url
						[x] [onset]: set url property and img.src directly (act as this url is an uploaded image url)
						[x] [onupload]: set uploaded url to ImageUploader.prototype.url
					[x] ImageUploader.prototype.status: <get> upload status, 'uploaded'/'uploading'/'error'/'waiting'/'empty'
					[x] ImageUploader.prototype.autoUpload: <get/set> whether upload user selected images automatically

					Events:
					[?] error: triggers on upload error
					[?] load: triggers on upload finish

					Behaviors:
					[x] handle drag & drop
						[x] dragover: show cover
						[x] dragleave: hide cover
						[x] drop: auto set image and upload if ImageUploader.prototype.autoUpload === true

					Apperence:
					[x] CSS
					*/
					class ImageUploader extends HTMLElement {
						#elements = {};
						#image = {};
						#url = '';
						#status = 'empty';
						#autoUpload = true;
						#UPLOADER = new Uploader();

						constructor() {
							// Always call super() first
							super();

							// Const
							const CSS_ImageUploader = `#wrapper{cursor: pointer; position: relative;} #closebtn{position: absolute; right: 0; top: 0; width: 1em; height: 1em; font-size: 2em; text-align: center; border: 1px dashed rgba(0,0,0,0);} #closebtn:hover,#closebtn:focus{border: 1px dashed rgba(0,0,0,0.3);} #img{width: 100%; height: 100%;} #placeholder{text-align: center;} #cover{position: absolute; left: 0; top: 0; width: calc(100% - 20px); height: calc(100% - 20px); z-index: 1; background-color: rgba(255, 255, 255, 0.8); border: 10px grey dashed; pointer-events: none;} #covertext{position: absolute; top: calc(50% + 10px); left: 50%; margin-top: -0.5em; margin-left: -3em; font-size: 30px;}`;

							// Save this
							const _this = this;

							// Element references
							const elements = this.#elements;

							// Shadowroot
							elements.shadow = this.attachShadow({mode: 'closed'});

							// Wrapper
							elements.wrapper = $CrE('div');
							elements.wrapper.id = 'wrapper';
							elements.shadow.appendChild(elements.wrapper);
							mousetip.settip(elements.wrapper, CONST.Text.UploaderHoverText);

							// Img container
							elements.imgContainer = $CrE('div');
							elements.imgContainer.id = 'imgContainer';
							elements.wrapper.appendChild(elements.imgContainer);

							// Close button
							elements.closebtn = $CrE('span');
							elements.closebtn.id = 'closebtn';
							elements.closebtn.classList.add(CommonStyle.ClassName.Button);
							elements.closebtn.innerHTML = '<svg xmlns="http://www.w3.org/2000/svg" height="1em" viewBox="0 0 384 512"><!--! Font Awesome Free 6.4.0 by @fontawesome - https://fontawesome.com License - https://fontawesome.com/license (Commercial License) Copyright 2023 Fonticons, Inc. --><path d="M342.6 150.6c12.5-12.5 12.5-32.8 0-45.3s-32.8-12.5-45.3 0L192 210.7 86.6 105.4c-12.5-12.5-32.8-12.5-45.3 0s-12.5 32.8 0 45.3L146.7 256 41.4 361.4c-12.5 12.5-12.5 32.8 0 45.3s32.8 12.5 45.3 0L192 301.3 297.4 406.6c12.5 12.5 32.8 12.5 45.3 0s12.5-32.8 0-45.3L237.3 256 342.6 150.6z"/></svg>';
							elements.imgContainer.appendChild(elements.closebtn);
							$AEL(elements.closebtn, 'click', e => {
								e.stopPropagation();
								_this.clear();
							});
							mousetip.settip(elements.closebtn, CONST.Text.ClickToClear);

							// Img
							elements.img = $CrE('img');
							elements.img.id = 'img';
							elements.imgContainer.appendChild(elements.img);
							$AEL(elements.img, 'load', e => elements.imgContainer.style.removeProperty('display'));
							$AEL(elements.img, 'error', e => elements.imgContainer.style.display = 'none');

							// br
							elements.br = $CrE('br');
							elements.br.id = 'br';
							elements.imgContainer.appendChild(elements.br);

							// Placeholder
							elements.placeholder = $CrE('div');
							elements.placeholder.id = 'placeholder';
							elements.wrapper.appendChild(elements.placeholder);

							// Cover
							elements.cover = $CrE('div');
							elements.cover.id = 'cover';
							elements.cover.style.display = 'none';
							elements.wrapper.appendChild(elements.cover);

							// Cover text
							elements.covertext = $CrE('div');
							elements.covertext.id = 'covertext';
							elements.covertext.innerText = CONST.Text.CoverText;
							elements.cover.appendChild(elements.covertext);

							// Style
							elements.style = $CrE('style');
							elements.style.innerHTML = CSS_ImageUploader;
							elements.shadow.appendChild(elements.style);
							elements.shadow.appendChild(CommonStyle.ElmStyle.cloneNode(true));

							// Drag & drop
							$AEL(elements.wrapper, 'dragover', e => {
								// Preventdefault to allow drop
								e.preventDefault();

								// Display text hint on placeholder when dragover
								elements.placeholder.innerText = CONST.Text.CoverText;

								// Display cover when dragover && img shown
								if (elements.img.src && elements.img.width && elements.img.height) {
									const cover = elements.cover;
									cover.style.removeProperty('display');
								}
							});
							$AEL(elements.wrapper, 'drop', e => {
								// Preventdefault
								e.preventDefault();

								// Recover text hint on placeholder
								_this.#setStatus(_this.#status);

								// Hide cover when drop
								elements.cover.style.display = 'none';

								// Handle dropped data
								const items = e.dataTransfer.items;
								if (items.length > 1 && items[0].type !== 'text/uri-list') {
									alertify.alert(CONST.Text.ScriptTitle, CONST.Text.OneImageOnly);
									return false;
								}
								const item = items[0];
								if (item.kind !== 'file' && item.kind !== 'string') {
									// item is in DataTransferItem format, not directly a file or a string
									// Check whether item data is a file or a urilist, return if not
									return false;
								}
								if (item.type.split('/')[0] === 'image') {
									// Drag & drop image file
									const file = item.getAsFile();
									_this.image = file;
									_this.#autoUpload && _this.upload();
								} else if (item.type === 'text/uri-list') {
									// Drag & drop web image ([{type: 'text/uri-list'}, {type: 'text/html'}])
									item.getAsString(str => {
										if (!/^https?:\/\/[ \S]/.test(str) && !str.startsWith('blob:')) {
											alertify.alert(CONST.Text.ScriptTitle, CONST.Text.ImageOnly);
											return false;
										}
										_this.load(str);
									});
								}
							});
							$AEL(elements.wrapper, 'dragleave', e => {
								// Preventdefault
								e.preventDefault();

								// Recover text hint on placeholder
								_this.#setStatus(_this.#status);

								// Hide cover when drop
								elements.cover.style.display = 'none';
							});

							// Click to select
							$AEL(elements.wrapper, 'click', e => {
								const input = $$CrE({
									tagName: 'input',
									props: {
										type: 'file'
									}
								});
								$AEL(input, 'change', e => {
									const file = input.files[0];
									if (file.type.split('/')[0] !== 'image') {
										alertify.alert(CONST.Text.ScriptTitle, CONST.Text.ImageOnly);
										return false;
									}
									_this.image = file;
									_this.#autoUpload && _this.upload();
								});
								input.click();
							});

							this.#setStatus('empty');
						}

						get image() {
							return this.#image.image || null;
						}

						set image(val) {
							if (isBlobFile(val)) {
								const image = this.#image;

								// Revoke old object url
								image.objurl && URL.revokeObjectURL(image.objurl);

								// Save new image and object url
								image.image = val;
								image.objurl = URL.createObjectURL(val);

								// Set image src
								this.#elements.img.src = image.objurl;

								// Update status
								this.#setStatus('waiting');
							}
							return val;

							function isBlobFile(val) {
								return typeof val.toString === 'function' && ['[object Blob]', '[object File]'].includes(val.toString());
							}
						}

						get url() {
							return this.#url;
						}

						set url(val) {
							if ((typeof val !== 'string' || !val.match(/(https?:\/\/|data:\/\/)/)) && val) {
								return val;
							}
							if (!val) {
								val = '';
							}
							this.#url = val;
							this.#elements.img.src.startsWith('blob:') && URL.revokeObjectURL(this.#elements.img.src);
							this.#elements.img.src = val;
							this.#setStatus('empty');

							return val;
						}

						get autoUpload() {
							return this.#autoUpload;
						}

						set autoUpload(val) {
							this.#autoUpload = typeof val === 'boolean' ? val : this.#autoUpload;
							return val;
						}

						get status() {
							return this.#status;
						}

						clear() {
							this.#image = {};
							this.#url = '';
							this.#setStatus('empty');
							this.#elements.img.src = '';
						}

						load(url) {
							GM_xmlhttpRequest({
								method: 'GET',
								url: url,
								responseType: 'blob',
								onload: res => {
									const blob = res.response;
									this.image = blob;
									this.#autoUpload && this.upload();
								},
								onerror: err => {
									typeof err.error === 'string' && err.error.startsWith('Refused to connect to ') &&
										alertify.alert(CONST.Text.ConnectRequired_Title, CONST.Text.ConnectRequired);
								},
							});
						}

						upload() {
							const _this = this;
							this.#setStatus('uploading');
							this.#upload(function onload(url) {
								_this.#elements.img.src.startsWith('blob:') && URL.revokeObjectURL(_this.#elements.img.src);
								_this.#elements.img.src = url;
								_this.#url = url;
								_this.#setStatus('uploaded');
								_this.dispatchEvent(new Event('load'));
							}, function onerror() {
								_this.#setStatus('error');
								_this.dispatchEvent(new Event('error'));
							})
						}

						#upload(onload, onerror) {
							const image = this.#image;
							this.#url = null;

							if (!image) {
								return false;
							}

							const file = new File([image.image], `image.${getExtname(image.image.type)}`);
							this.#UPLOADER.upload(file, onload, onerror);

							function getExtname(...args) {
								const map = {
									'image/png': 'png',
									'image/jpg': 'jpg',
									'image/gif': 'gif',
									'image/bmp': 'bmp',
									'image/jpeg': 'jpeg',
									'image/webp': 'webp',
									'image/tiff': 'tiff',
									'image/vnd.microsoft.icon': 'ico',
									'default': 'jpg'
								};
								return map[args.find(a => map[a]) || 'default'];
							}
						}

						#setStatus(status) {
							const STATUSES = ['empty', 'waiting', 'uploading', 'uploaded', 'error'];
							const TEXT = CONST.Text.StatusText;
							const list = this.#elements.placeholder.classList;
							STATUSES.forEach(s => {
								s === status && !list.contains(s) && list.add(s);
								s !== status && list.contains(s) && list.remove(s);
							});
							this.#elements.placeholder.innerText = TEXT[status];
							this.#status = status;
						}
					}

					window.customElements.define('image-uploader', ImageUploader);
				}
			}
		},

		// ChapterUnlocker
		{
			name: '在线阅读解锁',
			description: '技术测试',
			id: 'ChapterUnlocker',
			system: false,
			checker: {
				type: 'func',
				value: () => {
					return (location.pathname.startsWith('/novel/') || location.pathname.match(/\/modules\/article\/reader.php/)) && unsafeWindow.chapter_id !== '0'
				}
			},
			func: function() {
				const AndroidAPI = require('AndroidAPI');
				const CommonStyle = require('CommonStyle');
				const utils = require('utils');
				const TEXT_GUI_NOVEL_FILLING = `</br><span class=${escJsStr(CommonStyle.ClassName.Text)}>[轻小说文库+] 正在获取章节内容...</span>`;

				detectDom('#footlink', function(footlink) {
					// Check whether needs filling
					if ($('#contentmain>span')) {
						if ($('#contentmain>span').innerText.trim() !== 'null') {
							return false;
						}
					} else {return false;}

					// prepare
					const content = $('#content');
					content.innerHTML = TEXT_GUI_NOVEL_FILLING;
					const charset = utils.getLang();

					// Get content xml
					AndroidAPI.getNovelContent({
					aid: unsafeWindow.article_id,
					cid: unsafeWindow.chapter_id,
					lang: charset,
					callback: function(text) {
						const imgModel = '<div class="divimage"><a href="{U}" target="_blank"><img src="{U}" border="0" class="imagecontent"></a></div>';

						// Trim whitespaces
						text = text.trim();

						// Get images like <!--image-->http://pic.wenku8.com/pictures/0/716/24406/11588.jpg<!--image-->
						const imgUrls = text.match(/<!--image-->[^<>]+?<!--image-->/g) || [];

						// Parse <img> for every image url
						let html = '';
						for (const url of imgUrls) {
							const index = text.indexOf(url);
							const src = utils.htmlEncode(url.match(/<!--image-->([^<>]+?)<!--image-->/)[1]);
							html += utils.htmlEncode(text.substring(0, index)).replaceAll('\r\n', '\n').replaceAll('\r', '\n').replaceAll('\n', '</br>');
							html += imgModel.replaceAll('{U}', src);
							text = text.substring(index + url.length);
						}
						html += utils.htmlEncode(text);

						// Set content
						content.innerHTML = html;
					}
				});
				});

				return true;
			}
		},

		// DownloadUnlocker
		{
			name: '下载解锁',
			description: '技术测试',
			id: 'DownloadUnlocker',
			system: false,
			checker: {
				type: 'regurl',
				value: [
					/^https?:\/\/www\.wenku8\.(net|cc)\/book\/\d+\.htm/,
					/^https?:\/\/www\.wenku8\.(net|cc)\/modules\/article\/articleinfo\.php/,
					/^https?:\/\/www\.wenku8\.(net|cc)\/modules\/article\/packshow\.php\?/
				]
			},
			func: function() {
				const CommonStyle = require('CommonStyle');
				const utils = require('utils');

				const CONST = {
					Text: {
						AnnounceText: '解锁功能加载完毕,仅供技术测试',
						PackshowInit: '正在初始化下载页面,请稍候...',
						PackshowInitTitle: '初始化...',
						PackshowTitle: {
							txt: '{N} 轻小说TXT下载 - 轻小说文库',
							txtfull: '{N} 轻小说TXT全本下载 - 轻小说文库',
							umd: '{N} 轻小说UMD电子书下载 - 轻小说文库',
							jar: '{N} 轻小说JAR电子书下载 - 轻小说文库',
						},
						PackshowReady: '初始化下载页面成功',
						UnkownValue: '未知'
					},
					URL: {
						BookIntro: `https://${location.host}/book/{A}.htm`
					},
					DomRes: {
						DownloadPanel: {
							title: ['《{Name}》小说TXT、UMD、JAR电子书下载', '《{Name}》小說TXT、UMD、JAR電子書下載'],
							style: 'width:820px;height:35px;margin:0px auto;padding:0px;',
							links: [{
								style: 'width:210px; float:left; text-align:center;',
								url: `https://${location.host}/modules/article/packshow.php?id={AID}&type=txt`,
								text: ['TXT简繁分卷', 'TXT簡繁分卷']
							}, {
								style: 'width:210px; float:left; text-align:center;',
								url: `https://${location.host}/modules/article/packshow.php?id={AID}&type=txtfull`,
								text: ['TXT简繁全本', 'TXT簡繁全本']
							}, {
								style: 'width:210px; float:left; text-align:center;',
								url: `https://${location.host}/modules/article/packshow.php?id={AID}&type=umd`,
								text: ['UMD分卷下载', 'UMD分卷下載']
							}, {
								style: 'width:190px; float:left; text-align:center;',
								url: `https://${location.host}/modules/article/packshow.php?id={AID}&type=jar`,
								text: ['JAR分卷下载', 'JAR分卷下載']
							}]
						}
					}
				};

				const functions = [{
					checker: {
						type: 'regurl',
						value: [
							/^https?:\/\/www\.wenku8\.(net|cc)\/book\/\d+\.htm/,
							/^https?:\/\/www\.wenku8\.(net|cc)\/modules\/article\/articleinfo\.php/
						]
					},
					detectDom: '.main.m_foot',
					func: function() {
						const DPM = new DownloadPanelManager(CONST.DomRes.DownloadPanel);

						function DownloadPanelManager(detail) {
							const DPM = this;

							const L = utils.getLang();
							const content = $('#content');
							const aid = getAID();
							const name = $('#content table table b').innerText;
							const panel = $('#content fieldset>legend>b') || makePanel();

							function makePanel() {
								const board = $(content, 'table:nth-of-type(2) td:nth-of-type(2) .hottext:nth-of-type(2)');
								board.className = CommonStyle.ClassName.Text;
								board.innerText = CONST.Text.AnnounceText

								const container = $CrE('div');
								const fieldset = $CrE('fieldset');
								fieldset.style.cssText = detail.style;
								container.appendChild(fieldset);

								const legend = $CrE('legend');
								const b = $CrE('b');
								b.innerText = replaceText(detail.title[L], {'{AID}': aid, '{Name}': name});
								legend.appendChild(b);
								fieldset.appendChild(legend);

								for (const link of detail.links) {
									const div = $CrE('div');
									const a = $CrE('a');
									div.style.cssText = link.style;
									a.href = replaceText(link.url, {'{AID}': aid, '{Name}': name});
									a.innerText = link.text[L];
									div.appendChild(a);
									fieldset.appendChild(div);
								}

								$('#content>div:first-child').insertAdjacentElement('beforeend', container);

								return container;
							}

							function getAID() {
								const m1 = location.pathname.split('/').pop().match(/\d+/);
								return m1 ? m1[0] : getUrlArgv('id');
							}
						}
					}
				}, {
					checker: {
						type: 'regurl',
						value: /^https?:\/\/www\.wenku8\.(net|cc)\/modules\/article\/packshow\.php\?/
					},
					detectDom: '.blocknote, .main.m_foot',
					func: function(returnValue) {
						if ($All('.block').length > 1) {
							FL_postMessage('PackshowReady');
							return {packshowReady: true};
						}

						let xmlInfo, xmlIndex;
						const AndroidAPI = require('AndroidAPI');
						const AM = new AsyncManager();
						const aid = getUrlArgv('id');
						const type = getUrlArgv('type');
						const lang = utils.getLang();

						// Soft alert
						const block_loading = $('.blockcontent>:first-child');
						block_loading.innerText = `\n\n${CONST.Text.PackshowInit}\n\n\n`;
						block_loading.classList.add(require('CommonStyle').ClassName.Text);
						block_loading.style.fontSize = '2em';
						block_loading.style.textAlign = 'center';
						$('.blocktitle').innerText = CONST.Text.PackshowInitTitle;
						document.title = CONST.Text.PackshowInitTitle;

						// Load model page
						const url = location.href.replace(/([\?&]id=)\d+/, '$11');
						utils.getDocument(url, doc => {
							const div = $('.block').parentElement.parentElement;
							div.style.display = 'none';
							for (const elm of [...doc.body.children]) {
								document.body.insertBefore(elm, div);
							}
							div.remove();
							$('#plus-api-beautify')?.remove();
							AM.finish();
						});
						AM.add();

						// Load shortInfo
						AndroidAPI.getNovelShortInfo({
							aid, lang,
							callback: xml => {
								xmlInfo = xml;
								AM.finish();
							}
						});
						AM.add();

						// Load index
						AndroidAPI.getNovelIndex({
							aid, lang,
							callback: xml => {
								xmlIndex = xml;
								AM.finish();
							}
						});
						AM.add();

						AM.onfinish = fetchFinish;
						AM.finishEvent = true;

						function fetchFinish() {
							// Elements
							const content = $(document, '#content');
							const table = $(content, 'table');
							const tbody = $(table, 'tbody');

							// Data
							const name = $(xmlInfo, 'data[name="Title"]').childNodes[0].nodeValue;
							const lastupdate = $(xmlInfo, 'data[name="LastUpdate"]').getAttribute('value');
							const aBook = $(table, 'caption>a:first-child');
							const charsets = ['gbk', 'utf-8', 'big5', 'gbk', 'utf-8', 'big5'];
							const innerTexts = [
								['简体(G)', '简体(U)', '繁体(U)', '简体(G)', '简体(U)', '繁体(U)'],
								['簡體(G)', '簡體(U)', '繁體(U)', '簡體(G)', '簡體(U)', '繁體(U)']
							][lang];

							// Set Title
							document.title = replaceText(CONST.Text.PackshowTitle[type], {'{N}': name});

							// Set book
							aBook.innerText = name;
							aBook.href = replaceText(CONST.URL.BookIntro, {'{A}': aid});

							// Load book index
							loadIndex();

							// Soft alert
							alertify.success(CONST.Text.PackshowReady);
							block_loading.innerText = `\n\n${CONST.Text.PackshowReady}\n\n\n`;

							// callback
							FL_postMessage('PackshowReady');
							returnValue.packshowReady = true;

							// Book index loader
							function loadIndex() {
								switch (type) {
									case 'txt':
										loadIndex_txt();
										break;
									case 'txtfull':
										loadIndex_txtfull();
										break;
									case 'umd':
										loadIndex_umd();
										break;
									case 'jar':
										loadIndex_jar();
										break;
								}
							}

							// Book index loader for type txt
							function loadIndex_txt() {
								// Clear tbody trs
								for (const tr of $All(table, 'tr+tr')) {
									tbody.removeChild(tr);
								}

								// Make new trs
								for (const volume of $All(xmlIndex, 'volume')) {
									const tr = makeTr(volume);
									tbody.appendChild(tr);
								}

								function makeTr(volume) {
									const tr = $CrE('tr');
									const [tdName, td1, td2] = [$CrE('td'), $CrE('td'), $CrE('td')];
									const a = Array(6);
									const vid = volume.getAttribute('vid');
									const vname = volume.childNodes[0].nodeValue;

									// init tds
									tdName.classList.add('odd');
									td1.classList.add('even');
									td2.classList.add('even');
									td1.align = td2.align = 'center';

									// Set volume name
									tdName.innerText = vname;

									// Make <a> links
									for (let i = 0; i < a.length; i++) {
										a[i] = $CrE('a');
										a[i].target = '_blank';
										a[i].href = 'http://dl.wenku8.com/packtxt.php?aid=' + aid +
											'&vid=' + vid +
											(i >= 3 ? '&aname=' + $URL.encode(name) : '') +
											(i >= 3 ? '&vname=' + $URL.encode(vname) : '') +
											'&charset=' + charsets[i];
										a[i].innerText = innerTexts[i];
										(i < 3 ? td1 : td2).appendChild(a[i]);
									}

									// Insert whitespace textnode
									for (const i of [1, 2, 4, 5]) {
										(i < 3 ? td1 : td2).insertBefore(document.createTextNode('\n'), a[i]);
									}

									tr.appendChild(tdName);
									tr.appendChild(td1);
									tr.appendChild(td2);

									return tr;
								}
							}

							// Book index loader for type txtfull
							function loadIndex_txtfull() {
								const tr = $(tbody, 'tr+tr');
								const tds = Array.prototype.map.call(tr.children, (elm) => (elm));

								tds[0].innerText = lastupdate;
								tds[1].innerText = CONST.Text.UnkownValue;
								for (const a of $All(tds[2], 'a')) {
									a.href = a.href.replace(/id=\d+/, 'id='+aid).replace(/fname=[^&]+/, 'fname='+$URL.encode(name));
								}
							}

							// Book index loader for type umd
							function loadIndex_umd() {
								const tr = $(tbody, 'tr+tr');
								const tds = Array.from(tr.children);

								tds[0].innerText = tds[1].innerText = CONST.Text.UnkownValue;
								tds[2].innerText = lastupdate;
								tds[3].innerText = $(xmlIndex, 'volume:first-child').childNodes[0].nodeValue + '—' + $(xmlIndex, 'volume:last-child').childNodes[0].nodeValue;
								const as = [].concat(Array.from($All(tds[4], 'a'))).concat(Array.from($All(table, 'caption>a+a')));
								for (const a of as) {
									a.href = a.href.replace(/id=\d+/, 'id='+aid);
								}
							}

							// Book index loader for type jar
							function loadIndex_jar() {
								// Currently type jar is the same as type umd
								loadIndex_umd();
							}
						}
					}
				}];

				return utils.loadFuncs(functions);
			}
		},

		// Download Enhance
		{
			name: '下载增强',
			description: '下载界面增强,提供一键下载全部分卷、快捷切换下载服务器等小功能',
			id: 'DownloadEnhance',
			system: false,
			checker: {
				type: 'regurl',
				value: /^https?:\/\/www\.wenku8\.(net|cc)\/modules\/article\/packshow\.php\?/
			},
			func: function() {
				const DownloadUnlocker = require('DownloadUnlocker');
				const CommonStyle = require('CommonStyle');
				const mousetip = require('mousetip');
				const utils = require('utils');

				const CONST = {
					Text: {
						CheckHeader: '全选',
						DownloadAllChecked: '下载全部选中项,请点击右侧按钮:',
						NothingHere: '-- Nothing Here --',
						DIYDownloadHeader: '自定义名称格式下载',
						SetdownloadFormatTip: '点击设置下载文件的名称格式',
						SetdownloadFormatTitle: '自定义下载名称格式 - {TYPE}',
						SetdownloadFormatMessage: '下载{TYPE}时使用以下格式命名文件,以下标识符将会替换为对应的内容:</br>{SM}: 书名</br>{JM}: 卷名</br>{KZM}:文件扩展名(如"txt"/"umd"等,不含".")</br>{ID}: 文库的小说id</br>注:根据下载类型的不同,某些标识符可能不会被替换(比如下载TXT全本时,不存在分卷,所以{JM}不会被替换)',
						PleaseSelect: '请先在左边勾选需要下载的章节,</br>再点击批量下载按钮</br>最上面的复选框可以进行全选',
						DownloadStatus_Queued: '(排队中)',
						DownloadStatus_Downloading: '(下载中)',
						DownloadStatus_Finished: '(已下载)',
						DownloadStatus_Error: '(下载失败,请重试)',
						ServerChangerTip: '点击切换到此下载服务器',
						ServerChanged: '已切换到{Server}',
						DownloadType: {
							txt: 'TXT分卷',
							txtfull: 'TXT全本',
							umd: 'UMD电子书',
							jar: 'JAR电子书'
						}
					},
					Config_Ruleset: {
						'version-key': 'config-version',
						'ignores': ["LOCAL-CDN"],
						'defaultValues': {
							//'config-key': {},
							downloadFormat: {
								txt: '{SM} {JM}.{KZM}',
								txtfull: '{SM}.{KZM}',
								umd: '{SM}.{KZM}',
								jar: '{SM}.{KZM}',
								'config-version': 2
							}
						},
						'updaters': {
							/*'config-key': [
								function() {
									// This function contains updater for config['config-key'] from v0 to v1
								},
								function() {
									// This function contains updater for config['config-key'] from v1 to v2
								}
							]*/
							downloadFormat: [
								function(downloadFormat) {
									return {
										txt: downloadFormat.txt,
										txtfull: '{SM}.{KZM}'
									};
								},
								function(downloadFormat) {
									return {
										...downloadFormat,
										umd: '{SM}.{KZM}',
										jar: '{SM}.{KZM}'
									};
								}]
						}
					}
				};

				const CM = new ConfigManager(CONST.Config_Ruleset);
				const CONFIG = CM.Config;
				CM.updateAllConfigs();

				detectDom('.main.m_foot', enhance);

				function enhance() {
					const doc = DownloadUnlocker.iframe ? DownloadUnlocker.iframe.contentDocument : document;
					return {
						'txt': enhance_txt,
						'txtfull': ehance_txtfull,
						'jar': enhance_jar,
						'umd': enhance_umd
					}[getUrlArgv('type')](doc);
				}

				function enhance_txt(doc) {
					const content = $(doc, '#content');
					const header = $(content, 'tr:first-child');

					// Header
					const thChecker = $CrE('th');
					thChecker.innerText = CONST.Text.CheckHeader;
					thChecker.style.textAlign = 'center';
					header.insertAdjacentElement('afterbegin', thChecker);

					const thDownload = $CrE('th');
					const text = $CrE('span');
					const nameSetting = $CrE('i');
					text.innerText = CONST.Text.DIYDownloadHeader;
					nameSetting.className = 'fa-solid fa-gear';
					nameSetting.style.marginLeft = '0.3em';
					nameSetting.style.cursor = 'pointer';
					mousetip.settip(nameSetting, CONST.Text.SetdownloadFormatTip);
					$AEL(nameSetting, 'click', setDownloadFormat.bind(null, 'txt'));
					thDownload.appendChild(text);
					thDownload.appendChild(nameSetting);
					header.appendChild(thDownload);

					const widths = ['5%', '35%', '20%', '20%', '20%'];
					[...$All(content, 'tr:first-child>th')].forEach((th, i) => {
						th.style.width = widths[i];
					});

					// Row
					for (const tr of $All(content, 'tbody>tr:not(:first-child)')) {
						// Batch checker
						const cTd = $CrE('td');
						const checker = $CrE('input');
						checker.type = 'checkbox';
						cTd.style.textAlign = 'center';
						cTd.style.cursor = checker.style.cursor = 'pointer';
						$AEL(cTd, 'click', e => e.srcElement === cTd && checker.click());
						cTd.appendChild(checker);
						tr.insertAdjacentElement('afterbegin', cTd);

						// Download display
						const tdName = tr.children[1];
						const vtitle = $CrE('span');
						vtitle.innerText = tdName.innerText;
						tdName.innerHTML = '';
						tdName.appendChild(vtitle);
						const dltext = $CrE('span');
						dltext.style.marginLeft = '0.3em';
						tdName.appendChild(dltext);

						// Download button
						const td = tr.children[2].cloneNode(true);
						[...td.children].forEach(a => {
							a.classList.add(CommonStyle.ClassName.Button);
							$AEL(a, 'click', e => {
								e.preventDefault();
								dltext.innerText = CONST.Text.DownloadStatus_Queued;

								downloadAndUnserielize({
									url: a.href,
									name: replaceText(CONFIG.downloadFormat.txt, {
									'{SM}': $(content, 'caption>a').innerText,
									'{JM}': tdName.children[0].innerText,
									'{KZM}': 'txt',
									'{ID}': getUrlArgv('id')
								}),
									encoding: getUrlArgv(a.href, 'charset'),
									options: {
										onloadstart: function() {
											dltext.innerText = CONST.Text.DownloadStatus_Downloading;
										},
										onload: function() {
											dltext.innerText = CONST.Text.DownloadStatus_Finished;
										},
										onerror: function() {
											dltext.innerText = CONST.Text.DownloadStatus_Error;
										}
									}
								});
							});
						});
						tr.appendChild(td);
					}

					// Batch download
					const getAllCheckbox = () => [...$All(content, 'tbody>tr:not([plus_batch])>td>input[type="checkbox"]')];
					const batchTr = $CrE('tr');
					batchTr.setAttribute('plus_batch', '');
					for (let i = 0; i < 5; i++) {
						const td = $CrE('td');
						td.style.textAlign = 'center';
						switch (i) {
							case 0: {
								const allcheck = $CrE('input');
								allcheck.type = 'checkbox';
								td.style.cursor = allcheck.style.cursor = 'pointer';
								$AEL(allcheck, 'change', e => {
									getAllCheckbox().forEach(box => {box.checked = allcheck.checked});
								});
								$AEL(td, 'click', e => e.srcElement === td && allcheck.click());
								td.appendChild(allcheck);

								const tbody = $(content, 'tbody');
								$AEL(tbody, 'change', e => {
									allcheck.checked = getAllCheckbox().every(input => input.checked);
								});
								break;
							}
							case 1: {
								td.innerText = CONST.Text.DownloadAllChecked;
								td.classList.add('odd');
								td.style.removeProperty('text-align');
								td.classList.add(CommonStyle.ClassName.Text);
								break;
							}
							case 2:
							case 3: {
								td.classList.add('even');
								td.style.color = 'grey';
								td.innerText = CONST.Text.NothingHere;
								break;
							}
							case 4: {
								td.classList.add('even');
								[...$(content, 'tbody>tr:last-child>td:last-child').children].forEach((_a, i) => {
									const a = _a.cloneNode(true);
									a.href = 'javascript:void(0);';
									a.target = 'self';
									$AEL(a, 'click', e => {
										e.preventDefault();
										const trs = [...$All(content, 'tbody>tr:not(:first-child):not([plus_batch])')].filter(tr => $(tr, 'input[type="checkbox"]').checked);
										if (trs.length) {
											trs.forEach(tr => $(tr, `td:last-child>a:nth-of-type(${i+1})`).click());
										} else {
											alertify.message(CONST.Text.PleaseSelect);
										}
										/*
										[...$All(content, 'tbody>tr:not(:first-child):not([plus_batch])')].forEach(tr => {
											if ($(tr, 'input[type="checkbox"]').checked) {
												$(tr, `td:last-child>a:nth-of-type(${i+1})`).click();
											}
										});
										*/
									});
									td.appendChild(document.createTextNode(' '));
									td.appendChild(a);
								});
								break;
							}
						}
						batchTr.appendChild(td);
					}
					$(content, 'tbody>tr:first-child').insertAdjacentElement('afterend', batchTr);

					// Sever changer
					serverChanger();
				}

				function ehance_txtfull(doc) {
					const content = $(doc, '#content');

					const td = $(content, 'tbody td:last-child');
					td.appendChild($CrE('br'));
					td.appendChild(document.createTextNode(CONST.Text.DIYDownloadHeader));
					const diyBtn = $CrE('i');
					diyBtn.className = 'fa-solid fa-gear';
					mousetip.settip(diyBtn, CONST.Text.SetdownloadFormatTip);
					diyBtn.style.marginLeft = '0.3em';
					diyBtn.style.cursor = 'pointer';
					$AEL(diyBtn, 'click', setDownloadFormat.bind(null, 'txtfull'));
					td.appendChild(diyBtn);
					td.appendChild(document.createTextNode('('));
					const text = [['简体(G)', '简体(U)', '繁体(U)'], ['簡體(G)', '簡體(U)', '繁體(U)']][utils.getLang()];
					[1, 3, 5].forEach((n, i) => {
						const _a = $(td, `a:nth-child(${n})`);
						const a = $CrE('a');
						a.href = 'javascript: void(0);';
						a.innerText = text[i];
						a.classList.add(CommonStyle.ClassName.Button);
						$AEL(a, 'click', e => {
							e.preventDefault();
							downloadAndUnserielize({
								url: _a.href,
								name: replaceText(CONFIG.downloadFormat.txtfull, {
									'{SM}': $(content, 'caption>a').innerText,
									'{KZM}': 'txt',
									'{ID}': getUrlArgv('id')
								}),
								encoding: getUrlArgv(_a.href, 'type').replace(/^txt$/, 'gbk'),
							});
						});
						td.appendChild(a);
						n < 5 && td.appendChild(document.createTextNode(' '));
					});
					td.appendChild(document.createTextNode(')'));

					// Sever changer
					serverChanger();
				}

				function enhance_jar(doc) {
					const content = $(doc, '#content');

					// DIY name download
					const th = $CrE('th');
					th.innerText = CONST.Text.DIYDownloadHeader;
					$(doc, 'tbody>:first-child').appendChild(th);
					const setting = $CrE('i');
					setting.className = 'fa-solid fa-gear';
					mousetip.settip(setting, CONST.Text.SetdownloadFormatTip);
					setting.style.marginLeft = '0.3em';
					setting.style.cursor = 'pointer';
					$AEL(setting, 'click', setDownloadFormat.bind(null, 'jar'));
					th.appendChild(setting);
					const td = $CrE('td');
					td.align = 'center';
					$(doc, 'tbody>:last-child').appendChild(td);
					const _td = td.previousElementSibling;

					const widths = [5, 5, 9, 41, 20, 20].map(num => `${num}%`);
					[...$All(doc, 'table th')].forEach((th, i) => th.width = widths[i]);

					[...$All(_td, 'a')].forEach((a, i, arr) => {
						const dlBtn = $CrE('span');
						dlBtn.classList.add(CommonStyle.ClassName.Button);
						dlBtn.innerText = a.innerText;
						$AEL(dlBtn, 'click', e => dl_GM({
							url: $(_td, `a:nth-child(${i+1})`).href,
							name: replaceText(CONFIG.downloadFormat.jar, {
								'{SM}': $(content, 'caption>a').innerText,
								'{KZM}': ['jar', 'jad'][i],
								'{ID}': getUrlArgv('id')
							})
						}));
						td.appendChild(dlBtn);

						i < arr.length && td.appendChild(doc.createTextNode(' '));
					});

					// Sever changer
					serverChanger();
				}

				function enhance_umd(doc) {
					const content = $(doc, '#content');

					// DIY name download
					const th = $CrE('th');
					th.innerText = CONST.Text.DIYDownloadHeader;
					$(doc, 'tbody>:first-child').appendChild(th);
					const setting = $CrE('i');
					setting.className = 'fa-solid fa-gear';
					mousetip.settip(setting, CONST.Text.SetdownloadFormatTip);
					setting.style.marginLeft = '0.3em';
					setting.style.cursor = 'pointer';
					$AEL(setting, 'click', setDownloadFormat.bind(null, 'umd'));
					th.appendChild(setting);
					const td = $CrE('td');
					td.align = 'center';
					$(doc, 'tbody>:last-child').appendChild(td);
					const _td = td.previousElementSibling;

					const widths = [5, 5, 9, 41, 20, 20].map(num => `${num}%`);
					[...$All(doc, 'table th')].forEach((th, i) => th.width = widths[i]);

					[...$All(_td, 'a')].forEach((a, i, arr) => {
						const dlBtn = $CrE('span');
						dlBtn.classList.add(CommonStyle.ClassName.Button);
						dlBtn.innerText = a.innerText;
						$AEL(dlBtn, 'click', e => dl_GM({
							url: $(_td, `a:nth-child(${i+1})`).href,
							name: replaceText(CONFIG.downloadFormat.jar, {
								'{SM}': $(content, 'caption>a').innerText,
								'{KZM}': ['umd'][i],
								'{ID}': getUrlArgv('id')
							})
						}));
						td.appendChild(dlBtn);

						i < arr.length && td.appendChild(doc.createTextNode(' '));
					});

					// Sever changer
					serverChanger();
				}

				function setDownloadFormat(type) {
					alertify.prompt(replaceText(CONST.Text.SetdownloadFormatTitle, {'{TYPE}': CONST.Text.DownloadType[type]}), replaceText(CONST.Text.SetdownloadFormatMessage, {'{TYPE}': CONST.Text.DownloadType[type]}), CONFIG.downloadFormat[type], (e, val) => CM.setConfig(`downloadFormat/${type}`, val), function() {});
				}

				function serverChanger() {
					// Sever changer
					const doc = DownloadUnlocker.iframe ? DownloadUnlocker.iframe.contentDocument : document;
					const content = $(doc, '#content');
					for (const b of $All(doc, '#content>b')) {
						const host = b.innerText;
						const matcher = /dl(\d+)?\.wenku8\.com/;
						if (host.match(matcher)) {
							b.classList.add(CommonStyle.ClassName.Button);
							$AEL(b, 'click', e => {
								[...$All(content, 'a[href]')].filter(a => a.href.match(matcher)).forEach(a => a.href = a.href.replace(matcher, host));
								mousetip.showtip(replaceText(CONST.Text.ServerChanged, {'{Server}': host}));
							});
							mousetip.settip(b, CONST.Text.ServerChangerTip);
						}
					}
				}

				// Args: url, name, encoding, options || <object>details
				// Encoding must be one of follows: 'utf-8'(and alias like 'utf8', 'UTF-8', 'UTF8' ...), 'gbk', 'big5'
				function downloadAndUnserielize() {
					let [url, name, encoding, options] = parseArgs([...arguments], [
						function(args, defaultValues) {
							const arg = args[0];
							return ['url', 'name', 'encoding', 'options'].map((prop, i) => arg.hasOwnProperty(prop) ? arg[prop] : defaultValues[i]);
						},
						[1, 2],
						[1, 2, 3],
						[1, 2, 3, 4]
					], ['', 'text.txt', 'utf-8', {}]);
					encoding = encoding.toLowerCase();
					const xhrOptions = {
						method: 'GET', url,
						responseType: 'blob',
						onload: function(response) {
							const reader = new FileReader();
							reader.onload = function() {
								let text = reader.result;
								for (const match of text.match(/&#\d{1,6};/g)) {
									text = text.replace(match, utils.htmlDecode(match));
								}
								const buf = encoding.replace('-', '') === 'utf8' ? new TextEncoder().encode(text) : $URL[encoding].encodeBuffer(text);
								const blob = new Blob([buf], { type: 'text/plain' });
								const objurl = URL.createObjectURL(blob);
								dl_browser(objurl, name);
								setTimeout(e => URL.revokeObjectURL(objurl));
							}
							reader.readAsText(response.response, encoding);
						},
					};
					for (const [key, value] of Object.entries(options)) {
						if (typeof value === 'function' && typeof xhrOptions[key] === 'function') {
							// Call both originalCallback and userCallback if userCallback provided
							const [originalCallback, userCallback] = [xhrOptions[key], value];
							xhrOptions[key] = function() {
								originalCallback.apply(this, arguments);
								return userCallback.apply(this, arguments);
							}
						} else {
							// Overwrite other properties
							xhrOptions[key] = value;
						}
					}
					GM_xmlhttpRequest(xhrOptions);
				}
			}
		},

		// Image batch download
		// 现在是全部插图下载,预计要做成一个小说多格式自定义下载器
		{
			name: '插图批量下载',
			description: '批量下载插图',
			id: 'ImageBatchDownloader',
			system: false,
			checker: {
				type: 'regurl',
				value: [
					/^https?:\/\/www\.wenku8\.(net|cc)\/book\/\d+\.htm([\?#][\s\S]*)?$/,
					/^https?:\/\/www\.wenku8\.(net|cc)\/modules\/article\/articleinfo\.php([\?#][\s\S]*)?$/,
				]
			},
			CONST: {
				Text: {
					DownloadAllImages: '全部插图下载',
					Image: ['插图', '插圖']
				}
			},
			func: function() {
				const CONST = FuncInfo.CONST;
				const AndroidAPI = require('AndroidAPI');
				const utils = require('utils');

				const aid = getUrlArgv('id') || location.pathname.match(/\/book\/(\d+)\.htm/)[1];
				const lang = utils.getLang();

				const container = $$CrE({
					tagName: 'div',
					styles: {
						float: 'left',
						'text-align': 'center'
					}
				});
				const btn = $$CrE({
					tagName: 'a',
					props: {
						innerText: CONST.Text.DownloadAllImages,
						href: 'javascript: void(0);'
					},
					listeners: [['click', e => dlAllImgs()]]
				});
				container.appendChild(btn);

				detectDom('#content>div:nth-child(3)', e => {
					const fieldset = $('legend+div[style^="width"]').parentElement;
					fieldset.appendChild(container);
					Array.from($All(fieldset, 'div')).forEach(div => div.style.width = '164px');
				});

				function dlAllImgs() {
					DoLog('Download All Images');
					AndroidAPI.getNovelIndex({
						aid, lang,
						callback: xml => {
							[AndroidAPI, aid, lang];

							/*
							const box = alertify.
							for (const volumn of $All(xml, 'volumn')) {
								for (const chapter of $All(volumn, 'chapter')) {
									//
								}
							}
							*/

							// Simply downloads all images
							let dl = 0, all = 0;
							const update = () => btn.innerText = `${CONST.Text.DownloadAllImages}(${dl}/${all})`;
							for (const chapter of $All(xml, 'chapter')) {
								const cid = chapter.getAttribute('cid');
								AndroidAPI.getNovelContent({
									aid, cid, lang,
									callback: text => {
										const imgurls = Array.from(text.matchAll(/<!--image-->([^<>]+?)<!--image-->/g)).map(match => match[1]);
										imgurls.forEach((url, i) => {
											const bookName = $('table table span>b').innerText;
											const volumnName = chapter.parentNode.firstChild.nodeValue;
											const num = (i+1).toString().padStart(imgurls.length.toString().length, '0');
											const ext = url.match(/\.([^\/]+)$/) ? url.match(/\.([^\/]+)$/)[1] : 'jpg';
											const name = `${bookName}_${volumnName} ${CONST.Text.Image[lang]}_${num}.${ext}`;
											dl_GM({
												url, name,
												onload: e => update(++dl) // this argument has no effect, just need to increase dl lol
											});
										});
										all += imgurls.length;
									}
								});
							}
							update();
						}
					});
				}
			}
		},

		// Beautifier
		{
			name: '页面美化',
			description: '自定义页面背景图',
			id: 'beautifier',
			system: false,
			Config_Ruleset: {
				'version-key': 'config-version',
				'ignores': ["LOCAL-CDN"],
				'defaultValues': {
					//'config-key': {},
					common: {
						enable: false,
						image: null,
						cover: {
							light: {
								opacity: 70,
								color: 'white',
								blur: 0,
							},
							dark: {
								opacity: 5,
								color: 'white',
								blur: 5,
							}
						}
					},
					novel: {
						enable: false,
						image: null,
						center: false,
						'config-version': 1
					},
					review: {
						enable: false,
						image: null,
						textScale: 100
					},
					image: null,
				},
				'updaters': {
					/*'config-key': [
						function() {
							// This function contains updater for config['config-key'] from v0 to v1
						},
						function() {
							// This function contains updater for config['config-key'] from v1 to v2
						}
					]*/
					novel: [
						function(novel) {
							// add prop 'center' to novel
							return {...novel, center: false};
						}
					]
				}
			},
			func: function() {
				const CommonStyle = require('CommonStyle');
				const utils = require('utils');

				const CONST = {
					Text: {
						CommonBeautify: '通用页面美化',
						NovelBeautify: '阅读页面美化',
						ReviewBeautify: '书评页面美化',
						DefaultBeautify: '默认页面美化图片',
						Enable: '启用',
						BackgroundImage: '背景图片',
						AlertTitle: '页面美化设置',
						InvalidImageUrl: '图片链接格式错误</br>仅仅接受http/https/data链接',
						textScale: '文字大小缩放'
					},
					ClassName: {
						BgImage: 'plus-cbty-image',
						BgCover: 'plus-cbty-cover',
						CSSCommon: 'plus-beautifier-common',
						CSSNovel: 'plus-beautifier-novel',
						CSSReview: 'plus-beautifier-review',
					},
					CSS: {
						Common: ':not(.plus-cbty)>:is(.plus-cbty-image, .plus-cbty-cover) {display: none;}.plus-cbty .plus-cbty-image {position: fixed;top: 0;left: 0;z-index: -2;width: 100vw;height: 100vh;}.plus-cbty .plus-cbty-cover {position: fixed;top: 0;left: calc((100vw - 960px) / 2);z-Index: -1;width: 960px;height: 100vh;backdrop-filter: blur([CoverBlur-Light]px);}.plus-cbty.plus-darkmode .plus-cbty-cover {backdrop-filter: blur([CoverBlur-Dark]px);}.plus-cbty .plus-cbty-cover::before {content: "";position: absolute;top: 0;right: 0;left: 0;bottom: 0;z-index:-1;background-color: [CoverColor-Light];opacity: [CoverOpacity-Light];}.plus-cbty.plus-darkmode .plus-cbty-cover::before {background-color: [CoverColor-Dark];opacity: [CoverOpacity-Dark];}body.plus-cbty {overflow: auto;margin: 0;}body.plus-cbty>:is(.main, table.css) {position: relative;margin-left: calc((100vw - 960px) / 2);}.plus-cbty :is(.textarea, .text) {background-color: rgba(255,255,255,0.9);}.plus-cbty #headlink{background-color: rgba(255,255,255,0.1);}.plus-cbty:is(.plus-darkmode, :not(.plus-darkmode)) :is(table.grid td, .odd, .even, .blockcontent) {background-color: transparent !important;}.plus-cbty:is(.plus-darkmode, :not(.plus-darkmode)) :is(.pagelink, .pagelink strong, .pagelink a:hover) {background-color: transparent;}.DoNotUse.plus-cbty:is(.plus-darkmode, :not(.plus-darkmode)) :is(.main.m_top, .nav, .navinner, .navlist, .nav li, .nav a.current, .nav a:hover, .nav a:active):not(#NotAElement>#NotAElement) {background: transparent;}.plus-cbty #mask {width: 100vw !important;height: 100vh !important;}.plus-cbty :is(.blocktitle, .nav, a) {user-select: none;}',
						Novel: 'body.plus-cbty {width: 100vw;height: 100vh;overflow: overlay;margin: 0px;background-color: rgba(255,255,255,0.7);}.plus-cbty #contentmain {overflow-y: auto;height: calc(100vh - [H]);min-width: 0px;max-width: [W];}.plus-cbty :is(#adv1, #adtop, #headlink, #footlink, #adbottom) {overflow: overlay;min-width: 0px;max-width: 100vw;}.plus-cbty :is(#adv900, #adv5) {max-width: 100vw;}.plus-cbty #contentmain {scrollbar-color: rgba(0,0,0,0.4) rgba(255,255,255,0.3);}.plus-cbty img.imagecontent{width:100% }.plus-cbty #contentmain::-webkit-scrollbar-thumb {background-color: rgba(0,0,0,0.4);}.plus-cbty #contentmain::-webkit-scrollbar {background-color: rgba(255,255,255,0.3);}',
						Review: '.plus-cbty #content > table > tbody > tr > td {background-color: transparent;overflow: auto;}.plus-cbty:is(.plus-darkmode, :not(.plus-darkmode)) :is(table.grid th, form[name="frmreview"] caption, input[type="checkbox"]::after) {background-color: transparent;}.plus-cbty #content {height: 100vh;overflow: auto;}.plus-cbty :is(.m_top, .m_head, .main.nav, .m_foot) {display: none;}.plus-cbty .main {margin-top: 0px;}.plus-cbty #content>table>tbody>tr{font-size: [S]%;line-height: 100%;}.plus-cbty :is(.jieqiQuote, .jieqiCode, .jieqiNote) {font-size: inherit;}.plus-cbty #content>table>tbody>tr>td:nth-of-type(2) {overflow-wrap: anywhere;}.plus-cbty #content>table>tbody>tr>td:nth-of-type(2) a {user-select: auto;}.plus-cbty .jieqiQuote *{overflow-wrap:break-word;max-width:calc(937px / 100 * 80);}.plus-cbty form[name="frmreview"]{position:relative;}'
					}
				};

				// Config
				const CM = new ConfigManager(FuncInfo.Config_Ruleset);
				const CONFIG = CM.Config;
				CM.updateAllConfigs();

				// Beautifier pages
				detectDom('body', function() {
					const API = getAPI();
					if (/\.(jpe?g|png|webp|gif|bmp|txt|js|css)/.test(location.pathname)) {
						// Stop running in image, code, or pure text papes
						return false;
					} else if ((API[0] === 'novel' || location.pathname.match(/\/modules\/article\/reader.php/)) && unsafeWindow.chapter_id !== '0') {
						CONFIG.novel.enable && (document.readyState !== 'loading' ? novel() : $AEL(document, 'DOMContentLoaded', e => novel()));
					} else if (API.join('/') === 'modules/article/reviewshow.php') {
						//CONFIG.review.enable && (document.readyState !== 'loading' ? review() : $AEL(window, 'load', e => review()));
						CONFIG.review.enable && detectDom('.main.m_foot', e => review());
					} else {
						CONFIG.common.enable && common();
					}
				});

				return {};

				// Beautifier for all wenku pages
				function common(from='common') {
					const src = (CONFIG[from] && CONFIG[from].image) || CONFIG.image;

					if (src) {
						const img = $CrE('img');
						img.src = src;
						img.classList.add(CONST.ClassName.BgImage);
						document.body.appendChild(img);
					}

					const cover = $CrE('div');
					cover.classList.add(CONST.ClassName.BgCover);
					document.body.appendChild(cover);

					document.body.classList.add('plus-cbty');
					addStyle(replaceText(CONST.CSS.Common, {
						'[CoverBlur-Light]': CONFIG.common.cover.light.blur.toString(),
						'[CoverOpacity-Light]': (CONFIG.common.cover.light.opacity / 100).toString(),
						'[CoverColor-Light]': CONFIG.common.cover.light.color.toString(),
						'[CoverBlur-Dark]': CONFIG.common.cover.dark.blur.toString(),
						'[CoverOpacity-Dark]': (CONFIG.common.cover.dark.opacity / 100).toString(),
						'[CoverColor-Dark]': CONFIG.common.cover.dark.color.toString(),
					}), CONST.ClassName.CSSCommon);
					return true;
				}

				// Novel reading page
				function novel() {
					common('novel');

					const src = CONFIG.novel.image || CONFIG.image;
					const usedHeight = getRestHeight();
					const contentWidth = CONFIG.novel.center ? '958px' : '100vw';

					addStyle(CONST.CSS.Novel
						 .replaceAll('[BGI]', src)
						 .replaceAll('[H]', usedHeight)
						 .replaceAll('[W]', contentWidth),
						 CONST.ClassName.CSSNovel
					);

					document.ondblclick = beautiful_beginScroll;
					document.onmousedown = beautiful_stopScroll;
					unsafeWindow.scrolling = beautiful_scrolling;

					// Get rest height without #contentmain
					function getRestHeight() {
						let usedHeight = 0;
						['adv1', 'adtop', 'headlink', 'footlink', 'adbottom'].forEach((id) => {
							const node = $('#'+id);
							if (node instanceof Element && node.id !== 'contentmain') {
								const cs = getComputedStyle(node);
								['height', 'marginTop', 'marginBottom', 'paddingTop', 'paddingBottom', 'borderTop', 'borderBottom'].forEach((style) => {
									const reg = cs[style].match(/([\.\d]+)px/);
									reg && (usedHeight += parseInt(reg[1], 10));
								});
							};
						});
						usedHeight = usedHeight.toString() + 'px';
						return usedHeight;
					}

					// Prevent dblclick-selecting text while dblclick start scrolling
					function beautiful_beginScroll(e) {
						// filter alertify and sidepanel
						let elm = e.target;
						const FORBID = {
							classes: ['alertify'],
							ids: ['sidepanel-panel']
						};
						while (elm = elm.parentElement) {
							if ([...elm.classList].some(cls => FORBID.classes.includes(cls)) || FORBID.ids.includes(elm.id)) {
								return;
							}
						}

						unsafeWindow.timer = setInterval(beautiful_scrolling, 300 / unsafeWindow.speed);
						$('#contentmain').style.userSelect = 'none';
					}
					function beautiful_stopScroll() {
						clearInterval(unsafeWindow.timer);
						$('#contentmain').style.userSelect = '';
					}

					// Mouse dblclick scroll with beautifier applied
					function beautiful_scrolling() {
						var contentmain = $('#contentmain');
						var currentpos = contentmain.scrollTop || 0;
						contentmain.scrollTo(0, ++currentpos);
						var nowpos = contentmain.scrollTop || 0;
						if(currentpos != nowpos) unsafeWindow.clearInterval(unsafeWindow.timer);
					}
				}

				// Review reading page
				function review() {
					common('review');

					const src = CONFIG.review.image || CONFIG.image;
					const textScale = CONFIG.review.textScale;
					const main = $('#content');
					const recScrollY = e => {
						//scrollY = unsafeWindow.scrollY - ['.main.m_top', '.main.m_head', '.main.nav'].reduce((height, selector) => height + $(selector).scrollHeight, 0);
						scrollY = $('html').scrollTop - ['.main.m_top', '.main.m_head', '.main.nav'].reduce((height, selector) => height + $(selector).scrollHeight, 0);
					};
					let scrollY = 0;
					recScrollY();
					$AEL(window, 'scroll', recScrollY);
					addStyle(CONST.CSS.Review
							 .replaceAll('[BGI]', src)
							 .replaceAll('[S]', textScale.toString())
							 , CONST.ClassName.CSSReview);
					main.scrollTo(0, scrollY); // Cannot record scrollY correctly (always finds window.scrollY = 0 when recording), so stop this feature for a while
					window.removeEventListener('scroll', recScrollY);
					//hookPosition();
					// 2023-06-21 With new CSS specified form[name="frmreview"]{position:relative;}, there's no need to hook UBBEditor.GetPosotion anymore

					// Deletable
					function hookPosition() {
						if (typeof unsafeWindow.UBBEditor !== 'object') {
							hookPosition.wait = hookPosition.wait ? hookPosition.wait : 0;
							if (++hookPosition.wait > 50) {return false;}
							hookPosition.wait % 10 === 0 && DoLog('hookPosition: UBBEditor not loaded, waiting...');
							setTimeout(hookPosition, 500);
							return false;
						}
						unsafeWindow.UBBEditor.GetPosition = function (obj) {
							var r = new Array();
							r.x = obj.offsetLeft;
							r.y = obj.offsetTop;
							while (obj = obj.offsetParent) {
								if (unsafeWindow.$(obj).getStyle('position') == 'absolute' || unsafeWindow.$(obj).getStyle('position') == 'relative') break;
								r.x += obj.offsetLeft;
								r.y += obj.offsetTop;
							}
							r.x -= main.scrollLeft;
							r.y -= main.scrollTop;
							return r;
						}
					}
				}

				function getAPI() {
					return location.pathname.replace(/^\//, '').split('/');
				}
			},
			setting: function setter() {
				const SettingPanel = require('SettingPanel');
				const CONST = {
					Text: {
						CommonBeautify: '通用页面美化',
						NovelBeautify: '阅读页面美化',
						ReviewBeautify: '书评页面美化',
						DefaultBeautify: '默认页面美化图片',
						Enable: '启用',
						BackgroundImage: '背景图片',
						AlertTitle: '页面美化设置',
						InvalidImageUrl: '图片链接格式错误</br>仅仅接受http/https/data链接',
						CenterText: '保持正文文字居中',
						textScale: '文字大小缩放',
						CoverColorLight: '遮罩层颜色',
						CoverOpacityLight: '遮罩层不透明度',
						CoverBlurLight: '遮罩层背景模糊',
						CoverColorDark: '遮罩层颜色(深色模式)',
						CoverOpacityDark: '遮罩层不透明度(深色模式)',
						CoverBlurDark: '遮罩层背景模糊(深色模式)',
					}
				}

				const CM = new ConfigManager(FuncInfo.Config_Ruleset);
				CM.updateAllConfigs();

				const SetPanel = new SettingPanel.easySettings({
					title: CONST.Text.AlertTitle,
					areas: [{
						title: CONST.Text.CommonBeautify,
						items: [{
							text: CONST.Text.Enable,
							path: 'common/enable',
							type: 'boolean'
						}, {
							text: CONST.Text.BackgroundImage,
							path: 'common/image',
							type: ['image', 'string'],
							checker: imageUrlChecker,
						}, {
							text: CONST.Text.CoverColorLight,
							path: 'common/cover/light/color',
							type: ['string']
						}, {
							text: CONST.Text.CoverOpacityLight,
							path: 'common/cover/light/opacity',
							type: ['number'],
							children: $$CrE({
								tagName: 'span',
								props: { innerText: '%' }
							})
						}, {
							text: CONST.Text.CoverBlurLight,
							path: 'common/cover/light/blur',
							type: ['number']
						}, {
							text: CONST.Text.CoverColorDark,
							path: 'common/cover/dark/color',
							type: ['string']
						}, {
							text: CONST.Text.CoverOpacityDark,
							path: 'common/cover/dark/opacity',
							type: ['number'],
							children: $$CrE({
								tagName: 'span',
								props: { innerText: '%' }
							})
						}, {
							text: CONST.Text.CoverBlurDark,
							path: 'common/cover/dark/blur',
							type: ['number']
						}]
					}, {
						title: CONST.Text.NovelBeautify,
						items: [{
							text: CONST.Text.Enable,
							path: 'novel/enable',
							type: 'boolean'
						}, {
							text: CONST.Text.BackgroundImage,
							path: 'novel/image',
							type: ['image', 'string'],
							checker: imageUrlChecker,
						}, {
							text: CONST.Text.CenterText,
							path: 'novel/center',
							type: 'boolean'
						}]
					}, {
						title: CONST.Text.ReviewBeautify,
						items: [{
							text: CONST.Text.Enable,
							path: 'review/enable',
							type: 'boolean'
						}, {
							text: CONST.Text.BackgroundImage,
							path: 'review/image',
							type: ['image', 'string'],
							checker: imageUrlChecker,
						}, {
							text: CONST.Text.textScale,
							path: 'review/textScale',
							type: 'number',
							children: $$CrE({
								tagName: 'span',
								props: { innerText: '%' }
							})
						}]
					}, {
						title: CONST.Text.DefaultBeautify,
						items: [{
							text: CONST.Text.BackgroundImage,
							path: 'image',
							type: ['image', 'string'],
							checker: imageUrlChecker,
						}]
					}]
				}, CM);

				/*
				const Panel = SettingPanel.SettingPanel;
				const _SetPanel = new Panel({
					buttons: 'saver',
					header: CONST.Text.AlertTitle,
					tables: [{
						rows: [{
							blocks: [{
								isHeader: true,
								colSpan: 2,
								innerText: CONST.Text.CommonBeautify
							}]
						},{
							blocks: [{
								innerText: CONST.Text.Enable
							},{
								options: [{
									path: 'common/enable',
									type: 'boolean'
								}]
							}]
						},{
							blocks: [{
								innerText: CONST.Text.BackgroundImage
							},{
								options: [{
									path: 'common/image',
									type: ['image', 'string'],
									checker: imageUrlChecker,
								}]
							}]
						},{
							blocks: [{
								innerText: CONST.Text.CoverColorLight
							},{
								options: [{
									path: 'common/cover/light/color',
									type: ['string']
								}]
							}]
						},{
							blocks: [{
								innerText: CONST.Text.CoverOpacityLight
							},{
								options: [{
									path: 'common/cover/light/opacity',
									type: ['number']
								}],
								children: [(() => {
									const span = $CrE('span');
									span.innerText = '%';
									return span;
								}) ()]
							}]
						},{
							blocks: [{
								innerText: CONST.Text.CoverBlurLight
							},{
								options: [{
									path: 'common/cover/light/blur',
									type: ['number']
								}]
							}]
						},{
							blocks: [{
								innerText: CONST.Text.CoverColorDark
							},{
								options: [{
									path: 'common/cover/dark/color',
									type: ['string']
								}]
							}]
						},{
							blocks: [{
								innerText: CONST.Text.CoverOpacityDark
							},{
								options: [{
									path: 'common/cover/dark/opacity',
									type: ['number']
								}],
								children: [(() => {
									const span = $CrE('span');
									span.innerText = '%';
									return span;
								}) ()]
							}]
						},{
							blocks: [{
								innerText: CONST.Text.CoverBlurDark
							},{
								options: [{
									path: 'common/cover/dark/blur',
									type: ['number']
								}]
							}]
						}]
					},{
						rows: [{
							blocks: [{
								isHeader: true,
								colSpan: 2,
								innerText: CONST.Text.NovelBeautify
							}]
						},{
							blocks: [{
								innerText: CONST.Text.Enable
							},{
								options: [{
									path: 'novel/enable',
									type: 'boolean'
								}]
							}]
						},{
							blocks: [{
								innerText: CONST.Text.BackgroundImage
							},{
								options: [{
									path: 'novel/image',
									type: ['image', 'string'],
									checker: imageUrlChecker,
								}]
							}]
						},{
							blocks: [{
								innerText: CONST.Text.CenterText
							},{
								options: [{
									path: 'novel/center',
									type: 'boolean'
								}]
							}]
						}]
					},{
						rows: [{
							blocks: [{
								isHeader: true,
								colSpan: 2,
								innerText: CONST.Text.ReviewBeautify
							}]
						},{
							blocks: [{
								innerText: CONST.Text.Enable
							},{
								options: [{
									path: 'review/enable',
									type: 'boolean'
								}]
							}]
						},{
							blocks: [{
								innerText: CONST.Text.BackgroundImage
							},{
								options: [{
									path: 'review/image',
									type: ['image', 'string'],
									checker: imageUrlChecker,
								}]
							}]
						},{
							blocks: [{
								innerText: CONST.Text.textScale
							},{
								options: [{
									path: 'review/textScale',
									type: 'number'
								}],
								children: [(() => {
									const span = $CrE('span');
									span.innerText = '%';
									return span;
								}) ()]
							}]
						}]
					},{
						rows: [{
							blocks: [{
								isHeader: true,
								colSpan: 2,
								innerText: CONST.Text.DefaultBeautify
							}]
						},{
							blocks: [{
								innerText: CONST.Text.BackgroundImage
							},{
								options: [{
									path: 'image',
									type: ['image', 'string'],
									checker: imageUrlChecker,
								}]
							}]
						}]
					}]
				}, CM);
				*/

				function imageUrlChecker(e, value) {
					if (value && !value.match(/.+:/)) {
						alertify.alert(CONST.Text.AlertTitle, CONST.Text.InvalidImageUrl);
						return false;
					}
					e.target.value = value || null;
					return true;
				}

				function opacityChecker(e, value) {
					return value >= 0 && value <= 100
				}
			},
		},

		// AdBlock
		{
			name: '广告拦截',
			description: '去除书籍目录页面和章节阅读页面的横幅广告',
			id: 'adblock',
			system: false,
			checker: {
				type: 'regurl',
				value: [
					// Reading page
					/^https?:\/\/www\.wenku8\.(net|cc)\/novel\//,
					/^https?:\/\/www\.wenku8\.(net|cc)\/modules\/article\/reader\.php/,

					// .h_banner
					/^https?:\/\/www\.wenku8\.(net|cc)\/book\/\d+\.htm/,
					/^https?:\/\/www\.wenku8\.(net|cc)\/modules\/article\/reviewshow\.php/,
					/^https?:\/\/www\.wenku8\.(net|cc)\/index\.php/,
				]
			},
			func: function() {
				const utils = require('utils');

				const rulepages = [{
					checker: {
						type: 'regurl',
						value: [
							/^https?:\/\/www\.wenku8\.(net|cc)\/novel\//,
							/^https?:\/\/www\.wenku8\.(net|cc)\/modules\/article\/reader.php/,
						]
					},
					rules: [{
						type: 'selector',
						value: '#adv900'
					}, {
						type: 'selector',
						value: '#adv300'
					}, /*{
						type: 'selector',
						value: '#adbottom'
					},*/ {
						type: 'selector',
						value: '#adv1'
					},/* {
						type: 'selector',
						value: '#adtop'
					}*/]
				}, {
					checker: {
						type: 'regurl',
						value: [
							/^https?:\/\/www\.wenku8\.(net|cc)\/book\/\d+\.htm/,
							/^https?:\/\/www\.wenku8\.(net|cc)\/modules\/article\/reviewshow\.php/,
							/^https?:\/\/www\.wenku8\.(net|cc)\/index\.php/,
						]
					},
					rules: [{
						type: 'selector',
						value: '.main.m_head>.h_banner>*'
					}]
				}];

				for (const page of rulepages) {
					if ((!page.checker || utils.testChecker(page.checker))) {
						for (const rule of page.rules) {
							deal(rule);
						}
					}
				}

				function deal(rule) {
					({
						'selector': function(selector) {
							detectDom({
								selector: selector,
								callback: elm => elm.remove(),
								once: false
							});
							//$(selector) && [...$All(selector)].forEach(elm => elm.remove());
						}
					})[rule.type](rule.value);
				}
			}
		},

		// Reader Enhance
		{
			name: '阅读增强',
			description: '在线阅读页面优化',
			id: 'ReaderEnhance',
			system: false,
			checker: {
				type: 'func',
				value: () => {
					return location.pathname.startsWith('/novel/') && location.pathname.split('/').pop() !== 'index.htm';
				}
			},
			func: function() {
				const SidePanel = require('SidePanel');

				const CONST = {
					Text: {
						PreviousChapter: '上一章',
						NextChapter: '下一章'
					},
					FontSizes: [8,10,12,14,16,18,24,30,36,42,48,54,60,70,80,90,100],
					FontColors: {
						'暗白': '#C8C8C8'
					}
				};

				// Wait till wenku8 script loaded, so detectDom is not enough
				$AEL(window, 'load', e => CSMwork());
				hotkeyJumppage();

				function CSMwork() {
					const CSM = new ConfigSetManager();
					CSM.install();

					sideButtons();
					moreFontColors();
					moreFontSizes();
					fontChanger();
					unsafeWindow.loadSet();

					function sideButtons() {
						SidePanel.insert({
							index: 2,
							faicon: 'fa-solid fa-chevron-left',
							tip: CONST.Text.PreviousChapter,
							onclick: e => $('#foottext>a:nth-of-type(3)').click()
						});
						SidePanel.insert({
							index: 2,
							faicon: 'fa-solid fa-chevron-right',
							tip: CONST.Text.NextChapter,
							onclick: e => $('#foottext>a:nth-of-type(4)').click()
						});
					}

					function moreFontColors() {
						const select = $('#txtcolor');

						// make options
						for (const [name, color] of Object.entries(CONST.FontColors)) {
							const option = $$CrE({
								tagName: 'option',
								props: {
									innerText: name,
									value: color
								}
							});
							select.appendChild(option);
						}
					}

					function moreFontSizes() {
						const select = $('#fonttype');
						const options = [...select.children];

						// make options
						for (const size of CONST.FontSizes) {
							const strSize = `${size}px`;
							const oriOption = options.find(o => o.value === strSize);
							if (oriOption) {
								// modify old option text
								oriOption.innerText = `${strSize}(${oriOption.innerText})`;
							} else {
								// make new option
								const option = $CrE('option');
								option.innerText = option.value = strSize;
								options.push(option);
							}
						}

						// sort options
						const getSizeNum = o => Math.floor(o.value.match(/\d+/)[0]);
						options.sort((o1, o2) => getSizeNum(o1) - getSizeNum(o2));

						// append to select
						options.forEach(o => select.appendChild(o));
					}

					// Provide font changer
					function fontChanger() {
						// Button
						const bcolor = $('#bcolor');
						const txtfont = $CrE('select');
						txtfont.id = 'txtfont';
						txtfont.addEventListener('change', e => CSM.ConfigSets.txtfont.set());
						bcolor.insertAdjacentElement('afterend', txtfont);
						bcolor.insertAdjacentText('afterend', '\t\t\t  字体选择');

						// Provided fonts
						const FONTS = [{"name":"默认","value":"unset"}, {"name":"微软雅黑","value":"Microsoft YaHei"},{"name":"黑体","value":"SimHei"},{"name":"微软正黑体","value":"Microsoft JhengHei"},{"name":"宋体","value":"SimSun"},{"name":"仿宋","value":"FangSong"},{"name":"新宋体","value":"NSimSun"},{"name":"细明体","value":"MingLiU"},{"name":"新细明体","value":"PMingLiU"},{"name":"楷体","value":"KaiTi"},{"name":"标楷体","value":"DFKai-SB"}]
						for (const font of FONTS) {
							const option = $CrE('option');
							option.innerText = font.name;
							option.value = font.value;
							txtfont.appendChild(option);
						}

						// Function
						CSM.ConfigSets.txtfont = {
							type: 'select',
							cookie: 'txtfont',
							default: 'unset',
							element: txtfont,
							effect: value => $('#content').style['font-family'] = value
						};
						CSM.initSet(CSM.ConfigSets.txtfont);

						// Load saved font
						CSM.ConfigSets.txtfont.load();
					}
				}

				function ConfigSetManager() {
					const CSM = this;
					CSM.initSet = initSet;
					CSM.saveSet = saveSet;
					CSM.loadSet = loadSet;
					CSM.install = install;

					// Config sets
					CSM.ConfigSets = {
						'bcolor': {
							type: 'select',
							cookie: 'bcolor',
							default: '#f6f6f6',
							element: unsafeWindow.bcolor,
							effect: value => document.bgColor = value
						},
						'txtcolor': {
							type: 'select',
							cookie: 'txtcolor',
							default: '#000000',
							element: unsafeWindow.txtcolor,
							effect: value => $('#content').style.color = value
						},
						'fonttype': {
							type: 'select',
							cookie: 'fonttype',
							default: '16px',
							element: unsafeWindow.fonttype,
							effect: value => $('#content').style.fontSize = value
						},
						'scrollspeed': {
							type: 'input',
							cookie: 'scrollspeed',
							default: 5,
							element: unsafeWindow.scrollspeed,
							effect: value => unsafeWindow.setSpeed()
						}
					};

					// Init Config sets
					for (const set of Object.values(CSM.ConfigSets)) {
						initSet(set);
					}

					function initSet(set) {
						set.get = () => ({'select': () => set.element.value, 'input': () => set.element.value})[set.type]();
						set.set = () => set.effect(set.get());
						set.save = () => unsafeWindow.setCookies(set.cookie, set.get());
						set.load = () => ({
							'select': () => {
								const savedVal = unsafeWindow.ReadCookies(set.cookie) || set.default;
								const index = [...set.element.options].findIndex(o => o.value === savedVal);
								set.element.selectedIndex = index;
								set.set();
							},
							'input': () => set.element.value = unsafeWindow.ReadCookies(set.cookie) || set.default
						})[set.type]();
					}

					function saveSet() {
						for (const [name, set] of Object.entries(CSM.ConfigSets)) {
							set.save();
						}
					};

					function loadSet() {
						for (const [name, set] of Object.entries(CSM.ConfigSets)) {
							set.load();
						}
					};

					function install() {
						Object.defineProperty(unsafeWindow, 'saveSet', {
							configurable: false,
							enumerable: true,
							value: CSM.saveSet,
							writable: false
						});
						Object.defineProperty(unsafeWindow, 'loadSet', {
							configurable: false,
							enumerable: true,
							value: CSM.loadSet,
							writable: false
						});
					};
				}

				// Make sure hotkey jumppage works
				function hotkeyJumppage() {
					$AEL(document, 'keydown', e => unsafeWindow.jumpPage && unsafeWindow.jumpPage());
				}
			}
		},

		// Bookcase Manager
		{
			name: '书架管理器',
			description: '书架功能增强',
			id: 'BookcaseManager',
			system: false,
			checker: {
				type: 'regurl',
				value: /^https?:\/\/www\.wenku8\.(net|cc)\/modules\/article\/bookcase.php(\?[\s\S]*)?$/,
			},
			func: function() {
				const autovote = FL_getFunction('autovote').enabled ? require('autovote') : null;
				const WenkuBlockGUI = require('WenkuBlockGUI');
				const mousetip = require('mousetip');
				const CommonStyle = require('CommonStyle');
				const utils = require('utils');
				const CONST = {
					Text: {
						RemoveConfirm: '确实要将选中书目移出书架么?',
						RemoveConfirm_Title: '移出书架',
						EnterName: '设置此书架的名称为:',
						EnterName_Title: '设置名称',
						RenameHint: '双击我,为我取个好听的名字吧(ˊᗜˋ*)..',
						Moveto: '移动到 {NAME}',
						Autovote: '自动推书',
						VoteNow: '立即推书',
						TodayVoted: '今日已推书',
						TodayNotVoted: '今日尚未推书'
					},
					DomRes: {
						BookcaseWidth: ['3%', '19%', '9%', '25%', '20%', '9%', '5%', '10%']
					},
					Config_Ruleset: {
						'version-key': 'config-version',
						'ignores': ["LOCAL-CDN"],
						'defaultValues': {
							//'config-key': {},
							bookcaseNames: ['默认书架', '第1组书架', '第2组书架', '第3组书架', '第4组书架', '第5组书架']
						},
						'updaters': {
							/*'config-key': [
								function() {
									// This function contains updater for config['config-key'] from v0 to v1
								},
								function() {
									// This function contains updater for config['config-key'] from v1 to v2
								}
							]*/
						}
					}
				};
				const CM = new ConfigManager(CONST.Config_Ruleset);
				const CONFIG = CM.Config;

				detectDom('#checkform', function() {
					mergeBookcase(function() {
						nameBookcase();
						autovoteGUI();
					});
				});

				function mergeBookcase(onAllMerged) {
					const curform = $('#checkform');
					const cur_classid = getFormClassid(curform);
					const AM = new AsyncManager();
					dealform(curform);
					for (let classid of [0,1,2,3,4,5]) {
						if (classid === cur_classid) {continue;}
						const url = `https://${location.host}/modules/article/bookcase.php?classid=${classid}${utils.getLang() ? '&charset=big5' : ''}`;
						utils.getDocument(url, pageload);
						AM.add();
					}
					AM.onfinish = onAllMerged;
					AM.finishEvent = true;

					function pageload(oDom) {
						const form = $(oDom, '#checkform');
						dealform(form);
						insertform(form);
						AM.finish();
					}

					function dealform(form) {
						form.onsubmit = check_confirm.bind(null, form);
					}

					function insertform(form) {
						const content = $('#content')
						const forms = [...$All(content, '#checkform'), form];
						forms.sort((f1, f2) => getFormClassid(f1) - getFormClassid(f2));
						forms.forEach(f => content.appendChild(f));
					}

					function check_confirm(form, e) {
						if (e.submitter.name !== 'btnsubmit') {return false;}
						const checknum = [...$All(form, 'input[name="checkid[]"]')].filter(input => input.checked).length;
						if (checknum == 0) {
							alertify.alert('请先选择要操作的书目!');
							return false;
						}
						const newclassid = $(form, '#newclassid');
						if (newclassid.value == -1) {
							alertify.confirm(CONST.Text.RemoveConfirm_Title, CONST.Text.RemoveConfirm, form.submit.bind(form), function() {});
							return false;
						} else {
							return true;
						}
					}
				}

				function nameBookcase() {
					const content = $('#content')
					for (const form of $All(content, '#checkform')) {
						deal(form);
					}

					function deal(form) {
						const titlebar = $(form, '.gridtop');
						const classid = getFormClassid(form);

						// Hide select
						$(form, 'select[name="classlist"]').style.display = 'none';

						// Display name
						titlebar.childNodes[0].nodeValue = CONFIG.bookcaseNames[classid];

						// Display name in select#newclassid
						[...$(form, '#newclassid').children].forEach(option => {
							const classid = Math.floor(option.value);
							if (classid >= 0) {
								option.innerText = replaceText(CONST.Text.Moveto, {'{NAME}': CONFIG.bookcaseNames[classid]});
							}
						});

						// Edit
						$AEL(titlebar, 'dblclick', function() {
							titlebar.style.userSelect = 'none';
							alertify.prompt(CONST.Text.EnterName_Title, CONST.Text.EnterName, CONFIG.bookcaseNames[classid], function(e, value) {
								CONFIG.bookcaseNames[classid] = value;
								titlebar.childNodes[0].nodeValue = value;
								titlebar.style.removeProperty('user-select');
							}, function() {
								titlebar.style.removeProperty('user-select');
							});
						});

						// Edit hint
						mousetip.settip(titlebar, CONST.Text.RenameHint);
					}
				}

				function autovoteGUI() {
					if (!autovote) {return false;}

					// Edit header tr
					for (const headTr of $All('tr[align="center"]')) {
						// Add vote column
						const th = $CrE('th');
						th.innerText = CONST.Text.Autovote;
						headTr.appendChild(th);

						// Modify width
						for (let i = 0; i < headTr.children.length; i++) {
							headTr.children[i].setAttribute('width', CONST.DomRes.BookcaseWidth[i]);
						}
					}

					// Edit book tr
					for (const bookTr of $All('#checkform tbody tr:not([align="center"]):not(:last-child)')) {
						const bookInfo = getBookInfo(bookTr);
						const td = $CrE('td');
						const input = $CrE('input');
						input.type = 'number';
						input.inputmode = 'numeric';
						input.style.width = '85%';
						input.setAttribute('form', '');
						input.value = autovote.getVoteNum(bookInfo.aid);
						$AEL(input, 'change', saveAutovotes.bind(input, bookInfo));
						td.style.textAlign = 'center';
						td.appendChild(input);
						bookTr.appendChild(td);
					}

					// add autovote block
					const left = $('#left');
					$(left, '.block:last-child').remove();
					const block = WenkuBlockGUI.makeBookcaseBlock({
						title: CONST.Text.Autovote,
						links: [{
							text: autovote.todayVoted() ? CONST.Text.TodayVoted : CONST.Text.TodayNotVoted,
							classes: CommonStyle.ClassName.Text,
							styles: {'cursor': 'default'}
						},{
							text: CONST.Text.VoteNow,
							listeners: [
								['click', e => autovote.checkRun(true)]
							],
							classes: CommonStyle.ClassName.Button
						}]
					});
					left.appendChild(block.block);

					function saveAutovotes(bookInfo, e) {
						const vote = Math.floor(this.value);
						vote >= 0 && autovote.setAutoVote({...bookInfo, vote});
						this.value = autovote.getVoteNum(bookInfo.aid);
					}

					function getBookInfo(bookTr) {
						return {
							aid: getUrlArgv($(bookTr.children[1], 'a[href*="readbookcase.php"]').href, 'aid'),
							name: $(bookTr.children[1], 'a[href*="readbookcase.php"]').innerText,
							author: $(bookTr, 'a[href*="authorarticle.php"]').innerText,
						}
					}
				}

				function getFormClassid(form) {
					return Math.floor($(form, 'select[name="classlist"]').value);
				}
			}
		},

		// Autovote
		{
			name: '自动推书',
			description: '每日自动按照设定好的次数和书籍完成[推一下]任务',
			id: 'autovote',
			system: false,
			checker: {
				type: 'switch',
				value: true
			},
			func: function() {
				const CONST = {
					URL: {
						Vote: `https://${location.host}/modules/article/uservote.php?id={AID}`
					},
					Text: {
						ConfirmTitle: '手动运行自动推书',
						ConfirmRepeatVote: '今天已运行过自动推书了,是否还要再次运行自动推书?',
						VoteTitle: '自动推书',
						VoteStart: '正在自动推书...',
						VoteFinish: '自动推书完毕,共完成{FINISH}次推书任务, 失败{ERROR}次</br>点击这里显示详情',
						VoteOverflow: '注意:当前设置的自动推书总数量({ALL})超出了账号每日最多推书数量({MAX})',
						VoteSaved: '已保存:每日推荐 {NAME} {NUM}次<br>当前总推荐次数为 {ALL} 次',
						VoteDeleted: '已设置:不再每日推荐 {NAME}<br>当前总推荐次数为 {ALL} 次',
						VoteNumLessThanZero: '不可设置每日推荐次数为负数!',
					},
					Config_Ruleset: {
						'version-key': 'config-version',
						'ignores': ["LOCAL-CDN"],
						'defaultValues': {
							//'config-key': {},
							autovotes: {},
							lastrun: '0000-00-00',
						},
						'updaters': {
							/*'config-key': [
								function() {
									// This function contains updater for config['config-key'] from v0 to v1
								},
								function() {
									// This function contains updater for config['config-key'] from v1 to v2
								}
							]*/
						}
					}
				}
				const utils = require('utils');

				const CM = new ConfigManager(CONST.Config_Ruleset);
				const CONFIG = CM.Config;

				checkRun();
				return {todayVoted, checkRun, setAutoVote, getVoteInfo, getVoteNum};

				function checkRun(userConfirm=false) {
					if (!todayVoted()) {
						run();
					} else if (userConfirm) {
						alertify.confirm(CONST.Text.ConfirmTitle, CONST.Text.ConfirmRepeatVote, run, function() {});
					}
				}

				function todayVoted() {
					return utils.getTime('-', false) === CONFIG.lastrun;
				}

				function run() {
					const autovotes = CM.getConfig('autovotes');
					const all = Object.keys(autovotes).length;
					if (all) {
						const AM = new AsyncManager();
						let finish = 0, error = 0, results = [];
						alertify.notify(CONST.Text.VoteStart);

						for (const voteObj of Object.values(autovotes)) {
							const url = replaceText(CONST.URL.Vote, {'{AID}': voteObj.aid});
							for (let i = 0; i < voteObj.vote; i++) {
								request(url, e => {
									utils.parseDocument(e.response, oDoc => {
										try {
											results.push({
												voteObj,
												title: $(oDoc, '.blocktitle').innerText,
												content: $(oDoc, '.blockcontent>div:first-child').innerText
											});
											finish++; AM.finish();
										} catch(err) {
											error++; AM.finish();
										}
									});
								}, e => { error++; AM.finish(); });
								AM.add();
							}
						}

						AM.onfinish = function() {
							const message = replaceText(CONST.Text.VoteFinish, {'{FINISH}': finish, '{ERROR}': error});
							const box = alertify.success(message, undefined, clicked => {
								if (clicked) {
									const content = results.map(r => `[${r.voteObj.name}]<br>${r.title}<br>${r.content}`).join('<br><br>');
									alertify.alert(CONST.Text.VoteTitle, content);
								}
							});
							CONFIG.lastrun = utils.getTime('-', false);
						}
						AM.finishEvent = true;
					}
				}

				function setAutoVote(voteObj) {
					if (voteObj.vote > 0) {
						CONFIG.autovotes[voteObj.aid] = voteObj;
						notify(replaceText(CONST.Text.VoteSaved, { '{NAME}': voteObj.name, '{NUM}': voteObj.vote , '{ALL}': total() }));
					} else if (voteObj.vote === 0) {
						delete CONFIG.autovotes[voteObj.aid];
						notify(replaceText(CONST.Text.VoteDeleted, { '{NAME}': voteObj.name, '{ALL}': total() }));
					} else {
						alertify.warning(CONST.Text.VoteNumLessThanZero);
					}

					function notify(content) {
						if (setAutoVote.box) {
							setAutoVote.box.setContent(content);
							clearTimeout(setAutoVote.timer);
						} else {
							setAutoVote.box = alertify.message(content, 0);
						}
						setAutoVote.timer = setTimeout(() => {
							setAutoVote.box.dismiss();
							delete setAutoVote.box;
						}, 5000);
					}

					function total() {
						return Object.values(CONFIG.autovotes).reduce((all, obj) => all + obj.vote, 0);
					}
				}

				function getVoteInfo(aid) {
					return CONFIG.autovotes[aid];
				}

				function getVoteNum(aid) {
					const voteInfo = CONFIG.autovotes[aid];
					return voteInfo ? voteInfo.vote : 0;
				}

				function request(url, onload, onerror, retry=3) {
					tryReq(retry);

					function tryReq(retry=3) {
						GM_xmlhttpRequest({
							method: 'GET', url,
							responseType: 'blob',
							timeout: 15 * 1000,
							ontimeout: retry > 0 ? e => tryReq(retry-1) : onerror,
							onerror: retry > 0 ? e => tryReq(retry-1) : onerror,
							onload: onload
						});
					}
				}
			},
			setting: function() {
				window.open(`https://${location.host}/modules/article/bookcase.php`);
			}
		},

		// ReadingPlan
		{
			name: '稍后再读',
			description: '本地稍后再读',
			id: 'ReadingPlan',
			system: false,
			checker: {
				type: 'regurl',
				value: [
					/^https?:\/\/www\.wenku8\.(net|cc)\/index\.php\/?/,
					/^https?:\/\/www\.wenku8\.(net|cc)\/book\/\d+\.htm/,
					/^https?:\/\/www\.wenku8\.(net|cc)\/modules\/article\/articleinfo\.php/
				]
			},
			func: function() {
				const WenkuBlockGUI = require('WenkuBlockGUI');
				const CommonStyle = require('CommonStyle');
				const utils = require('utils');

				const CONST = {
					Text: {
						EmptyHint: '-- 稍后再读的书籍会显示在这里 --',
						IndexTitle: '稍后再读(拖动调整顺序)',
						ConfirmRemoval: '确定要将《<a href="{url}" class="{Class_Button}">{name}</a>》从稍后再读移除吗?',
						ReaditLater: '稍后再读',
						DontReaditLater: '移出稍后再读',
						Added: '已加入稍后再读',
						Removed: '已从稍后再读中移除',
						MoreThanDisplay: '您已经在稍后再读里面放了{N}本书了!</br>尽管稍后再读理论上可以存放无数本书,但是首页最多只能展示前10本哦~'
					},
					CSS: {
						Index: '.plus_rp_item {position: absolute; top: 0; right: 0; font-size: 2em; border: 1px dashed rgba(0,0,0,0); width: 1em; height: 1em;} .plus_rp_item:hover {border: 1px dashed rgba(0,0,0,0.3);}'
					},
					Config_Ruleset: {
						'version-key': 'config-version',
						'ignores': ["LOCAL-CDN"],
						'defaultValues': {
							//'config-key': {},
							books: [/*{
								aid: 1,
								name: '文学少女',
								url: 'https://www.wenku8.net/book/1.htm',
								cover: 'https://img.wenku8.com/image/1/1973/1973s.jpg'
								// array的排列顺序就是用户手动排列的顺序,默认由旧到新(新加书籍放后面)
							}*/]
						},
						'updaters': {
							/*'config-key': [
								function() {
									// This function contains updater for config['config-key'] from v0 to v1
								},
								function() {
									// This function contains updater for config['config-key'] from v1 to v2
								}
							]*/
						}
					}
				};

				const CM = new ConfigManager(CONST.Config_Ruleset);
				const CONFIG = CM.Config;

				const functions = [{
					checker: {
						type: 'regurl',
						value: /^https?:\/\/www\.wenku8\.(net|cc)\/index\.php\/?/
					},
					func: function() {
						// Make block
						const block = WenkuBlockGUI.makeIndexCenterToplistBlock({
							title: CONST.Text.IndexTitle,
							books: CM.getConfig('books').filter((book, i) => i < 10)
						});

						// Top-right remove button
						for (const item of block.items) {
							item.b_container.style.position = 'relative';
							item.b_container.appendChild($$CrE({
								tagName: 'i',
								classes: ['fa-solid', 'fa-xmark', CommonStyle.ClassName.Button, 'plus_rp_item'],
								listeners: [['click', function(e) {
									alertify.confirm(
										replaceText(CONST.Text.ConfirmRemoval, {
											'{url}': item.url, '{name}': item.name,
											'Class_Button': CommonStyle.ClassName.Button
										}),
										function onok() {
											CONFIG.books.splice(CONFIG.books.findIndex(b => b.aid === item.aid), 1);
											block.items.splice(block.items.findIndex(b => b.aid === item.aid), 1);
											item.b_container.remove();
											alertify.success(CONST.Text.Removed);
										}
									)
								}]]
							}));
						}

						// Drag-drop to move books
						const sortable = new Sortable.default(block.blockcontent.children[0], {
							draggable: 'div'
						});
						sortable.on('sortable:sorted', function(e) {
							const div = e.dragEvent.data.originalSource;
							const books = moveItem(CM.getConfig('books'), e.oldIndex, e.newIndex);
							CONFIG.books = books;
							/* Wrong coding:
							The following code CANNOT WORK! Because CONFIG.books[index] saves the path that
							contains index, and always gives the current value of this path in GM_storage.
							As we know, moveItem removes arr[from] and then inserts it to arr[to], and
							as moveItem inserts(using arr.splice) CONFIG.books[oldIndex] into newIndex,
							the proxy for CONFIG.books[oldIndex]'s path is currently pointing to
							the new value at CONFIG.books[oldIndex], which is the next book of the
							dragging one! Tempermonkey will look into its(the next book's) props, and save
							them into disk, at CONFIG.books[newIndex], which should be storing the dragging
							book's properties.
							To avoid this from happenning, we get the pure object using CM.getConfig, sort it
							using moveItem, and then save it to GM_storage manually, just like code above.

							The wrong code is written below:
							moveItem(CONFIG.books, e.oldIndex, e.newIndex);
							*/
						});

						// Display hint while no books in reading plan
						if (!CONFIG.books.length) {
							const hint = $CrE('div');
							hint.innerText = CONST.Text.EmptyHint;
							hint.style.color = 'grey';
							hint.style.fontSize = '1.2em';
							block.blockcontent.style.display = 'flex';
							block.blockcontent.style.alignItems = 'center';
							block.blockcontent.style.justifyContent = 'center';
							block.blockcontent.appendChild(hint);
						}

						// Make container
						const container = $CrE('div');
						container.classList.add('main');
						container.appendChild(block.block);

						// Append to dom
						detectDom('#left', left => {
							document.body.insertBefore(container, left.parentElement.nextElementSibling);
						});

						// CSS
						detectDom('head', head => {
							addStyle(CONST.CSS.Index, 'plus-readingplan-index-css');
						})
					}
				}, {
					checker: {
						type: 'regurl',
						value: [
							/^https?:\/\/www\.wenku8\.(net|cc)\/book\/\d+\.htm/,
							/^https?:\/\/www\.wenku8\.(net|cc)\/modules\/article\/articleinfo\.php/
						]
					},
					detectDom: '.main.m_foot',
					func: function() {
						const aid = getUrlArgv($('a[href*="/modules/article/uservote.php?"]').href, 'id');
						const btn = makeBtn();
						installBtn(btn);
						refreshBtn(btn);

						function makeBtn() {
							const btn = $CrE('span');
							btn.classList.add(CommonStyle.ClassName.Button);
							$AEL(btn, 'click', toggle);
							return btn;
						}

						function toggle() {
							const book = CONFIG.books.findIndex(book => book.aid == aid);
							if (book >= 0) {
								CONFIG.books.splice(book, 1);
								alertify.success(CONST.Text.Removed);
							} else {
								CONFIG.books.push({
									aid,
									name: $('#content table table b').innerText,
									url: location.href,
									cover: $('#content>div:first-child>table:nth-of-type(2) img').src
								});
								CONFIG.books.length > 10 && alertify.notify(replaceText(CONST.Text.MoreThanDisplay, {'{N}': CONFIG.books.length.toString()}));
								alertify.success(CONST.Text.Added);
							}
							refreshBtn(btn);
						}

						function refreshBtn(btn) {
							const text = CONFIG.books.some(book => book.aid == aid) ? CONST.Text.DontReaditLater : CONST.Text.ReaditLater;
							btn.innerText = text;
						}

						function installBtn(btn) {
							$('#content table table b').insertAdjacentElement('afterend', btn);
							btn.insertAdjacentText('beforebegin', '[');
							btn.insertAdjacentText('afterend', ']');
							return btn;
						}
					}
				}];

				return utils.loadFuncs(functions);

				function moveItem(arr, from, to) {
					const item = arr.splice(from, 1)[0];
					arr.splice(to, 0, item);
					return arr;
				}
			}
		},

		// Apipage Enhance
		{
			name: 'api页面增强',
			description: 'api页面增强',
			id: 'ApiEnhance',
			system: false,
			checker: {
				type: 'starturl',
				value: [
					`https://${location.host}/modules/article/addbookcase.php`,
					`https://${location.host}/modules/article/packshow.php`
				]
			},
			func: function() {
				const utils = require('utils');
				const CONST = {
					CSS: {
						Beautify: 'body>div:not([class*="ajs-"],[class*="alertify-"]) {display: flex; align-items: center; justify-content: center;}'
					}
				};

				document.readyState !== 'loading' ? work() : $AEL(document, 'DOMContentLoaded', work);

				function work() {
					if ($All('.block').length > 1) {
						return {};
					}
					addStyle(CONST.CSS.Beautify, 'plus-api-beautify');

					const functions = [{
						checker: {
							type: 'regurl',
							value: /\/addbookcase\.php/
						},
						detectDom: '.blocknote',
						func: function() {
							// Append link to bookcase page
							addBottomButton({
								href: `https://${location.host}/modules/article/bookcase.php`,
								innerHTML: '管理书架'
							});
						}
					}];

					return utils.loadFuncs(functions);
				}

				// Add a bottom-styled botton into bottom line, to the first place
				function addBottomButton(details) {
					const aClose = $('a[href="javascript:window.close()"]');
					const bottom = aClose.parentElement;
					const a = $CrE('a');
					const t1 = document.createTextNode('[');
					const t2 = document.createTextNode(']');
					const blank = $CrE('span');
					blank.innerHTML = ' ';
					blank.style.width = '0.5em';
					a.href = details.href;
					a.innerHTML = details.innerHTML;
					a.onclick = details.onclick;
					[blank, t2, a, t1].forEach((elm) => {bottom.insertBefore(elm, bottom.childNodes[0]);});
				}
			}
		},

		// Meta copy
		{
			name: '书籍信息复制',
			id: 'MetaCopy',
			description: '复制书籍信息:文库、作者、状态、最后更新、字数',
			checker: {
				type: 'regurl',
				value: [
					/^https?:\/\/www\.wenku8\.(net|cc)\/book\/\d+\.htm([\?#][\s\S]*)?$/,
					/^https?:\/\/www\.wenku8\.(net|cc)\/modules\/article\/articleinfo\.php([\?#][\s\S]*)?$/,
				]
			},
			func: function() {
				const CONST = {
					Text: {
						Copy: '[复制]',
						Copied: '已复制',
					}
				};

				const CommonStyle = require('CommonStyle');
				const mousetip = require('mousetip');
				const utils = require('utils');

				detectDom('#content>div:first-of-type>table+div:nth-child(2)', e => work());
				function work() {
					const metas = $All('#content>div:first-child>table:first-child>tbody>tr>td:not([colspan])');
					for (const meta of metas) {
					const data = meta.innerText.split(':')[1];
					const copy = $$CrE({
						tagName: 'span',
						props: {
							innerText: CONST.Text.Copy
						},
						styles: {
							'margin-left': '0.5em'
						},
						classes: CommonStyle.ClassName.Button,
						listeners: [['click', () => {
							copyText(data);
							mousetip.showtip(CONST.Text.Copied);
						}]]
					});
					mousetip.settip(copy, data);
					meta.appendChild(copy);
				}
				}

				// Copy text to clipboard (needs to be called in an user event)
				function copyText(text) {
					// Create a new textarea for copying
					const newInput = $CrE('textarea');
					document.body.appendChild(newInput);
					newInput.value = text;
					newInput.select();
					document.execCommand('copy');
					document.body.removeChild(newInput);
				}
			}
		},

		// Tag button
		{
			name: '书籍tag跳转',
			description: '书籍页面点击书籍tag跳转到对应的tag页面',
			id: 'TagButton',
			STOP: false,
			checker: {
				type: 'regurl',
				value: [
					/^https?:\/\/www\.wenku8\.(net|cc)\/book\/\d+\.htm([\?#][\s\S]*)?$/,
					/^https?:\/\/www\.wenku8\.(net|cc)\/modules\/article\/articleinfo\.php([\?#][\s\S]*)?$/,
				]
			},
			CONST: {
				URL: {
					Tags: `https://${location.host}/modules/article/tags.php?t={tags}`
				}
			},
			func: function() {
				const CONST = FuncInfo.CONST;
				const CommonStyle = require('CommonStyle');
				const mousetip = require('mousetip');
				const utils = require('utils');
				detectDom('#content>div:first-of-type>table+div:nth-child(5)', e => work());

				function work() {
					const b = $('td[width="48%"]>.hottext:first-child>b');
					const text = b.innerText.split(/[︰:]/)[1];
					b.innerText = b.innerText.replace(text, '');

					text.split(' ').forEach((tag, i) => {
						i > 0 && b.insertAdjacentText('beforeend', ' ');
						b.appendChild($$CrE({
							tagName: 'a',
							props: {
								innerText: tag,
								href: replaceText(CONST.URL.Tags, { '{tags}': $URL.encode(tag) }),
								target: '_blank'
							},
							classes: [CommonStyle.ClassName.Button]
						}));
					});
				}
			}
		},

		// Single chapter download
		{
			name: '单章节下载',
			description: '单章节下载',
			id: 'ChapterDownload',
			checker: {
				type: 'func',
				value: () => {
					return (location.pathname.startsWith('/novel/') || location.pathname.match(/\/modules\/article\/reader.php/)) && !location.href.includes('index.htm');
				}
			},
			func: function() {
				const CommonStyle = require('CommonStyle');
				const SidePanel = require('SidePanel');
				const utils = require('utils');

				const CONST = {
					Text: {
						DownloadChapter: '下载本章',
						DownloadStatus: '下载中({Fin}/{All})',
						AllImagesDownloaded: '全部图片下载完毕',
						ImageDownloadDone: '图片下载已完成</br>成功:{Suc} 张</br>失败:{Fail} 张</br>下载失败的图片分别是第 {Fail_No} 张</br>所有下载失败的图片均已经过3次自动重试',
						Downloaded: '已下载',
						ImageDownloadError: '第 {N} 张图片下载失败',
					}
				};

				detectDom('#footlink', e => work());
				function work() {
					const headlink = $('#headlink');
					const linkleft = $('#linkleft');
					const linkright = $('#linkright');
					headlink.style.position = 'relative';
					linkright.style.position = 'absolute';
					linkright.style.right = '0';
					linkright.style.width = 'unset';

					const dlBtn = $$CrE({
						tagName: 'span',
						props: {
							innerText: CONST.Text.DownloadChapter,
							id: 'plus-chapter-dl'
						},
						classes: CommonStyle.ClassName.Button,
						listeners: [['click', download]]
					});
					linkright.lastElementChild.insertAdjacentHTML('afterend', ' | ');
					linkright.insertBefore(dlBtn, linkright.lastChild);

					SidePanel.insert({
					index: 2,
					tip: CONST.Text.DownloadChapter,
					faicon: 'fa-solid fa-download',
					onclick: download
				});
				}

				function download() {
					dText();
					dImg();
				}

				function dText() {
					const title = $('#title').innerText;
					const text = $('#content').innerText;
					const filename = title + '.txt';
					text.trim().length > 0 && !$('#title').innerText.includes(text.trim()) && utils.downloadText(text, filename);
				}

				function dImg() {
					if (!$('.divimage')) {return false;}

					const imgs = [...$All('.divimage img')];
					const title = $('#title').innerText;
					const AM = new AsyncManager();
					let finished = 0, failed = 0, fails = [];
					imgs.forEach((img, i) => {
						AM.add();
						dl(img, i);
					});
					AM.onfinish = onfinish;
					AM.finishEvent = true;

					function dl(img, index, r=3) {
						const onfail = (err, reason) => r ? dl(img, index, r-1) : fail(err, reason);
						const ext = (() => {
							const str = img.src.split('.').pop();
							const exts = ['jpg', 'jpeg', 'png', 'gif', 'webp'];
							return str && exts.includes(str.toLowerCase()) ? str : 'jpg';
						}) ();
						dl_GM({
							url: img.src,
							name: `${title}_${(index+1).toString().padStart('0', imgs.length.toString().length)}.${ext}`,
							onload: function() {
								$('#plus-chapter-dl').innerText = replaceText(CONST.Text.DownloadStatus, {
									'{Fin}': (++finished).toString(),
									'{All}': imgs.length
								});
								AM.finish();
							},
							onerror: err => onfail(err, 'onerror'),
							ontimeout: err => onfail(err, 'ontimeout'),
						});

						function fail(err, reason) {
							failed++;
							fails.push({img, index, err, reason});
							DoLog(LogLevel.Error, [`Image download error: ${reason}`, err]);
							alertify.error(replaceText(CONST.Text.ImageDownloadError, {
								'{N}': (index + 1).toString()
							}));
							AM.finish();
						}
					}

					function onfinish() {
						const all = imgs.length;
						if (finished === all) {
							alertify.success(CONST.Text.AllImagesDownloaded);
							setTimeout(e => $('#plus-chapter-dl').innerText = CONST.Text.Downloaded, 3000);
						} else if (finished + failed === all) {
							alertify.notify(replaceText(CONST.Text.ImageDownloadDone, {
								'{Suc}': finished.toString(),
								'{Fail}': failed.toString(),
								'{Fail_No}': fails.map(f => (f.index+1).toString()).join(',')
							}));
						}
					}
				}
			}
		},

		// Darkmode
		{
			name: '深色模式',
			description: '可单独开关、可跟随系统配置的深色模式',
			id: 'darkmode',
			CONST: {
				Text: {
					Toggle: '切换浅色/深色模式',
					AlertTitle: `深色模式设置`,
					autoMatchOsTheme: '深色模式自动跟随系统配置'
				},
				PageCSS: [
					// Sidepanel
					{
						id: 'sidepanel',
						checker: {
							type: 'regurl',
							value: /^https?:\/\/www\.wenku8\.(net|cc)\//
						},
						css: '.plus-darkmode .sidepanel-button {background-color: #333333;color: #6f9ff1;fill: #6f9ff1;}.plus-darkmode .sidepanel-button:hover {background-color: #6f9ff1;color: #333333;fill: #333333;}'
					},

					// Mouse tip
					{
						id: 'mousetip',
						checker: {
							type: 'regurl',
							value: /^https?:\/\/www\.wenku8\.(net|cc)\//
						},
						css: '.plus-darkmode #tips {background-color: #333333;color: white;border: 1px solid #0d548b;}'
					},

					// Common style darkend
					{
						id: 'commonstyle',
						checker: {
							type: 'regurl',
							value: /^https?:\/\/www\.wenku8\.(net|cc)\//
						},
						css: '.plus-darkmode .plus_text {color: #6f9ff1 !important;}.plus-darkmode .plus_button.plus_disabled {color: #888888;}'
					},

					// .block
					{
						id: 'block',
						checker: {
							type: 'regurl',
							value: /^https?:\/\/www\.wenku8\.(net|cc)\//
						},
						css: '.plus-darkmode :is(#left, #right, *) .blockcontent {background-color: #222222;}.plus-darkmode :is(#left, #right, *) .blocknote {background-color: #282828;}.plus-darkmode :is(#left, #right, #centers, *) :is(.blocktitle, .blocktitle *, .ultop li) {color: #6f9ff1;}.plus-darkmode :is(#left, #right, #centers, *) .blocktitle>:is(.txt, .txtr) {background-color: #383838;line-height: 27px;padding-top: 0;}.plus-darkmode .block {border: 1px solid #0d548b;}.plus-darkmode :is(.blocktitle, .blockcontent, .blocknote) {border: none;}.plus-darkmode .block :is(.ultop li, .ultops li) {border-bottom: 1px dashed #0d548b;}'
					},

					// header and footer
					{
						id: 'headfoot',
						checker: {
							type: 'regurl',
							value: /^https?:\/\/www\.wenku8\.(net|cc)\//
						},
						css: '.plus-darkmode :is(.main.m_top, .nav, .navinner, .navlist, .nav li, :is(#left, #right, #centers, *) :is(.blocktitle, .blocktitle *)) {background: #333333;}.plus-darkmode :is(.nav a.current, .nav a:hover, .nav a:active) {background: #444444;}.plus-darkmode .m_foot {border-top: 1px dashed #0d548b;border-bottom: 1px dashed #0d548b;}'
					},

					// elements (input textarea .button scrollbar, etc)
					{
						id: 'element',
						checker: {
							type: 'regurl',
							value: /^https?:\/\/www\.wenku8\.(net|cc)\//
						},
						css: '.plus-darkmode {color: #C8C8C8;background-color: #222222;}.plus-darkmode :is(.even, .odd) {background-color: #222222;}.plus-darkmode table.grid td {background-color: #222222 !important;}.plus-darkmode :is(input:not([type]:not([type="text"], [type="number"], [type="file"], [type="password"])), textarea, .plus_list_item, button) {background-color: #333333;color: #DDDDDD;}.plus-darkmode :is(.button, input[type="button"]) {color: #C8C8C8;background-color: #333333;}.plus-darkmode select {color: #AAAAAA;background-color: #333333;}.plus-darkmode :is(.hottext, a.hottext) {color: #f36d55;}.plus-darkmode :is(.button, select, textarea, input:not(.plus_list_item>input, .UBB_ColorList input), .plus_list_item):not(:disabled) {border: 2px solid #0d548b;}.plus-darkmode :is(input, textarea, button):disabled {border: 2px solid #444444;}.plus-darkmode a {color: #AAAAAA;}.plus-darkmode a:hover {color: #4a8dff;}.plus-darkmode a:is(.ultop li a, .poptext, a.poptext, .ultops li a) {color: #f36d55;}.plus-darkmode :is(table.grid caption, .gridtop, table.grid th, .head) {border: 1px solid #0d548b;background: #333333;color: #6f9ff1;}.plus-darkmode :is(table.grid, table.grid td) {border: 1px solid #0d548b;}.plus-darkmode input[type="checkbox"]::after {background-color: #333333;}.plus-darkmode :is(.pagelink, .pagelink a:hover) {background-color: #333333;color: #6f9ff1;}.plus-darkmode .pagelink strong {background-color: #444444;}.plus-darkmode .pagelink em {border-right: 1px solid #0d548b;}.plus-darkmode .pagelink kbd {border-left: 1px solid #0d548b;}.plus-darkmode .pagelink {border: 1px solid #0d548b;}:is(.plus-darkmode, .plus-darkmode *):not(#NotAElement>#NotAElement>#NotAElement>#NotAElement) {scrollbar-color: #444444 #333333;}:is(.plus-darkmode, .plus-darkmode *):not(#NotAElement>#NotAElement>#NotAElement>#NotAElement):hover {scrollbar-color: #484848 #333333;}:is(.plus-darkmode, .plus-darkmode *):not(#NotAElement>#NotAElement>#NotAElement>#NotAElement)::-webkit-scrollbar {background-color: #333333;}:is(.plus-darkmode, .plus-darkmode *):not(#NotAElement>#NotAElement>#NotAElement>#NotAElement)::-webkit-scrollbar-corner {background-color: #333333;}:is(.plus-darkmode, .plus-darkmode *):not(#NotAElement>#NotAElement>#NotAElement>#NotAElement)::-webkit-scrollbar-thumb, .plus-darkmode *:not(#NotAElement>#NotAElement>#NotAElement>#NotAElement)::-webkit-scrollbar-button {background-color: #444444;}:is(.plus-darkmode, .plus-darkmode *):not(#NotAElement>#NotAElement>#NotAElement>#NotAElement)::-webkit-scrollbar-thumb:hover, .plus-darkmode *:not(#NotAElement>#NotAElement>#NotAElement>#NotAElement)::-webkit-scrollbar-button:hover {background-color: #484848;}'
					},

					// dialog
					{
						id: 'dialog',
						checker: {
							type: 'regurl',
							value: /^https?:\/\/www\.wenku8\.(net|cc)\//
						},
						css: '.plus-darkmode #dialog {color: #C8C8C8;background-color: #222222;border: 5px solid #0d548b;}.plus-darkmode #dialog a[onclick="closeDialog()"] {border: 1px solid #0d548b !important;outline: thin solid #0d548b !important;}'
					},

					// alertify
					{
						id: 'alertify',
						checker: {
							type: 'regurl',
							value: /^https?:\/\/www\.wenku8\.(net|cc)\//
						},
						css: '.plus-darkmode .alertify :is(.ajs-dialog, .ajs-header, .ajs-footer) {background-color: #222222;}.plus-darkmode .alertify :is(.ajs-header, .ajs-body) {color: #C8C8C8;}.plus-darkmode .alertify .ajs-header {border-bottom: 1px solid #333333;}.plus-darkmode .alertify .ajs-footer {border-top: 1px solid #333333;}.plus-darkmode .alertify .ajs-footer .ajs-buttons .ajs-button.ajs-cancel {color: #C8C8C8;}.plus-darkmode .plus_func_block {border-top: 1px solid #333333;}.plus-darkmode .plus_func_sysalert {background-color: #40331f;color: #cd8c32;}.plus-darkmode .alertify-notifier .ajs-message {border: 1px solid #0d548b;}.plus-darkmode .ajs-message{background-color: #222222;color: #C8C8C8;}.plus-darkmode .ajs-message.ajs-success {background-color: #1f6a01;}.plus-darkmode .ajs-message.ajs-warning {background-color: #5f4e05;}.plus-darkmode .ajs-message.ajs-error {background-color: #730808;}'
					},

					// replyarea
					{
						id: 'replyarea',
						checker: {
							type: 'starturl',
							value: [
								// Page: reviews list
								`https://${location.host}/modules/article/reviews.php`,

								// Page: review
								`https://${location.host}/modules/article/reviewshow.php`,

								// Page: review edit
								`https://${location.host}/modules/article/reviewedit.php`,

								// Page: book
								`https://${location.host}/book`,
								`https://${location.host}/modules/article/articleinfo.php`,
							]
						},
						css: '.plus-darkmode form[name="frmreview"] caption {background: #333333;color: #6f9ff1;border: 1px solid #0d548b;}'
					},

					// index.php
					{
						id: 'index',
						checker: {
							type: 'regurl',
							value: /^https?:\/\/www\.wenku8\.(net|cc)\/index\.php([\?#][\s\S]*)?$/
						},
						css: 'body.plus-darkmode, .plus-darkmode :is(:is(#left, #right, #centers) .blockcontent, .blockcontent, .odd, .even) {background-color: #222222;color: #C8C8C8;}.plus-darkmode :is(#left, #right, #centers, *) :is(.blocktitle, .blocktitle *) a {color: #AAAAAA;}a[href^="http://tieba.baidu.com"] {color: #4a8dff !important;}',
					},

					// Book
					{
						id: 'book',
						checker: {
							type: 'regurl',
							value: [
								/^https?:\/\/www\.wenku8\.(net|cc)\/book\/\d+\.htm([\?#][\s\S]*)?$/,
								/^https?:\/\/www\.wenku8\.(net|cc)\/modules\/article\/articleinfo\.php([\?#][\s\S]*)?$/,
							]
						},
						css: 'body.plus-darkmode, .plus-darkmode :is(:is(#left, #right, #centers) .blockcontent, .blockcontent, .odd, .even) {background-color: #222222;color: #C8C8C8;}.plus-darkmode table.grid td {background-color: #222222 !important;}.plus-darkmode table.grid:not(form table) tr:first-of-type>td:nth-of-type(2n+1) {background-color: #333333 !important;}.plus-darkmode table.grid:not(form table, #content .main>table:first-of-type) tr:first-of-type>td:first-of-type {color: #6f9ff1;}.plus-darkmode fieldset {border: 2px solid #0d548b;}.plus-darkmode :is(table.grid, table.grid td, table.grid caption, .gridtop) {border: 1px solid #0d548b;}'
					},

					// Book index
					{
						id: 'bookindex',
						checker: {
							type: 'regurl',
							value: /^https?:\/\/www\.wenku8\.(net|cc)\/novel\/\d+\/\d+\/index\.htm([\?#][\s\S]*)?$/
						},
						css: 'body.plus-darkmode, .plus-darkmode :is(.css, .vcss, .ccss) {background-color: #222222;color: #C8C8C8;}.plus-darkmode #headlink {border-bottom: 1px solid #0d548b;border-top: 1px solid #0d548b;}.plus-darkmode :is(.css, .vcss, .ccss) {border: 1px solid #0d548b;border-collapse: collapse;}'
					},

					// Novel
					{
						id: 'novel',
						checker: {
							type: 'func',
							value: () => {
								return location.pathname.startsWith('/novel/') && location.pathname.split('/').pop() !== 'index.htm';
							}
						},
						css: '.plus-darkmode a {color: #4a8dff;}'
					},

					// Reviewshow
					{
						id: 'reviewshow',
						checker: {
							type: 'regurl',
							value: /^https?:\/\/www\.wenku8\.(net|cc)\/modules\/article\/reviewshow\.php\/?/
						},
						css: 'body.plus-darkmode, .plus-darkmode :is(:not(*)) {background-color: #222222;color: #C8C8C8;}.plus-darkmode table.grid td {background-color: #222222;}.plus-darkmode :is(#content table.grid hr, #content>table:nth-of-type(2) th, #pagelink) {border: 1px solid #0d548b;}.plus-darkmode :is(.jieqiQuote, .jieqiCode, .jieqiNote) {background-color: #282828;color: #6f9ff1;border: 1px solid #0d548b;}'
					},

					// Beautifier
					{
						id: 'beautifier',
						checker: {
							type: 'regurl',
							value: /^https?:\/\/www\.wenku8\.(net|cc)\//
						},
						css: ''
					},

					// tippy
					{
						id: 'tippy',
						checker: {
							type: 'regurl',
							value: /^https?:\/\/www\.wenku8\.(net|cc)\//
						},
						css: '.plus-darkmode .tippy-box[data-theme~="wenku_tip"] {background-color: #282828;color: #C3C3C3;border: 1px solid #0d548b;}.plus-darkmode .tippy-box[data-theme~="wenku_tip"][data-placement^="top"]>.tippy-arrow::before {border-top-color: #0d548b;}.plus-darkmode .tippy-box[data-theme~="wenku_tip"][data-placement^="left"]>.tippy-arrow::before {border-left-color: #0d548b;}.plus-darkmode .tippy-box[data-theme~="wenku_tip"][data-placement^="right"]>.tippy-arrow::before {border-right-color: #0d548b;}.plus-darkmode .tippy-box[data-theme~="wenku_tip"][data-placement^="bottom"]>.tippy-arrow::before {border-bottom-color: #0d548b;}'
					},

					// frmreview
					{
						id: 'frmreview',
						checker: {
							type: 'starturl',
							value: [
								// Page: reviews list
								`https://${location.host}/modules/article/reviews.php`,

								// Page: review
								`https://${location.host}/modules/article/reviewshow.php`,

								// Page: review edit
								`https://${location.host}/modules/article/reviewedit.php`,

								// Page: book
								`https://${location.host}/book`,
								`https://${location.host}/modules/article/articleinfo.php`,
							]
						},
						css: '.plus-darkmode .UBB_FontSizeList li {border: 1px solid #0d548b;}.plus-darkmode .UBB_ColorList :is(table, table td) {border: 1px solid #0d548b;}.plus-darkmode .UBB_ColorList {background-color: #222222;}'
					},

					/* Template
					{
						id: '',
						checker: {
							type: 'regurl',
							value: /^https?:\/\/www\.wenku8\.(net|cc)\//
						},
						css: ''
					},
					*/
				],
				Config_Ruleset: {
					'version-key': 'config-version',
					'ignores': ["LOCAL-CDN"],
					'defaultValues': {
						darkmode: false,
						autoMatchOsTheme: false,
					},
					'updaters': {
						/*'config-key': [
							function() {
								// This function contains updater for config['config-key'] from v0 to v1
							},
							function() {
								// This function contains updater for config['config-key'] from v1 to v2
							}
						]*/
					}
				},
			},
			STOP: false,
			func: function() {
				// Stop running in image, code, or pure text papes
				if (/\.(jpe?g|png|webp|gif|bmp|txt|js|css)/.test(location.pathname)) {
					return false;
				}

				const SPanel = require('SidePanel');
				const utils = require('utils');
				const CONST = FuncInfo.CONST;
				const CM = new ConfigManager(CONST.Config_Ruleset);
				const CONFIG = CM.Config;

				/* Darkmode.js doing pretty bad with background image beautifier on, so stop using it
				// Uses darkmode.js
				// See https://github.com/sandoche/Darkmode.js
				const options = {
					saveInCookies: false, // default: true,
					label: '', // default: ''
					autoMatchOsTheme: false // default: true // This doesn't work in my computer, use my own code instead
				}

				detectDom('body', function() {
					const darkmode = new Darkmode(options);
					const button = SPanel.insert({
						index: 2,
						faicon: 'fa-solid fa-circle-half-stroke',
						tip: CONST.Text.Toggle,
						onclick: e => darkmode.toggle()
					});
					if (CONFIG.autoMatchOsTheme) {
						const match = window.matchMedia('(prefers-color-scheme: dark)');
						$AEL(match, 'change', followOsTheme);
						followOsTheme(match);
					}

					function followOsTheme(e) {
						e.matches && !darkmode.isActivated() && darkmode.toggle();
						return e.matches;
					}
				});
				*/

				// Append css
				CONST.PageCSS.filter(cssObj => utils.testChecker(cssObj.checker))
					.forEach(cssObj => addStyle(cssObj.css, `plus-darkmode-${cssObj.id}`));

				// Side Panel toggle button
				const button = SPanel.insert({
					index: 2,
					faicon: 'fa-solid fa-circle-half-stroke',
					tip: CONST.Text.Toggle,
					onclick: e => toggle()
				});

				// Follow os theme
				detectDom('body', function() {
					if (CONFIG.autoMatchOsTheme) {
						const match = window.matchMedia('(prefers-color-scheme: dark)');
						$AEL(match, 'change', followOsTheme);
						followOsTheme(match);
					} else {
						CONFIG.darkmode !== isActivated() && toggle();
					}
				});

				function followOsTheme(mediaMatch) {
					const isOsDarkmode = mediaMatch.matches;
					isOsDarkmode !== isActivated() && toggle();
					return isOsDarkmode;
				}

				function toggle() {
					document.body.classList[isActivated() ? 'remove' : 'add']('plus-darkmode');
					CONFIG.darkmode = isActivated();
				}

				function isActivated() {
					return document.body.classList.contains('plus-darkmode');
				}
			},
			setting: function setter() {
				const SettingPanel = require('SettingPanel');
				const CONST = FuncInfo.CONST;

				const CM = new ConfigManager(CONST.Config_Ruleset);
				CM.updateAllConfigs();

				SettingPanel.easySettings({
					title: CONST.Text.AlertTitle,
					items: [{
						text: CONST.Text.autoMatchOsTheme,
						path: 'autoMatchOsTheme',
						type: 'boolean'
					}]
				}, CM);
			}
		},

		// ReviewEnchance
		{
			name: '书评吐槽增强',
			description: '书评吐槽页面提供引用回复、页面内编辑回帖内容、页面自动刷新等等增强功能',
			id: 'ReviewEnhance',
			STOP: false,
			checker: {
				type: 'regurl',
				value: [
					/^https?:\/\/www\.wenku8\.(net|cc)\/modules\/article\/reviewshow\.php\/?/,
				]
			},
			CONST: {
				Text: {
					Quote: '引用',
					QuoteNum: ['或者,', '仅引用序号'],
					AutoRefresh_Off: '自动刷新',
					AutoRefresh: '自动刷新({t}秒)',
					FloorContentModified: '(内容已更新)',
					ReplyErrorTitle: '回复发生错误',
					DownloadFormat: '保存格式:',
					FormatTip: '纯文本:</br>目前所提供的几种格式中,一般而言最适合阅读的格式。但是,这种格式仅能保存文本内容。</br></br>BBCODE格式:</br>即文库评论的代码格式,相当于引用楼层时自动填入回复框的内容。保存为此格式可以一定程度上在保持可阅读性的同时,保留排版及多媒体信息。</br></br>JSON格式:</br>一种方便程序读取的格式,包含所有脚本能够获取到的文本、排版和多媒体信息。</br>需要注意的是,这种格式一般来讲不适合直接阅读。',
					DownloadPost: '下载本贴(共{N}页)',
					DownloadingPost: '正在下载({finished}/{all})',
					DownloadFailed: '下载失败,请重试',
					DownloadSuccess: '下载完毕',
					DownloadFormats: {
						text: {
							name: '纯文本',
							ext: 'txt'
						},
						bbcode: {
							name: 'BBCODE格式',
							ext: 'txt'
						},
						json: {
							name: 'JSON',
							ext: 'json'
						}
					},
					DownloadTemplate: {
						Template: `轻小说文库 书评吐槽 [ID: {Review ID}]\n主题:{Review Title}\n保存自:{Review URL}\n保存时间:{Download Time}\n保存格式:{Format}\nBy ${GM_info.script.name} Ver. ${GM_info.script.version} author ${GM_info.script.author}\n\n\n\n\n\n{Floor Content}`,
						FloorTemplate: `[{FloorNum}#] [{Username} {UserID}]{Floor Title} [{Time}]\n{Floor Content}`,
						FloorTitleTemplate: ` [{Title}]`,
						FloorDelimiter: '\n\n\n\n———— - ———— - ———— - ————\n',
					}
				},
				Number: {
					RefreshInterval: 20 // time by second
				},
				URL: {
					ReviewShow: `https://${location.host}/modules/article/reviewshow.php?rid={RID}&page={Page}`
				},
				Selector: {
					Floor: '#content>table:not(:is(:last-of-type, :first-of-type, :nth-of-type(2), :nth-last-of-type(2):nth-last-child(2)))',
					FloorA: 'tbody>tr>td:nth-of-type(2)>div:nth-of-type(2)>a[href^="#yid"]',
					PagesTable: '#content>table:is(:last-of-type:not(:last-child),:nth-last-of-type(2):nth-last-child(2))',
					PagesTr: '#content>table:is(:last-of-type:not(:last-child),:nth-last-of-type(2):nth-last-child(2))>tbody>tr',
					ReplyTable: '#content>form>table',
					FloorContent: 'hr+div',
					FloorTitle: 'tbody>tr>td:nth-of-type(2)>div:first-of-type>strong',
					FloorTitleRight: 'tbody>tr>td:nth-of-type(2)>div:nth-of-type(2)'
				},
				CSS: {
					Common: '#content>table:not(:is(:last-of-type, :first-of-type, :nth-of-type(2), :nth-last-of-type(2):nth-last-child(2)))>tbody>tr>td:last-of-type>:is(div:nth-of-type(1),div:nth-of-type(2)) {width: unset !important;}'
				},
				Config_Ruleset: {
					'version-key': 'config-version',
					'ignores': ["LOCAL-CDN"],
					'defaultValues': {
						downloadFormat: 'bbcode',
					},
					'updaters': {
						/*'config-key': [
							function() {
								// This function contains updater for config['config-key'] from v0 to v1
							},
							function() {
								// This function contains updater for config['config-key'] from v1 to v2
							}
						]*/
					}
				},
			},
			func: function() {
				const CommonStyle = require('CommonStyle');
				const ReplyAreaEnhance = require('ReplyAreaEnhance');
				const mousetip = require('mousetip');
				const utils = require('utils');
				const CONST = FuncInfo.CONST;
				const CM = new ConfigManager(CONST.Config_Ruleset);
				const CONFIG = CM.Config;
				const STORAGE = {}; // Public storage for all functions

				// Function for the page.
				// This shouldn't rely on any floor elements, because floor elements might be removed or modified
				const basicFuncs = [
					function css() {
						addStyle(CONST.CSS.Common, 'plus_review_css');
					},
					function linkJump() {
						const content = $('#content');
						$AEL(content, 'click', function(e) {
							const links = [...$All(content, 'table>tbody>tr>td:nth-of-type(2)>div:nth-of-type(2)>a[href^="#yid"]')];
							const getYID = a => a.href.match(/#yid\d+/)[0];
							const linkObjs = links.map(a => ({ a, yid: getYID(a) }));
							const isReplyLink = elm => elm.tagName === 'A' && elm.pathname === '/modules/article/reviewshow.php' &&
								  elm.host === location.host && elm.href.includes('#yid') && !links.includes(elm);
							if (isReplyLink(e.target)) {
								const a = e.target;
								const linkObj = linkObjs.find(obj => obj.yid === getYID(a));
								if (linkObj) {
									destroyEvent(e);
									linkObj.a.click();
								}
							}
						})
					},
					function autoRefresh() {
						let timer, t=1;

						// GUI
						const container = $$CrE({
							tagName: 'td',
							styles: {
								'text-align': 'left'
							}
						});
						const checkbox = $$CrE({
							tagName: 'input',
							props: {
								type: 'checkbox'
							},
							styles: {
								'vertical-align': 'middle',
							},
							listeners: [['change', function(e) {
								checkbox.checked ? start() : stop();
							}]]
						});
						const text = $$CrE({
							tagName: 'span',
							props: {
								innerText: CONST.Text.AutoRefresh_Off
							},
							styles: {
								'margin-left': '1em',
								'vertical-align': 'middle',
								'cursor': 'default',
							},
							listeners: [['click', function(e) {
								checkbox.click();
							}]]
						});
						container.appendChild(checkbox);
						container.appendChild(text);
						$(CONST.Selector.PagesTr).insertAdjacentElement('afterbegin', container);

						function start() {
							// When start, set current waiting time to 1s
							// Not 0s, in order to avoid someone frequently refreshing by clicking the checkbox again and again
							t = 1;
							timer = setInterval(countdown, 1000);
							text.innerText = replaceText(CONST.Text.AutoRefresh, {'{t}': t.toString()});
						}

						function stop() {
							text.innerText = CONST.Text.AutoRefresh_Off;
							clearInterval(timer);
						}

						function countdown() {
							if (--t < 0) {
								refresh();
								t = CONST.Number.RefreshInterval;
							}
							text.innerText = replaceText(CONST.Text.AutoRefresh, {'{t}': t.toString()});
						}
					},
					function replyInPage() {
						FL_recieveMessage('ReplySent', dealResponse, 'ReplyAreaEnhance');
						FL_recieveMessage('ReplyFailed', data => refresh(), 'ReplyAreaEnhance');

						function dealResponse(data) {
							const blob = data.response.response;
							const dom = utils.parseDocument(blob, function(dom) {
								const redirector = $(dom.head, 'meta[http-equiv="refresh"]');
								if (dom.body.childElementCount === 1 && !redirector) {
									// Error page
									const closeBtn = $(dom, 'a[href="javascript:window.close()"]');
									closeBtn.href = 'javascript: void(0);';
									$AEL(closeBtn, 'click', e => altbox.close());
									const altbox = alertify.alert(CONST.Text.ReplyErrorTitle, $(dom, '.block'));

									// Refresh with current page when reply returns an error
									refresh(getUrlArgv({
										url: location.href,
										name: 'page',
										defaultValue: '1'
									}));
								} else if (dom.body.childElementCount === 1 && redirector) {
									// Reply modification successed
									data.form.reset();
									$('#dialog') && unsafeWindow.closeDialog();

									// Refresh with current page when reply returns an error
									refresh(getUrlArgv({
										url: location.href,
										name: 'page',
										defaultValue: '1'
									}));
								} else {
									// Reply send successed
									data.form.reset();

									// Refresh to last page while successfully sent a new reply
									refresh();
								}
							});
						}
					},
					function reviewDownload() {
						// Adjust element width
						[...$All('#content>table:first-of-type td td')].forEach(td => td.style.width = 'unset');

						// Make button
						const container = $CrE('span');
						const formatText = $$CrE({
							tagName: 'span',
							props: {
								innerText: CONST.Text.DownloadFormat
							},
							styles: {
								'vertical-align': 'middle',
								'margin-right': '1em',
								'padding-left': '0.5em',
								'cursor': 'default',
							}
						});
						const formatChooser = $$CrE({
							tagName: 'select',
							styles: {
								'vertical-align': 'middle',
								'margin-right': '0.5em',
							},
							listeners: [['change', e => CONFIG.downloadFormat = formatChooser.value]]
						});
						const button = $$CrE({
							tagName: 'span',
							classes: [CommonStyle.ClassName.Button],
							props: {
								innerText: replaceText(CONST.Text.DownloadPost, {'{N}': $('#pagelink .last').innerText})
							},
							styles: {
								'vertical-align': 'middle',
							},
							listeners: [['click', e => download(formatChooser.value)]]
						});
						for (const [value, prop] of Object.entries(CONST.Text.DownloadFormats)) {
							formatChooser.appendChild($$CrE({
								tagName: 'option',
								props: {
									value: value,
									innerText: prop.name
								}
							}));
						}
						formatChooser.value = CONFIG.downloadFormat;
						//mousetip.settip(formatText, CONST.Text.FormatTip);
						//mousetip.settip(formatChooser, CONST.Text.FormatTip);
						tippy(container, {
							content: $$CrE({
								tagName: 'div',
								props: {
									innerHTML: CONST.Text.FormatTip
								},
							}),
							theme: 'wenku_tip',
							placement: 'bottom',
						});
						[formatText, formatChooser, button].forEach(elm => container.appendChild(elm));
						$('#content>table:first-of-type td td:last-of-type').appendChild(container);

						function download(format) {
							// Temporarily use current document's last page text as maxPage
							let maxPage = parseInt($('#pagelink .last').innerText, 10);
							let finished = 0;

							// Set button text
							button.innerText = replaceText(CONST.Text.DownloadingPost, {'{finished}': '0', '{all}': maxPage.toString()});

							// Get review's id
							const rid = getUrlArgv('rid');

							// Result json object
							const review = {
								rid,
								title: '',
								floors: [],
							};

							// Request all review pages
							const AM = new AsyncManager();
							for (let page = 1; page <= maxPage; page++) {
								getPage(AM, rid, page, downloadFailed);
							}

							// When all request finished, merge result
							AM.onfinish = e => {
								output(format);
								button.innerText = CONST.Text.DownloadSuccess;
							}
							AM.finishEvent = true;

							// Request specified page of specified review and concat all floors' data, with AM task management
							function getPage(AM, rid, page, onerror, retryCount=3) {
								// Append AM task
								AM.add();

								// Request specified page of specified review
								const url = replaceText(CONST.URL.ReviewShow, {'{RID}': getUrlArgv('rid'), '{Page}': page.toString()})
								utils.getDocument({url, callback: function(doc) {
									// Append floors' json into review.floors
									const data = getPageData(doc);
									review.floors.push.apply(review.floors, data);

									// Update button text
									button.innerText = replaceText(CONST.Text.DownloadingPost, {'{finished}': (++finished).toString(), '{all}': maxPage.toString()});

									// Finish AM task
									AM.finish();
								}, onerror: retry});

								function retry(err) {
									if (--retryCount > 0) {
										DoLog(`Review download: Retrying ${url}, retryCount=${retryCount}`);
										getPage(AM, rid, page, onerror, retryCount);
									} else {
										if (typeof onerror === 'function') {
											// onerror shouldn't throw any error. onerror first then finish task
											onerror(err);
											AM.finish();
										} else {
											// No onerror, just finish task then throw the error
											AM.finish();
											Err(`Review download: download failed after all retrying.`);
										}
									}
								}
							}

							// Returns an array of floors' data
							function getPageData(doc) {
								// Record realtime pages count
								maxPage = parseInt($('#pagelink .last').innerText, 10);

								// Parse floors' json
								const floors = parseFloors(doc);
								const data = floors.map(floor => jsonFloor(floor));

								return data;
							}

							// Returns a json of a floor's data
							// Return: {username, userid, title, time, floorNum, contentText, contentBBCode}
							function jsonFloor(floor) {
								const userA = $(floor, '.avatar+br+strong>a');
								const username = userA.innerText;
								const userid = getUrlArgv(userA.href, 'uid');
								const title = $(floor, CONST.Selector.FloorTitle).innerText;
								const time = $(floor, CONST.Selector.FloorTitleRight).innerText.match(/\d{4}-\d{2}-\d{2} \d{2}:\d{2}:\d{2}/)[0];
								const floorNum = parseInt($(floor, CONST.Selector.FloorA).innerText.match(/\d+/)[0], 10);
								const contentText = $(floor, CONST.Selector.FloorContent).innerText;
								const contentBBCode = getFloorContent($(floor, CONST.Selector.FloorContent));
								return {username, userid, title, time, floorNum, contentText, contentBBCode};
							}

							// Output in specified format
							function output(format) {
								// Sort floors
								review.floors.sort((f1, f2) => f1.floorNum - f2.floorNum);

								// Set review title as first floor's title
								review.title = review.floors[0].title;

								// Get blob and filename in format
								const blob = ({
									text: outputText,
									bbcode: outputBBCode,
									json: outputJson,
								})[format]();
								const filename = `${review.rid} - ${review.title}(${CONST.Text.DownloadFormats[format].name}).${CONST.Text.DownloadFormats[format].ext}`;

								// Output blob
								const url = URL.createObjectURL(blob);
								$$CrE({
									tagName: 'a',
									attrs: {
										href: url,
										download: filename
									}
								}).click();
								setTimeout(e => URL.revokeObjectURL(url), 0);

								function outputText() {
									const floorText = review.floors.map(floordata => replaceText(CONST.Text.DownloadTemplate.FloorTemplate, {
										'{FloorNum}': floordata.floorNum,
										'{Username}': floordata.username,
										'{UserID}': floordata.userid,
										'{Time}': floordata.time,
										'{Floor Title}': floordata.title ? replaceText(CONST.Text.DownloadTemplate.FloorTitleTemplate, {'{Title}': floordata.title}) : '',
										'{Floor Content}': floordata.contentText
									})).join(CONST.Text.DownloadTemplate.FloorDelimiter);
									const text = replaceText(CONST.Text.DownloadTemplate.Template, {
										'{Review ID}': review.rid,
										'{Review Title}': review.title,
										'{Review URL}': replaceText(CONST.URL.ReviewShow, {'{RID}': rid, '{Page}': 1}),
										'{Download Time}': new Date().toLocaleString(),
										'{Format}': format,
										'{Floor Content}': floorText,
									});
									const blob = new Blob([text], {
										type: 'text/plain',
										endings: 'native'
									});
									return blob;
								}

								function outputBBCode() {
									const floorText = review.floors.map(floordata => replaceText(CONST.Text.DownloadTemplate.FloorTemplate, {
										'{FloorNum}': floordata.floorNum,
										'{Username}': floordata.username,
										'{UserID}': floordata.userid,
										'{Time}': floordata.time,
										'{Floor Title}': floordata.title ? replaceText(CONST.Text.DownloadTemplate.FloorTitleTemplate, {'{Title}': floordata.title}) : '',
										'{Floor Content}': floordata.contentBBCode
									})).join(CONST.Text.DownloadTemplate.FloorDelimiter);
									const text = replaceText(CONST.Text.DownloadTemplate.Template, {
										'{Review ID}': review.rid,
										'{Review Title}': review.title,
										'{Review URL}': replaceText(CONST.URL.ReviewShow, {'{RID}': rid, '{Page}': 1}),
										'{Download Time}': new Date().toLocaleString(),
										'{Format}': format,
										'{Floor Content}': floorText,
									});
									const blob = new Blob([text], {
										type: 'text/plain',
										endings: 'native'
									});
									return blob;
								}

								function outputJson() {
									const blob = new Blob([JSON.stringify(review)], {
										type: 'text/plain',
										endings: 'transparent'
									});
									return blob;
								}
							}

							function downloadFailed(err) {
								alertify.error(CONST.Text.DownloadFailed);
								DoLog(LogLevel.Error, ['downloadFailed', err])
							}
						}
					},
					function pageChangeInPage() {
						// Clicking #pagelink>a
						$AEL(document, 'click', e => {
							const elm = e.target;
							if ([...$All('#pagelink>a')].includes(elm)) {
								e.preventDefault();
								refresh(getUrlArgv({
									url: elm.href,
									name: 'page',
									defaultValue: '1'
								}));
							}
						});

						// Input pagenum and hit enter
						// Deal <input> everytime <input> refreshed
						detectDom({
							selector: '#pagelink [name="page"]',
							callback: input => {
								input.type = 'number';
								input.style.width = '10ex'; // 6ex(maxlength=6) + 4ex(increase/decrease button)
								input.onkeydown = e => {};
								$AEL(input, 'keydown', e => {
									if (e.key === 'Enter' && !e.ctrlKey && !e.metaKey && !e.shiftKey) {
										refresh(input.value, 10);
									}
								});
							},
							once: false
						});
					}
				];

				// Functions for every floor <table>
				// These will be executed for every floor <table> once, containing all in html and add via refreshing
				// These may be executed without floors inserted into current dom
				const floorFuncs = [
					function floorQuote(floor, doc) {
						// '本帖已经超过允许回复时间'
						if (!$('#pcontent')) {
							return false;
						}

						const floorA = $(floor, CONST.Selector.FloorA);
						const button = $$CrE({
							tagName: 'span',
							classes: [CommonStyle.ClassName.Button],
							listeners: [['click', quote]],
							props: {
								innerText: CONST.Text.Quote
							}
						});
						const tip_panel = $$CrE({
							tagName: 'div',
							props: {
								innerText: CONST.Text.QuoteNum[0]
							}
						});
						const btn = $$CrE({
							tagName: 'span',
							props: {
								innerText: CONST.Text.QuoteNum[1]
							},
							classes: [CommonStyle.ClassName.Button],
							listeners: [['click', quoteNum]]
						});
						tip_panel.appendChild(btn);
						const panel = tippy(button, {
							content: tip_panel,
							theme: 'wenku_tip',
							placement: 'top',
							interactive: true,
						});
						floorA.insertAdjacentElement('beforebegin', button);
						floorA.insertAdjacentHTML('beforebegin', ' | ');

						function quote() {
							const bbcode = `${getFloorLink(floor)} [quote]${getFloorContent($(floor, CONST.Selector.FloorContent))}[/quote]`;
							insert(bbcode);
						}

						function quoteNum() {
							const bbcode = getFloorLink(floor);
							insert(bbcode);
						}

						// Insert into <textarea id='pcontent'>
						function insert(bbcode, focus=true) {
							const textarea = $('#pcontent:not(#dialog #pcontent)');
							utils.insertText(textarea, bbcode, focus);
						}
					},
					function editInPage(floor, doc) {
						const editBtn = $(floor, `a[href^="https://${location.host}/modules/article/reviewedit.php?yid="]`);
						editBtn && editBtn.addEventListener('click', (e) => {
							e.preventDefault();

							utils.openDialog(e.target.href + '&ajax_gets=jieqi_contents');
						})
					},
					function scaleimgs(floor, doc) {
						const w = $('#content').clientWidth * 0.8 - 3 - 3; // td.width = "80%", .even {padding: 3px;}, table.grid {padding: 3px;}
						[...$All(floor, '.divimage>img')].forEach(img => {
							const loaded = img.width && img.height;
							const myOnload = e => {
								img.resized = img.width > w;
								img.width = img.resized ? w : img.width;
							}
							loaded ? myOnload() : img.onload = myOnload;
							img.onmouseover = e => img.resized && (img.style.cursor = 'pointer');
						});
					}
				];

				// Do not load functions when required by other modules and not running in reviewshow page
				if (utils.testChecker(FuncInfo.checker)) {
					detectDom('.main.m_foot', function() {
						loadBasicFuncs(basicFuncs);
						loadFloorFuncs(floorFuncs);
					});
				}
				return {refresh};

				// Refresh current page without reload
				// Argument: pageOrDom: page number (string or number) or a document
				function refresh(pageOrDom='last') {
					if (typeof pageOrDom === 'string' || typeof pageOrDom === 'number') {
						const page = pageOrDom;
						utils.getDocument(
							replaceText(CONST.URL.ReviewShow, {'{RID}': getUrlArgv('rid'), '{Page}': page.toString()}),
							applyDocument
						);
					} else {
						const dom = pageOrDom;
						applyDocument(dom);
					}

					function applyDocument(dom) {
						loadFloorFuncs(floorFuncs, dom);

						// Record scroll status
						const scrollStatus = [window.scrollY, $('#content').scrollTop];

						// Get floors' info
						const oldInfos = parseFloors(document).map(floor => info(floor));
						const newInfos = parseFloors(dom).map(floor => info(floor));

						// Get all floors' info that should stay in page and remove unreserved old floors from page
						// Fill newInfos directly first, then use oldInfos to replace same-floor-infos
						const reservedInfos = [...newInfos];
						oldInfos.forEach(old_info => {
							const sameNew = reservedInfos.findIndex(new_info => isSameFloor(new_info, old_info).same);
							const yidsameNewInfo = reservedInfos.find(new_info => isSameFloor(new_info, old_info).yid);
							if (sameNew >= 0) {
								reservedInfos.push(old_info);
								reservedInfos.splice(sameNew, 1);
							} else {
								old_info.floor.remove();
								if (yidsameNewInfo) {
									const titleRight = $(yidsameNewInfo.floor, CONST.Selector.FloorTitleRight);
									titleRight.insertAdjacentHTML('afterbegin', ' | ');
									titleRight.insertAdjacentElement('afterbegin', $$CrE({
										tagName: 'span',
										classes: [CommonStyle.ClassName.Text],
										props: {
											innerText: CONST.Text.FloorContentModified
										}
									}));
								}
							}
						});
						reservedInfos.sort((info1, info2) => info1.num_yid - info2.num_yid);

						// Insert new floors that do not currently exist in page and sort all floors in page
						reservedInfos.forEach(info => appendFloor(info.floor));

						// Update page url
						const page = $(dom, '#pagestats').innerText.match(/(\d+)\/\d+/)[1];
						const url = replaceText(CONST.URL.ReviewShow, {'{RID}': getUrlArgv('rid'), '{Page}': page.toString()})
						utils.setPageUrl(url);

						// Update #pagelink
						const newPagelink = $(dom, '#pagelink');
						const oldPagelink = $('#pagelink');
						const parent = oldPagelink.parentElement;
						// Remove old first, to aviod #pagelink dealers misgot the old one
						oldPagelink.remove();
						// Append new #pagelink
						parent.appendChild(newPagelink);

						// Recover scroll status
						[window.scrollY, $('#content').scrollTop] = scrollStatus;
					}

					// Get floor info, returns {floor, yid, str_yid, num_yid, title, content}
					function info(floor) {
						const yid = $(floor, CONST.Selector.FloorA).href.match(/#(yid\d+)/)[1];
						const str_yid = yid;
						const num_yid = parseInt(str_yid.replace('yid', ''), 10);
						const title = $(floor, CONST.Selector.FloorTitle).innerHTML;
						const content = getFloorContent($(floor, CONST.Selector.FloorContent));
						return {floor, yid, str_yid, num_yid, title, content};
					}

					// Compare whether two floors/infos are completely or partly same
					// Returns: {same, yid, title, content}
					function isSameFloor(ft1, ft2) {
						const getInfo = ft => ft.toString() === '[object HTMLTableElement]' ? info(ft) : ft;
						const [info1, info2] = [getInfo(ft1), getInfo(ft2)];
						const yid = info1.yid === info2.yid;
						const title = info1.title === info2.title;
						const content = info1.content === info2.content;
						const same = yid && title && content
						return {same, yid, title, content};
					}

					// Append a floor into the bottom of page
					function appendFloor(floor) {
						$(CONST.Selector.PagesTable).insertAdjacentElement('beforebegin', floor);
					}
				}

				// Get floor content by BBCode format (content only, no title)
				// Argv: <div> content element
				function getFloorContent(contentEle, original=false) {
					const subNodes = contentEle.childNodes;
					let content = '';

					for (const node of subNodes) {
						const type = node.nodeName;
						switch (type) {
							case '#text':
								// Prevent 'Quote:' repeat
								content += node.data.replace(/^\s*Quote:\s*$/, ' ');
								break;
							case 'IMG':
								// wenku8 has forbidden [img] tag for secure reason (preventing CSRF)
								//content += '[img]S[/img]'.replace('S', node.src);
								content += original ? '[img]S[/img]'.replace('S', node.src) : ' S '.replace('S', node.src);
								break;
							case 'A':
								content += '[url=U]T[/url]'.replace('U', node.getAttribute('href')).replace('T', getFloorContent(node));
								break;
							case 'BR':
								// no need to add \n, because \n will be preserved in #text nodes
								//content += '\n';
								break;
							case 'DIV':
								if (node.classList.contains('jieqiQuote')) {
									content += getTagedSubcontent('quote', node);
								} else if (node.classList.contains('jieqiCode')) {
									content += getTagedSubcontent('code', node);
								} else if (node.classList.contains('divimage')) {
									content += getFloorContent(node, original);
								} else {
									content += getFloorContent(node, original);
								}
								break;
							case 'CODE': content += getFloorContent(node, original); break; // Just ignore
							case 'PRE':  content += getFloorContent(node, original); break; // Just ignore
							case 'SPAN': content += getFontedSubcontent(node); break; // Size and color
							case 'P':    content += getFontedSubcontent(node); break; // Text Align
							case 'B':    content += getTagedSubcontent('b', node); break;
							case 'I':    content += getTagedSubcontent('i', node); break;
							case 'U':    content += getTagedSubcontent('u', node); break;
							case 'DEL':  content += getTagedSubcontent('d', node); break;
							default:     content += getFloorContent(node, original); break;
								/*
									case 'SPAN':
										subContent = getFloorContent(node);
										size = node.style.fontSize.match(/\d+/) ? node.style.fontSize.match(/\d+/)[0] : '';
										color = node.style.color.match(/rgb\((\d+), ?(\d+), ?(\d+)\)/);
										break;
									*/
						}
					}
					return content;

					function getTagedSubcontent(tag, node) {
						const subContent = getFloorContent(node, original);
						return '[{T}]{S}[/{T}]'.replaceAll('{T}', tag).replaceAll('{S}', subContent);
					}

					function getFontedSubcontent(node) {
						let tag, value;

						let strSize = node.style.fontSize.match(/\d+/);
						let strColor = node.style.color;
						let strAlign = node.align;
						strSize = strSize ? strSize[0] : null;
						strColor = strColor ? rgbToHex.apply(null, strColor.match(/\d+/g)) : null;

						tag = tag || (strSize  ? 'size'  : null);
						tag = tag || (strColor ? 'color' : null);
						tag = tag || (strAlign ? 'align' : null);
						value = value || strSize || null;
						value = value || strColor || null;
						value = value || strAlign || null;

						const subContent = getFloorContent(node, original);
						if (tag && value) {
							return '[{T}={V}]{S}[/{T}]'.replaceAll('{T}', tag).replaceAll('{V}', value).replaceAll('{S}', subContent);
						} else {
							return subContent;
						}

						function rgbToHex(r, g, b) {return ((r << 16) | (g << 8) | b).toString(16).padStart('0', 6);}
					}
				}

				// Get link bbcode points to given floor
				// Argv: <table> floor element
				function getFloorLink(floor) {
					const floorA = $(floor, 'a[href^="#yid"]');
					return `[url=${floorA.href}]${floorA.innerText}[/url]`;
				}

				// func() for each func
				function loadBasicFuncs(basicFuncs) {
					basicFuncs.forEach(func => loadWithWrapper(func));
				}

				// func(floor, doc) for each floor and floorFunc
				function loadFloorFuncs(floorFuncs, doc=document) {
					const floors = parseFloors(doc);
					floorFuncs.forEach(func => {
						floors.forEach(floor => loadWithWrapper(func, floor, doc));
					});
				}

				// Get all floor <table> as an array
				function parseFloors(doc=document) {
					const content = $(doc, '#content');
					const tables = $All(doc, CONST.Selector.Floor);
					return [...tables];
				}

				function loadWithWrapper(func, ...args) {
					setTimeout(function() {
						/* This will lose erroor stack information
						try {
							func.apply(null, args);
						} catch (err) {
							Err(err);
						}
						*/
						func.apply(null, args);
					}, 0);
				}
			}
		},

		// Reply area enhance
		{
			name: '编辑器增强',
			description: '发帖/回帖 输入框功能增强,提供本地图片快速插入等小功能',
			id: 'ReplyAreaEnhance',
			STOP: false,
			checker: {
				type: 'starturl',
				value: [
					// Page: reviews list
					`https://${location.host}/modules/article/reviews.php`,

					// Page: review
					`https://${location.host}/modules/article/reviewshow.php`,

					// Page: review edit
					`https://${location.host}/modules/article/reviewedit.php`,

					// Page: book
					`https://${location.host}/book`,
					`https://${location.host}/modules/article/articleinfo.php`,
				]
			},
			CONST: {
				Text: {
					InsertWebImage: '插入网图链接',
					SelectLocalImage: '选择本地图片',
					IWItip: '直接插入网络图片的链接地址',
					SLItip: '选择本地图片上传到第三方图床,然后再插入图床提供的图片链接</br>您也可以直接拖拽图片到输入框,或者Ctrl+V直接粘贴您剪贴板里面的图片</br></br>上传图片请遵守法律以及图床使用规定</br><span style="color: orange;">请<span style="color: red;">不要滥用</span>(你知道什么样算是滥用,包括但不限于上传违反法律/图床规定的图片、短时间大量上传图片等),如果因此出现问题,<span style="color: red;">开发者不对用户的这种恶意行为负责</span>,出现此种情况开发者只能选择<span style="color: red;">关闭此功能</span></span>',
					InputImageURL: '请输入图片路径,支持https以及包含中文和特殊符号的链接',
					DefaultInputURL: 'https://',
					ProtocolMissing: '请输入完整的图片路径!',
					ExtensionInvalid: '图片的链接需要以.jp(e)g/.png/.gif/.bmp等常见图片扩展名结尾',
					ImageOnly: '抱歉,您选择的图片格式无法识别</br>(建议选择jpeg/png格式的图片)',
					Uploading: '正在上传图片…',
					UploadSuccess: '图片上传成功',
					UploadError: '图片上传失败',
					ErrorMarkerID: '上传失败',
					SubmitBtnHotkey: '(Ctrl+Enter)',
					alertTitle: '书评吐槽',
					NoEmptySubmit: '请不要提交空的评论',
					ContentTooShort: '回复内容太短,至少需要7个字节</br>(通常来讲,一个汉字占两个字节,一个数字/字母/普通符号占一个字节)',
					ReplyFailed: '发送失败,请重试',
					DraftInited: '书评草稿初始化完毕',
					DraftLoaded: '成功加载了于{time}编辑的草稿',
					DraftSaved: '编辑框内容已于{time}保存到草稿',
					DraftCleared: '已清空草稿',
					AlertTitle: '编辑器增强',
					DraftSetting: '书评草稿',
					Expires: '草稿保存时间(天)',
					StorageSetting: {
						SettingTitle: '草稿管理',
						title: '标题',
						content: '内容',
						time: '上次编辑时间',
						Edit: '打开'
					}
				},
				Config_Ruleset: {
					'version-key': 'config-version',
					'ignores': ["LOCAL-CDN"],
					'defaultValues': {
						draftexpires: 31,
						drafts: []
					},
					'updaters': {
						/*'config-key': [
							function() {
								// This function contains updater for config['config-key'] from v0 to v1
							},
							function() {
								// This function contains updater for config['config-key'] from v1 to v2
							}
						]*/
					}
				},
			},
			func: function() {
				const WenkuBlockGUI = require('WenkuBlockGUI');
				const imager = require('imager');
				const utils = require('utils');
				const CONST = FuncInfo.CONST;
				const CM = new ConfigManager(CONST.Config_Ruleset);
				const CONFIG = CM.Config;
				let ubbCode = '';

				detectDom({
					selector: 'form[name="frmreview"]',
					callback: form => enhance(form),
					once: false
				});
				return {enhance};

				// Enhance a UBBEditor
				// Arguments: form
				function enhance(form) {
					if (form.matches('#dialog form') && !$(form, '#UBB_Menu')) {
						// in dialog and no menu exist, load UBBEditor
						loadUBBEditor(UBBEditor => work(form, UBBEditor));
					} else {
						// Directly in page, wait for UBBEditor
						detectDom(form, '#UBB_Menu', menu => work(form, unsafeWindow.UBBEditor))
					}

					function work(form, UBBEditor) {
						const textarea = $(form, 'textarea[name="pcontent"]');
						const ptitle = $(form, 'input[name="ptitle"]') || makeptitle();

						imgSelector();
						extensiveEditor();
						editorDrafts();
						hotkeyReply();
						onsubmitEnhance();

						// Broadcast: a new UBBEditor has created (Not used yet)
						FL_postMessage('UBBEditorCreated', {form, UBBEditor});

						// Provide image inputs
						function imgSelector() {
							// Imager menu
							const menu = $(form, '#UBB_Menu');
							const elmImage = $(form, '#menuItemInsertImage');
							const imagers = new WenkuBlockGUI.PlusList({
								id: 'plus_imager',
								list: [
									{value: CONST.Text.InsertWebImage, tip: CONST.Text.IWItip, onclick: insertWebpic},
									{value: CONST.Text.SelectLocalImage, tip: CONST.Text.SLItip, onclick: pickfile}
								],
								parentElement: menu.parentElement,
								insertBefore: $(form, '#SmileListTable'),
								visible: false,
								onshow: onshow
							});
							elmImage.onclick = (e) => {
								e.stopPropagation();
								imagers.show();
							};
							$AEL(document, 'click', imagers.hide);

							// drag-drop & copy-paste
							textarea.addEventListener('paste', pictureGot);
							textarea.addEventListener('dragenter', destroyEvent);
							textarea.addEventListener('dragover', destroyEvent);
							textarea.addEventListener('drop', pictureGot);

							function onshow() {
								imagers.div.style.left = UBBEditor.GetPosition(elmImage).x + 'px';
								imagers.div.style.top = UBBEditor.GetPosition(elmImage).y + 20 + 'px';
							}

							function pickfile() {
								const fileinput = $CrE('input');
								fileinput.type = 'file';
								fileinput.addEventListener('change', pictureGot);
								fileinput.click();
							}

							function pictureGot(e) {
								// Get picture file
								const input = e.dataTransfer || e.clipboardData || window.clipboardData || e.target;
								if (!input.files || input.files.length === 0) {return false;};
								const file = input.files[0];
								const mimetype = file.type;
								const name = file.name;

								// Pasting an unrecognizable file is not a mistake
								// Maybe the user just wants to paste the filename here
								// Otherwise getting an unrecognizable file is a mistake
								if (!mimetype || mimetype.split('/')[0] !== 'image') {
									if (!e.clipboardData && !window.clipboardData) {
										destroyEvent(e);
										alertify.error(CONST.Text.ImageOnly);
									}
									return false;
								} else {
									destroyEvent(e);
								}

								// Insert picture marker
								const id = utils.randstr(16, true, textarea.value);
								const marker = replaceText('[image_uploading={ID} name={NAME}]', {
									'{ID}': id,
									'{NAME}': name
								});
								utils.insertText(textarea, marker);

								// Upload
								alertify.notify(CONST.Text.Uploading);
								imager.upload(file, function onload(url) {
									// Calculate selection position changes
									const [posStart, posEnd] = [textarea.selectionStart, textarea.selectionEnd].map(
										pos => calcPosition(pos, textarea.value.indexOf(marker), marker.length, url.length)
									);

									// Just replace but not replaceAll, because replaceAll make calculation selection position changes much more difficult
									textarea.value = textarea.value.replace(marker, url);

									// Apply selection position changes
									setTimeout(e => textarea.setSelectionRange(posStart, posEnd));

									// Show success
									alertify.success(CONST.Text.UploadSuccess);
								}, function onerror(err) {
									textarea.value = textarea.value.replace(marker, marker.replace(id, CONST.Text.ErrorMarkerID));
									alertify.error(CONST.Text.UploadError);
									DoLog(LogLevel.Error, ['Reply area enhance - pictureGot: Image upload error', err]);
								}, ['custom-extention']);

								// Calculate selection start/end with current position and text modification
								function calcPosition(curPos, textStart, oldLen, newLen) {
									const textEnd = textStart + oldLen;
									if (curPos < textStart) {
										// Selection position is not affected by text modification
										return curPos;
									} else if (curPos >= textStart && curPos <= textEnd) {
										// Selection position is somewhere inside old text - just set position to the end of new text
										return textStart + newLen;
									} else if (curPos > textEnd) {
										// Selection position is completly affected, move position with length modification
										return curPos + newLen - oldLen;
									}
									// NO WAY, CODE SHOULDNT BE HERE
									Err('calcPosition: Unexpected execution');
									return 0;
								}
							}

							function insertWebpic() {
								alertify.prompt(CONST.Text.alertTitle, CONST.Text.InputImageURL, CONST.Text.DefaultInputURL, function onok(e, val) {
									if (!/^https?:\/\//.test(val)) {
										alertify.alert(CONST.Text.alertTitle, CONST.Text.ProtocolMissing);
										return false;
									}
									if (!/\.(jpe?g|png|gif|bmp)$/.test(val)) {
										alertify.confirm(CONST.Text.alertTitle, CONST.Text.ExtensionInvalid);
										return false;
									}
									const url = new URL(val).href;
									utils.insertText(textarea, url);
								}, function oncancel() {});
							}
						}

						// Provide #ptitle if doesn't exist by default
						function makeptitle() {
							if (!$(form, 'input[name="ptitle"]')) {
								const html = '<tr>\n    <td class="odd" width="25%">标题</td>\n    <td class="even"><input type="text" class="text" name="ptitle" id="ptitle" size="60" maxlength="60" value=""></td>\n  </tr>'
								$(form, 'caption+tbody').insertAdjacentHTML('afterbegin', html);
							}
							return $(form, 'input[name="ptitle"]');
						}

						// Textarea auto expand (A HUGE MOUNTAIN OF SHIT CODE)
						// 你真的要点开看这里的代码吗?在这里放一双没有看过这代码的眼睛先(👀)
						// 如果想要了解其原理,可以参照 https://luszy.com/archives/383.html
						// 虽然但是,这段代码他能跑🏃🏻🏃🏻🏃🏻!!!
						function extensiveEditor() {
							if (![...$All('#dialog form')].includes(form)) {
								const cstyle = getComputedStyle(textarea);
								// Create elements
								const container = $$CrE({
									tagName: 'div',
									styles: {
										position: 'relative',
										'min-width': `${textarea.clientWidth}px`,
										'min-height': `${textarea.clientHeight}px`,
										'max-width': '650px',
										width: 'fit-content',
										height: 'fit-content',
										display: 'flow-root',
									}
								});
								const placeholder = $$CrE({
									tagName: 'div',
									styles: {
										visibility: 'hidden',
										'max-width': '650px',
										'max-height': '60vh',
										'overflow-y': 'scroll',
										...['white-space', 'word-wrap', 'font', 'padding', 'margin', 'border'].reduce((style, prop) => {
											style[prop] = cstyle[prop];
											return style;
										}, {}),
									}
								});
								// Adjust textarea's style
								textarea.style.width = `calc(100% - ${cstyle['border-left-width']} - ${cstyle['border-right-width']})`;
								textarea.style.height = `calc(100% - ${cstyle['border-top-width']} - ${cstyle['border-bottom-width']})`;
								textarea.style.top = textarea.style.left = '0';
								textarea.style.position = 'absolute';
								textarea.style.resize = 'none';
								textarea.style['overflow-y'] = 'scroll';
								textarea.style['max-width'] = '650px';
								textarea.style['max-height'] = '60vh';
								// Modify innerText while textarea.value changes
								const writeValue = e => {
									setTimeout(e => {
										placeholder.innerText = textarea.value;
										textarea.value.endsWith('\n') && (placeholder.innerHTML += ' ');
									}, 0);
								}
								$AEL(textarea, 'input', writeValue);
								$AEL(textarea.form, 'reset', writeValue);
								Object.defineProperty(textarea, 'value', {
									get: Object.getOwnPropertyDescriptor(HTMLTextAreaElement.prototype, 'value').get,
									set: function(val) {
										const rval = Object.getOwnPropertyDescriptor(HTMLTextAreaElement.prototype, 'value').set.apply(this, arguments);
										writeValue();
										return rval;
									},
									enumerable: true,
									configurable: true
								})
								writeValue();
								// Modify DOM
								textarea.insertAdjacentElement('afterend', container);
								container.appendChild(textarea);
								container.appendChild(placeholder);
							}
						}

						// Drafting
						function editorDrafts() {
							// No drafts for in-dialog forms
							if (form.matches('#dialog form')) {
								return false;
							}

							// Draft status text
							const text = $$CrE({
								tagName: 'span',
								props: { innerText: CONST.Text.DraftInited },
								styles: { 'margin-left': '0.5em' }
							});
							const container = $(form, 'table:not(table table)>tbody>tr:last-of-type>td:last-of-type');
							[...container.childNodes].filter(node => node.nodeName === '#text').forEach(node => node.remove());
							container.appendChild(text);

							// form properties, for recognizing different forms
							const props = [...$All(form, 'input[type="hidden"]')].reduce((obj, input) => ((obj[input.name] = obj[input.value], obj)), {});
							props.formaction = form.getAttribute('action');
							const key = Object.keys(props).sort().map(prop => `${prop}=${props[prop]}`).join(',');

							// Save draft when change fires, or every 30 seconds passed
							[textarea, ptitle].forEach(elm => ['change', 'blur'].forEach(evt => $AEL(elm, evt, e => saveDraft())));
							setInterval(saveDraft, 30 * 1000);

							// Load previous draft from config
							loadDraft();

							function loadDraft() {
								const draft = CONFIG.drafts.find(d => d.key === key);
								if (draft) {
									ptitle.value = draft.title;
									textarea.value = draft.content;
									text.innerText = replaceText(CONST.Text.DraftLoaded, {
										'{time}': utils.getTime(draft.time)
									});
								}
							}

							function saveDraft() {
								const i = CONFIG.drafts.findIndex(d => d.key === key);
								if (!ptitle.value && !textarea.value) {
									if (i >= 0) {
										CONFIG.drafts.splice(i, 1);
										text.innerText = CONST.Text.DraftCleared;
									}
									return false;
								}

								const draft = i >= 0 ? CM.getConfig(`drafts/${i}`) : {key};
								draft.title = ptitle.value;
								draft.content = textarea.value;
								draft.time = new Date().getTime();
								draft.url = location.href;
								i >= 0 ? CM.setConfig(`drafts/${i}`, draft) : CONFIG.drafts.push(draft);

								text.innerText = replaceText(CONST.Text.DraftSaved, {
									'{time}': utils.getTime(draft.time)
								});
							}
						}

						// Reply with hotkey
						function hotkeyReply() {
							const submitBtn = $(form, '[type="submit"]');
							submitBtn.value = submitBtn.value.trim() + ' ' + CONST.Text.SubmitBtnHotkey;
							submitBtn.style.padding = '0.4em 0.8em';
							submitBtn.style.height = '100%';
							$AEL(form, 'keydown', e => {
								if (e.key === 'Enter' && (utils.getOS() !== 'Windows' ? (e.metaKey || e.ctrlKey) : e.ctrlKey)) {
									submitBtn.click();
								}
							});
						}

						// onsubmit enhance
						function onsubmitEnhance() {
							$AEL(form, 'submit', function(e) {
								// Do not submit empty form
								if (!textarea.value) {
									alertify.alert(CONST.Text.alertTitle, CONST.Text.NoEmptySubmit);
									e.preventDefault();
									return false;
								}

								// Content length check (≥7)
								if (utils.formEncode(textarea.value).replaceAll(/%[0-9ABCDEF]{2}/g, '%').length < 7) {
									alertify.alert(CONST.Text.alertTitle, CONST.Text.ContentTooShort);
									e.preventDefault();
									return false;
								}

								// Reply without page reload while in reviewshow page
								if (utils.testChecker(FL_getFunction('ReviewEnhance').checker)) {
									e.preventDefault();
									utils.submitForm(form, function onload(response) {
										FL_postMessage('ReplySent', {response, form, UBBEditor});
									}, function onerror(err) {
										FL_postMessage('ReplyFailed', {err, form, UBBEditor});
										DoLog(LogLevel.Error, ['Send review reply err', err]);
										alertify.alert(CONST.Text.alertTitle, CONST.Text.ReplyFailed);
									});
								}
							});
						}
					}

					// Loadin UBBEditor with callback(UBBEditor)
					function loadUBBEditor(callback, retry=3) {
						const textarea = $(form, '#pcontent');
						const id = textarea.id = 'pcontent-' + utils.randstr(4); //id shouldn't contain '"' and '\'

						// Cache ubbCode
						if (!ubbCode) {
							GM_xmlhttpRequest({
								method: 'GET',
								url: `https://${location.host}/scripts/ubbeditor_gbk.js`,
								responseType: 'blob',
								onload: response => {
									const blob = response.response;
									const reader = new FileReader();
									$AEL(reader, 'load', e => {
										ubbCode = reader.result;
										loadin();
									});
									$AEL(reader, 'error', err => {
										Err('ReplyAreaEnhance: loadUBBEditor reader error');
									});
									reader.readAsText(blob, ['gbk', 'big5'][utils.getLang()]);
								},
								onerror: e => --retry ? loadUBBEditor(retry) : Err('ReplyAreaEnhance: loadUBBEditor xhr error')
							});
						} else {
							loadin();
						}

						function loadin() {
							const hideeve = 'function hideeve(){form.querySelector("#"+ubb_subdiv).style.display = "none";}'
							const code = `${ubbCode};\n${hideeve}\nUBBEditor.Create('${id}'); cb(UBBEditor);`;
							const func = Function('form', 'cb', code);
							setTimeout(e => func(form, callback), 0);
						}
					}
				}
			},
			alwaysRun: function() {
				const utils = require('utils');
				const CONST = FuncInfo.CONST;

				const CM = new ConfigManager(CONST.Config_Ruleset);
				const CONFIG = CM.Config;
				CM.updateAllConfigs();

				const time = new Date().getTime();
				const expires = t => (time - t) >= CONFIG.draftexpires * 24 * 60 * 60 * 1000;
				const drafts = CM.getConfig('drafts').filter(d => !expires(d.time));
				CM.setConfig('drafts', drafts);
			},
			setting: function setter() {
				const SettingPanel = require('SettingPanel');
				const CONST = FuncInfo.CONST;

				const CM = new ConfigManager(CONST.Config_Ruleset);
				CM.updateAllConfigs();

				const panel = new SettingPanel.easySettings({
					title: CONST.Text.AlertTitle,
					areas: [{
						title: CONST.Text.DraftSetting,
						items: [{
							text: CONST.Text.Expires,
							path: 'draftexpires',
							type: 'number'
						}],
					}]
				}, CM);

				new SettingPanel.easyStorage({
					title: CONST.Text.StorageSetting.SettingTitle,
					path: 'drafts',
					key: 'key',
					panel,
					props: {
						title: {
							type: 'string',
							name: CONST.Text.StorageSetting.title
						},
						content: {
							type: 'string',
							name: CONST.Text.StorageSetting.content
						},
						time: {
							type: 'time',
							name: CONST.Text.StorageSetting.time
						}
					},
					operations: [{
						type: 'func',
						text: CONST.Text.StorageSetting.Edit,
						func: (e, items, index) => {
							window.open(items[index].url);
							return items;
						}
					}, {type: 'delete'}]
				}, CM);
			}
		},

		// Index block folding
		{
			name: '首页版块折叠',
			description: '为首页所有版块提供具有记忆性的折叠/展开',
			id: 'IndexFolding',
			STOP: false,
			checker: {
				type: 'regurl',
				value: [
					/^https?:\/\/www\.wenku8\.(net|cc)\/index\.php([\?#][\s\S]*)?$/
				]
			},
			CONST: {
				Text: {
					Fold: '折叠',
					Unfold: '展开',
					ResetTitle: `${GM_info.script.name}-首页版块折叠 功能提示`,
					ResetTip: `由于文库首页新增了以往没有的版块,现在首页各个板块的折叠/展开状态需要您重新设置。`,
				},
				CSS: '.block.fold>.blockcontent {display: none;}.block>.blocktitle>.plus-fold {float: right;padding-right: 7px;cursor: pointer;}',
				Config_Ruleset: {
					'version-key': 'config-version',
					'ignores': ["LOCAL-CDN"],
					'defaultValues': {
						fold: {}
					},
					'updaters': {
						/*'config-key': [
							function() {
								// This function contains updater for config['config-key'] from v0 to v1
							},
							function() {
								// This function contains updater for config['config-key'] from v1 to v2
							}
						]*/
					}
				},
			},
			func: function() {
				const utils = require('utils');
				const CONST = FuncInfo.CONST;
				const CM = new ConfigManager(CONST.Config_Ruleset);
				const CONFIG = CM.Config;
				addStyle(CONST.CSS, 'plus-indexfolding');

				// Deal each blocks once they loaded
				detectDom({
					// Detect blocktitle so when we call dealBlock we have blocktitle loaded
					selector: '.block>.blocktitle',
					callback: blocktitle => dealBlock(blocktitle.parentElement),
					once: false
				})

				function dealBlock(block) {
					if ($(block, '.blocktitle>.plus-fold')) {
						return false;
					}

					// 'fold' class on .block
					const id = $(block, '.blocktitle').innerText;
					const blockfold = CONFIG.fold[id] || (CONFIG.fold[id] = false);

					// recorded: true - fold, false - not fold; unrecorded: undefined - not fold
					// repeating classList.add would not cause multiple same class on one elm, removing class that doesn't exist occurs no error
					block.classList[blockfold ? 'add' : 'remove']('fold');

					// Folding button
					const blocktitle = $(block, '.blocktitle');
					const button = $$CrE({
						tagName: 'span',
						props: {
							innerText: blockfold ? CONST.Text.Unfold : CONST.Text.Fold
						},
						classes: ['plus-fold'],
						styles: {
							'line-height': `${blocktitle.clientHeight}px`
						},
						listeners: [['click', e => {
							const newfold = !block.classList.contains('fold');
							block.classList[newfold ? 'add' : 'remove']('fold');
							button.innerText = newfold ? CONST.Text.Unfold : CONST.Text.Fold;
							CONFIG.fold[id] = newfold;
						}]]
					});
					$(block, '.blocktitle').appendChild(button);
				}
			}
		},

		// Review collection
		{
			name: '书评收藏',
			description: '收藏特定书评在文库首页展示',
			id: 'ReviewCollection',
			STOP: false,
			checker: {
				type: 'regurl',
				value: [
					/^https?:\/\/www\.wenku8\.(net|cc)\/modules\/article\/reviewshow\.php\/?/,
					/^https?:\/\/www\.wenku8\.(net|cc)\/index\.php([\?#][\s\S]*)?$/,
				]
			},
			CONST: {
				Text: {
					ReviewCollection: '书评收藏',
					StarButton: '加入/取消收藏此书评',
					AlertTitle: '书评收藏'
				},
				URL: {
					ReviewShow: `https://${location.host}/modules/article/reviewshow.php?rid={rid}`
				},
				Config_Ruleset: {
					'version-key': 'config-version',
					'ignores': ["LOCAL-CDN"],
					'defaultValues': {
						reviews: [{
							rid: '228884',
							title: '主题:文库导航姬'
						}],
					},
					'updaters': {
						/*'config-key': [
							function() {
								// This function contains updater for config['config-key'] from v0 to v1
							},
							function() {
								// This function contains updater for config['config-key'] from v1 to v2
							}
						]*/
					}
				},
			},
			func: function() {
				const WenkuBlockGUI = require('WenkuBlockGUI');
				const SPanel = require('SidePanel');
				const utils = require('utils');
				const CONST = FuncInfo.CONST;
				const CM = new ConfigManager(CONST.Config_Ruleset);
				const CONFIG = CM.Config;

				const functions = [{
					checker: {
						type: 'regurl',
						value: /^https?:\/\/www\.wenku8\.(net|cc)\/modules\/article\/reviewshow\.php\/?/
					},
					func: returnValue => {
						const starBtn = SPanel.insert({
							index: 3,
							faicon: `${CONFIG.reviews.some(r => r.rid === getUrlArgv('rid')) ? 'fa-solid' : 'fa-regular'} fa-star`,
							tip: CONST.Text.StarButton,
							onclick: e => {
								const rid = getUrlArgv('rid');
								const star = CONFIG.reviews.findIndex(r => r.rid === rid);
								starBtn.faicon.className = `${star >= 0 ? 'fa-regular' : 'fa-solid'} fa-star`;
								star >= 0 ? CONFIG.reviews.splice(star, 1) : CONFIG.reviews.push({
									rid, title: $('#content>table th>strong:not(table table strong)').innerText
								});
							}
						});
					}
				}, {
					checker: {
						type: 'regurl',
						value: /^https?:\/\/www\.wenku8\.(net|cc)\/index\.php([\?#][\s\S]*)?$/,
					},
					detectDom: '.main.m_foot',
					func: returnValue => {
						const block = WenkuBlockGUI.makeIndexRightToplistBlock({
							title: CONST.Text.ReviewCollection,
							links: CONFIG.reviews.map(r => ({
								url: replaceText(CONST.URL.ReviewShow, {'{rid}': r.rid}),
								text: r.title
							}))
						});
						$('#left').appendChild(block.block);

						// Drag-drop to move books
						const sortable = new Sortable.default(block.blockcontent.children[0], {
							draggable: 'li'
						});
						sortable.on('sortable:sorted', function(e) {
							const div = e.dragEvent.data.originalSource;
							const reviews = moveItem(CM.getConfig('reviews'), e.oldIndex, e.newIndex);
							CONFIG.reviews = reviews;
							/* Wrong coding:
							The following code CANNOT WORK! Because CONFIG.books[index] saves the path that
							contains index, and always gives the current value of this path in GM_storage.
							As we know, moveItem removes arr[from] and then inserts it to arr[to], and
							as moveItem inserts(using arr.splice) CONFIG.books[oldIndex] into newIndex,
							the proxy for CONFIG.books[oldIndex]'s path is currently pointing to
							the new value at CONFIG.books[oldIndex], which is the next book of the
							dragging one! Tempermonkey will look into its(the next book's) props, and save
							them into disk, at CONFIG.books[newIndex], which should be storing the dragging
							book's properties.
							To avoid this from happenning, we get the pure object using CM.getConfig, sort it
							using moveItem, and then save it to GM_storage manually, just like code above.

							The wrong code is written below:
							moveItem(CONFIG.reviews, e.oldIndex, e.newIndex);
							*/
						});
					}
				}];

				function moveItem(arr, from, to) {
					const item = arr.splice(from, 1)[0];
					arr.splice(to, 0, item);
					return arr;
				}

				return utils.loadFuncs(functions);
			}
		},

		// Account switching
		{
			name: '账号快捷切换',
			description: '顶栏快速切换账号',
			id: 'AccountSwitcher',
			STOP: false,
			global: true,
			checker: {
				type: 'switch',
				value: true
			},
			CONST: {
				Text: {
					Heading: '切换账号:',
					Empty: '未添加任何账号',
					NotLoggedin: '未登录',
					AddNewAccount: '[+] 添加账号',
					SaveAccount: `保存账号到 ${GM_info.script.name}`,
					LoginErrorTitle: '登录错误',
					LoginError: '登录错误,请检查您的网络后重试',
					AlertTitle: '账号快捷切换',
					SaveAccountWhenLogin: '登录时自动保存账号',
					AccountSetting: {
						SettingTitle: '账号管理',
						Username: '账号',
						Password: '密码(已加密,点击查看)',
					}
				},
				Data: {
					Login: '&username={username}&password={password}&usecookie={usecookie}&action=login'
				},
				URL: {
					LoginPage: `https://${location.host}/login.php?ajax_gets=jieqi_contents`,
					Login: `https://${location.host}/login.php?do=submit`
				},
				Config_Ruleset: {
					'version-key': 'config-version',
					'ignores': ["LOCAL-CDN"],
					'defaultValues': {
						saveAccounts: true,
						secret: '', // secret to encrypt & decrypt account passwords
						accounts: [/*{
							username: '用户名 明文',
							password: '密码 密文'
						}*/]
					},
					'updaters': {
						/*'config-key': [
							function() {
								// This function contains updater for config['config-key'] from v0 to v1
							},
							function() {
								// This function contains updater for config['config-key'] from v1 to v2
							}
						]*/
					}
				},
			},
			func: function() {
				const utils = require('utils');
				const CONST = FuncInfo.CONST;
				const CM = new ConfigManager(CONST.Config_Ruleset);
				const CONFIG = CM.Config;
				detectDom('.main.m_top .fr', e => work());
				if (!CONFIG.secret) {CONFIG.secret = utils.randstr()}

				function work() {
					accountSwitcher();
					accountSaver();
				}

				function accountSwitcher() {
					const curUsername = getCurUsername();
					const ACCADDKEY = `AddAccount_${utils.randstr()}`;
					const fl = $('.main.m_top .fl');
					const select = $$CrE({
						tagName: 'select',
						listeners: [['change', e => {
							if (select.value === ACCADDKEY) {
								newAccount();
								Array.from(select.children).find(opt => opt.value === curUsername).selected = true;
							} else if (select.value) {
								switchTo(select.value);
							}
						}]]
					});
					if (CONFIG.accounts.length > 0) {
						CONFIG.accounts.forEach(acc => select.appendChild($$CrE({
							tagName: 'option',
							props: {
								value: acc.username,
								innerText: acc.username,
								selected: acc.username === curUsername,
							}
						})));
						if (!curUsername) {
							select.insertAdjacentElement('afterbegin', $$CrE({
								tagName: 'option',
								props: {
									innerText: CONST.Text.NotLoggedin,
									selected: true
								}
							}));
						}
					} else {
						select.appendChild($$CrE({
							tagName: 'option',
							props: {
								innerText: CONST.Text.Empty
							}
						}));
					}
					select.appendChild($$CrE({
						tagName: 'option',
						props: {
							value: ACCADDKEY,
							innerText: CONST.Text.AddNewAccount
						}
					}));
					fl.insertAdjacentText('beforeend', CONST.Text.Heading);
					fl.appendChild(select);
				}

				// Record new account
				function newAccount() {
					utils.openDialog(CONST.URL.LoginPage);
				}

				// Switch to an exist account
				function switchTo(username) {
					const password = utils.decrypt(CONFIG.accounts.find(acc => acc.username === username).password, CONFIG.secret);

					GM_xmlhttpRequest({
						method: 'POST',
						url: CONST.URL.Login,
						headers: { "Content-Type": "application/x-www-form-urlencoded" },
						data: replaceText(CONST.Data.Login, {
							'{username}': utils.formEncode(username),
							'{password}': utils.formEncode(password),
							'{usecookie}': utils.formEncode('315360000'), // '有效期:保存一年'
						}),
						responseType: 'blob',
						onload: response => {
							utils.parseDocument(response.response, onAccountSwitched, response);
						},
						onerror: err => alertify.error(CONST.Text.LoginError),
						ontimeout: err => alertify.error(CONST.Text.LoginError)
					});
				}

				// Save account for every login form
				function accountSaver() {
					detectDom('form[name="frmlogin"]', form => {
						// Whether-to-save checkbox
						const checkbox = $$CrE({
							tagName: 'input',
							props: {
								type: 'checkbox',
								checked: CONFIG.saveAccounts
							},
							classes: ['plus-saveaccount'],
							styles: { 'vertical-align': 'middle' },
							listeners: [['change', e => {
								CONFIG.saveAccounts = checkbox.checked;
							}]]
						});
						const span = $$CrE({
							tagName: 'span',
							props: { innerText: CONST.Text.SaveAccount },
							styles: { 'vertical-align': 'middle' }
						});
						const label = $$CrE({
							tagName: 'label',
							props: { innerText: CONST.Text.SaveAccount },
							styles: { 'text-align': 'center' }
						});
						const tr = $$CrE({
							styles: {
								'text-align': 'center',
							}
						});
						label.insertAdjacentElement('afterbegin', checkbox);
						tr.appendChild(label);
						$(form, 'tbody').appendChild(tr);

						// Submit with ajax
						$AEL(form, 'submit', e => {
							e.preventDefault();

							utils.submitForm(form, function onload(response) {
								const blob = response.response;
								utils.parseDocument(blob, onAccountSwitched, response);
							}, function onerror(err) {
								alertify.error(CONST.Text.LoginError);
								Err(err);
							});
						});
					});
				}

				function onAccountSwitched(dom, response) {
					const redirector = $(dom.head, 'meta[http-equiv="refresh"]');
					if (dom.body.childElementCount === 1 && !redirector) {
						// Error page
						[...$All(dom, 'a[href^="javascript:"]')].forEach(button => {
							button.href = 'javascript: void 0;';
							$AEL(button, 'click', e => altbox.close());
						});
						const altbox = alertify.alert(CONST.Text.LoginErrorTitle, $(dom, '.block'));
					} else if (dom.body.childElementCount === 1 && redirector) {
						// Login successed
						// Set cookie for xbrowser
						if (window.mbrowser) {
							response.responseHeaders.split('\n').filter(str => str).map(str => str.split(/\s*:\s*/, 2)).filter(arr => arr[0].toLowerCase() === 'set-cookie').forEach(arr => document.cookie = arr[1]);
						}

						// Save account
						const form = $('form[name="frmlogin"]');
						if (form && $(form, '.plus-saveaccount').checked) {
							const username = $(form, 'input[name="username"]').value;
							const password = utils.encrypt($(form, 'input[name="password"]').value, CONFIG.secret);

							CONFIG.accounts.every(acc => acc.username !== username) && CONFIG.accounts.push({ username, password });
						}

						if ($('#dialog')) {
							// Login in dialog
							unsafeWindow.closeDialog();
							utils.refreshPage();
						} else {
							// Login in page or switcher
							const url = redirector.content.match(/url=([^;]+)/)[1];
							location.href = url;
						}
					} else {
						// Returned something that isn't an api page
						// Shouldn't be here
					}
				}

				function getCurUsername() {
					const match = $URL.decode(document.cookie).match(/[;,]\s*jieqiUserName=([0-9a-zA-Z]+)[;,]/);
					const curUsername = match ? match[1] : null;
					return curUsername;
				}
			},
			setting: function setter() {
				const SettingPanel = require('SettingPanel');
				const utils = require('utils');
				const CONST = FuncInfo.CONST;

				const CM = new ConfigManager(CONST.Config_Ruleset);
				const CONFIG = CM.Config;
				CM.updateAllConfigs();

				const panel = new SettingPanel.easySettings({
					title: CONST.Text.AlertTitle,
					items: [{
						text: CONST.Text.SaveAccountWhenLogin,
						path: 'saveAccounts',
						type: 'boolean'
					}]
				}, CM);

				SettingPanel.easyStorage({
					title: CONST.Text.AccountSetting.SettingTitle,
					path: 'accounts',
					key: 'username',
					panel,
					props: {
						username: {
							type: 'string',
							name: CONST.Text.AccountSetting.Username
						},
						password: {
							type: 'string',
							name: CONST.Text.AccountSetting.Password,
							styles: { cursor: 'default' },
							listeners: [['click', function(username, e) {
								if (!e.isTrusted) { return false; }
								const password = utils.decrypt(CONFIG.accounts.find(acc => acc.username === username).password, CONFIG.secret);
								this.innerText = password;
							}]],
							oncreate: function(username, block) {
								block.element.innerText = '***';
							}
						}
					},
					operations: [{type: 'delete'}]
				}, CM);
			}
		},

		// Old Config Importer
		{
			name: '旧版配置导入',
			description: '从1.x.x.x版本的旧版脚本自动导入旧配置',
			id: 'Old-Config-Importer',
			STOP: true,
			checker: {
				type: '',
				value: []
			},
			CONST: {},
			func: function() {},
			setting: function() {}
		},

		// Toy box
		{
			name: '小功能集合',
			description: '一些小功能,如未登录时www.wenku8.net自动跳转到www.wenku8.net/index.php等等',
			id: 'toybox',
			system: true,
			STOP: false,
			checker: {
				type: 'switch',
				value: true
			},
			CONST: {
				Text: {
					RunningTip: `${GM_info.script.name} 正在运行,版本 ${GM_info.script.version}`
				}
			},
			func: function() {
				const CommonStyle = require('CommonStyle');
				const utils = require('utils');
				const CONST = FuncInfo.CONST;

				const functions = [
					// https://www.wenku8.net/ 跳转到 https://www.wenku8.net/index.php
					// 通常来说网页会自动跳转,但是如果不登录就不会,所以帮未登录用户跳一下
					{
						toyid: 'IndexJump',
						checker: {
							type: 'regurl',
							value: /^https?:\/\/www\.wenku8\.(net|cc)\/?$/
						},
						func: e => location.href = `https://${location.host}/index.php`
					},

					// 没有www.开头时,跳转到www.开头的同网址
					// 无www.开头的证书对不上,www.开头的证书正常,跳转过来降安全风险,同时为其他FunctionModule提供便利
					{
						toyid: 'wwwJump',
						checker: {
							type: 'regurl',
							value: /^https?:\/\/wenku8\.(net|cc)\/?/
						},
						func: e => location.href = location.href.replace(/^https?:\/\/wenku8\.(net|cc)(\/?)/, 'https://www.wenku8.$1$2')
					},

					// http页面跳转到https(这种情况通常在旧版浏览器中出现)
					{
						toyid: 'httpsJump',
						checker: {
							type: 'regurl',
							value: /^http:\/\/www\.wenku8\.(net|cc)\/?/
						},
						func: e => location.href = location.href.replace(/^http:\/\//, 'https://')
					},

					// 首页显示一个正在运行的文字提示
					{
						toyid: 'runningText',
						checker: {
							type: 'regurl',
							value: /^https?:\/\/www\.wenku8\.(net|cc)\/index\.php([\?#][\s\S]*)?$/
						},
						detectDom: '#right',
						func: e => {
							const board = $('#centers>.block:first-of-type>.blockcontent');
							board.appendChild($CrE('br'));
							board.appendChild($$CrE({
								tagName: 'span',
								props: {
									innerText: CONST.Text.RunningTip
								},
								classes: [CommonStyle.ClassName.Text]
							}));
						}
					},

					// 兼容 haoa 的 轻小说文库下载
					{
						toyid: 'wenku8HaoaCompat',
						detectDom: 'body',
						func: e => {
							// Our script requires user's browser to be the latest, at least String.prototype.replaceAll exists
							if (String.prototype.replaceAll.toString() !== 'function replaceAll() { [native code] }') {
								const ifr = $$CrE({
									tagName: 'iframe',
									props: { srcdoc: '<html></html>' },
									styles: {
										border: '0',
										padding: '0',
										width: '0',
										height: '0',
										position: 'fixed',
										'pointer-events': 'none'
									}
								});
								document.body.appendChild(ifr);
								String.prototype.replaceAll = ifr.contentWindow.String.prototype.replaceAll;
							} else {
								const replaceAll = String.prototype.replaceAll;
								Object.defineProperty(String.prototype, 'replaceAll', {
									get: e => replaceAll,
									set: e => true
								});
							}
						}
					}
				];

				return utils.loadFuncs(functions);;
			}
		},

		// Function Module Model
		/*
		{
			name: 'XXXX',
			description: 'Xxxx',
			id: 'XxXx',
			STOP: true,
			checker: {
				type: '',
				value: []
			},
			CONST: {},
			func: function() {},
			setting: function() {}
		},
		*/
	];

	main();
	function main() {
		GMXHRHook(5);

		const FuncLoader = new FunctionLoader(Functions);
		FuncLoader.GlobalProvides = {
			GM_xmlhttpRequest, GM_getResourceText,
			LogLevel, DoLog, Err,
			$, $All, $CrE, $AEL, $$CrE, addStyle, detectDom, destroyEvent,
			copyProp, copyProps, parseArgs, escJsStr, replaceText,
			getUrlArgv, dl_browser, dl_GM,
			AsyncManager,
			ConfigManager, $URL,
			GreasyFork,
			tippy, alertify, Sortable, vkbeautify, Darkmode, CryptoJS
		};
		FuncLoader.loadAll();
	}

	function FunctionLoader(functions) {
		// function DoLog() {}
		// Arguments: level=LogLevel.Info, logContent, logger='log'
		const [LogLevel, DoLog] = (function() {
			const LogLevel = {
				None: 0,
				Error: 1,
				Success: 2,
				Warning: 3,
				Info: 4,
			};

			return [LogLevel, DoLog];
			function DoLog() {
				// Get window
				const win = (typeof(unsafeWindow) === 'object' && unsafeWindow !== null) ? unsafeWindow : window;

				const LogLevelMap = {};
				LogLevelMap[LogLevel.None] = {
					prefix: '',
					color: 'color:#ffffff'
				}
				LogLevelMap[LogLevel.Error] = {
					prefix: '[Error]',
					color: 'color:#ff0000'
				}
				LogLevelMap[LogLevel.Success] = {
					prefix: '[Success]',
					color: 'color:#00aa00'
				}
				LogLevelMap[LogLevel.Warning] = {
					prefix: '[Warning]',
					color: 'color:#ffa500'
				}
				LogLevelMap[LogLevel.Info] = {
					prefix: '[Info]',
					color: 'color:#888888'
				}
				LogLevelMap[LogLevel.Elements] = {
					prefix: '[Elements]',
					color: 'color:#000000'
				}

				// Current log level
				DoLog.logLevel = (win.isPY_DNG && win.userscriptDebugging) ? LogLevel.Info : LogLevel.Warning; // Info Warning Success Error

				// Log counter
				DoLog.logCount === undefined && (DoLog.logCount = 0);

				// Get args
				let [level, logContent, logger] = parseArgs([...arguments], [
					[2],
					[1,2],
					[1,2,3]
				], [LogLevel.Info, 'DoLog initialized.', 'log']);

				// Write storage log
				systemLog(isWenkuFunction(this) ? this.id : null, Object.keys(LogLevel)[Object.values(LogLevel).indexOf(level)], logContent, logger);

				// Log to console
				let msg = '%c' + LogLevelMap[level].prefix + (typeof GM_info === 'object' ? `[${GM_info.script.name}]` : '') + (isWenkuFunction(this) ? `[${this.id}]` : '') + (LogLevelMap[level].prefix ? ' ' : '');
				let subst = LogLevelMap[level].color;

				switch (typeof(logContent)) {
					case 'string':
						msg += '%s';
						break;
					case 'number':
						msg += '%d';
						break;
					default:
						msg += '%o';
						break;
				}

				// Log to console when log level permits
				if (level <= DoLog.logLevel) {
					if (++DoLog.logCount > 512) {
						console.clear();
						DoLog.logCount = 0;
					}
					console[logger](msg, subst, logContent);
				}
			}
		}) ();

		const FL = this;
		FL.Messager = new Messager();
		FL.DefaultProvides = {window, unsafeWindow, GM_info, require, isWenkuFunction, Messager: FL.Messager};
		FL.CodeProvides = {ConfigManager};
		FL.GrantFuncs = {
			FL_listFunctions(id) {
				return functions.map(objFunc => objFunc.id);
			},
			FL_getFunction(id) {
				const objFunc = functions.find(objFunc => objFunc.id === id);
				return objFunc ? objFunc : null;
			},
			FL_disableFunction(id) {
				const objFunc = functions.find(objFunc => objFunc.id === id);
				objFunc && CM.setConfig(`funcs/${id}/enabled`, false);
			},
			FL_enableFunction(id) {
				const objFunc = functions.find(objFunc => objFunc.id === id);
				objFunc && CM.setConfig(`funcs/${id}/enabled`, true);
			},
			FL_loadSetting(id) {
				const objFunc = functions.find(objFunc => objFunc.id === id);
				loadSetting(objFunc);
			},
			FL_getDebug() {
				return CM.getConfig('debug');
			},
			FL_exportConfig(filename='export.json') {
				const url = URL.createObjectURL(new Blob([ JSON.stringify({
					type: 'config',
					time: new Date().getTime(),
					data: CM.getConfig('funcs')
				}) ], { type: 'application/json' }));
				dl_browser(url, filename);
				setTimeout(() => URL.revokeObjectURL(url));
			},
			FL_importConfig(callback) {
				$$CrE({
					tagName: 'input',
					props: { type: 'file' },
					listeners: [['change', e => {
						const file = e.target.files[0];
						const reader = new FileReader();
						reader.onload = e => {
							let json;
							try {
								json = JSON.parse(reader.result);
							} catch(err) {
								callback(1); // File is not valid json
								return;
							}
							if (json?.type !== 'config' || !json.data) {
								callback(2); // Json is not valid config export
							}
							CM.setConfig('funcs', json.data);
							callback(0);
						};
						reader.readAsText(file);
					}]]
				}).click();
			},
			FL_postMessage(name, data=null, target=null) {
				const id = isWenkuFunction(this) ? this.id : null;
				FL.Messager.postMessage(name, data, id, target);
			},
			FL_recieveMessage(name, func, source=null, read_history=false) {
				const id = isWenkuFunction(this) ? this.id : null;
				FL.Messager.recieveMessage(name, func, source, id, read_history);
			}
		};
		FL.BindProvides = {DoLog, ...FL.GrantFuncs};
		FL.GlobalProvides = {};
		FL.returnValues = {};
		FL.load = load;
		FL.loadSetting = loadSetting;
		FL.executeAs = executeAs;
		FL.loadAll = loadAll;

		const CM = new ConfigManager(CONST.Config_Ruleset);
		const CONFIG = CM.Config;

		initStorage();
		initFunctions();
		initDebug();

		// Prepare storage structure
		function initStorage() {
			for (const objFunc of functions) {
				const userStorageKey = getUserStorageKey();
				// Function config
				!CONFIG.funcs.hasOwnProperty(objFunc.id) && (CONFIG.funcs[objFunc.id] = {
					enabled: true,
					storage: {},
				});

				// User config
				!objFunc.global && !CONFIG.funcs[objFunc.id].storage.hasOwnProperty(userStorageKey) && (CONFIG.funcs[objFunc.id].storage[userStorageKey] = {});
			}
		}

		// Pre-deal all objFunc
		/* set Symbol.toStringTag to 'Wenku8+_Function'
		   add storage object getter and setter
		*/
		function initFunctions() {
			for (const objFunc of functions) {
				deal(objFunc);
			}

			function deal(objFunc) {
				objFunc[Symbol.toStringTag] = 'Wenku8+_Function';
				for (const prop of Object.keys(CONFIG.funcs[objFunc.id])) {
					reflectProperty(CONFIG.funcs[objFunc.id], objFunc, prop);
				}
			}
		}

		// Prepare debug storage
		function initDebug() {
			const iframe = unsafeWindow !== unsafeWindow.top;
			const parent = iframe ? unsafeWindow.top.performance.timeOrigin : null;
			CONFIG.debug.push({
				timeOrigin: performance.timeOrigin,
				iframe, parent,
				console: []
			});

			// fill tab info when dom content loaded
			document.readyState === 'complete' ? saveTab() : $AEL(document, 'readystatechange', saveTab);
			function saveTab() {
				if (document.readyState === 'complete') {
					const DebugStore = CONFIG.debug.find(d => d.timeOrigin === performance.timeOrigin);
					if (DebugStore) {
						DebugStore.title = document.title;
						DebugStore.url = location.href;
					}
				}
			}

			// Catch window errors
			$AEL(unsafeWindow, 'error', function(e) {
				DoLog(LogLevel.Error, 'Global error catched');
				systemLog('window.onerror', 'Error', e.error, 'thrown error');
			});
		}

		function isWenkuFunction(obj) {
			return typeof obj === 'object' && obj !== null && obj.toString() === '[object Wenku8+_Function]';
		}

		// Load an objFunc
		function load(objFunc) {
			objFunc.STOP && Err('Cannot load an objFunc whose STOP sign is true');

			// Check if already loaded
			if (FL.returnValues.hasOwnProperty(objFunc.id)) {
				return FL.returnValues[objFunc.id];
			}

			// Call wrapper and save return value
			const return_value = executeAs(objFunc.func, objFunc);
			FL.returnValues[objFunc.id] = return_value;
			return return_value;
		}

		// Check whether an objFunc should be loaded automatically
		function check(objFunc) {
			if (!objFunc.enabled) {return false;}
			if (objFunc.STOP) {return false;}
			const checker = objFunc.checker;
			if (!checker) {return true;}
			const values = Array.isArray(checker.value) ? checker.value : [checker.value]
			return values.some(value => {
				switch (checker.type) {
					case 'regurl': {
						return !!location.href.match(value);
					}
					case 'func': {
						try {
							return value();
						} catch (err) {
							DoLog(LogLevel.Error, CONST.Text.Loader.CheckerError);
							DoLog(LogLevel.Error, err);
							return false;
						}
					}
					case 'switch': {
						return value;
					}
					case 'starturl': {
						return location.href.startsWith(value);
					}
					case 'startpath': {
						return location.pathname.startsWidth(value);
					}
					default: {
						DoLog(LogLevel.Error, CONST.Text.Loader.CheckerInvalid);
						return false;
					}
				}
			});
		}

		function loadSetting(objFunc) {
			typeof objFunc.setting === 'function' && executeAs(objFunc.setting, objFunc);
		}

		// Execute func on behalf of objFunc
		function executeAs(func, objFunc) {
			// Make wrapper
			const wrapperCode = wrapFuncCode(func);
			const storage = CM.makeSubStorage(`funcs/${objFunc.id}/storage/${objFunc.global ? '' : getUserStorageKey()}`);
			const BindProvides = bindProvides();
			const provides = {...FL.DefaultProvides, ...FL.GlobalProvides, ...BindProvides, ...storage, FuncInfo: objFunc};
			const wrapper = Function.apply(null, Object.keys(provides).concat(wrapperCode));

			// Execute
			return wrapper.apply(null, Object.values(provides));

			function wrapFuncCode(func) {
				let wrapperCode = `
					try {
						return (${func.toString()})();
					} catch (err) {
						DoLog(LogLevel.Error, \`Function ${escJsStr(objFunc.id)} error\`);
						if (unsafeWindow.isPY_DNG && unsafeWindow.userscriptDebugging) {
							// Thrown error contains error stacks in eval scope
							throw err;
						} else {
							// console.error only contains error stacks in outer(loader) scope
							DoLog(LogLevel.Error, err, 'error');
						}
					}
				`;
				for (const [pname, pfunc] of Object.entries(FL.CodeProvides)) {
					wrapperCode = `var ${pname} = ${pfunc.toString()};\n` + wrapperCode;
				}
				return wrapperCode;
			}

			function bindProvides() {
				const BindProvides = {...FL.BindProvides};
				for (const [name, func] of Object.entries(BindProvides)) {
					BindProvides[name] = func.bind(objFunc);
				}
				return BindProvides;
			}
		}

		function require(id) {
			const objFunc = functions.find(objFunc => objFunc.id === id);
			objFunc.STOP && Err('Required objFunc\'s STOP sign shouldn\'t be true');
			return load(objFunc);
		}

		// Log to storage
		function systemLog(id, level, content, logger) {
			const timestamp = (new Date()).getTime();
			content = convert(content);

			const DebugStore = CONFIG.debug.find(d => d.timeOrigin === performance.timeOrigin);
			if (DebugStore) {
				DebugStore.console.push({id, level, content, logger, timestamp});
				DebugStore.console.length > 100 && DebugStore.console.splice(0, DebugStore.console.length - 100);
				DebugStore.spliced = true;
			}

			if (CONFIG.debug.length > 10) {
				CONFIG.debug.splice(0, CONFIG.debug.length - 10);
			}

			// convert content into storage-acceptable value
			function convert(content) {
				// Errors
				if (content instanceof Error) {
					const obj = {};
					copyProps(content, obj, ['name', 'message', 'stack']);
					return obj;
				}

				// Special values
				const specialVals = [null, undefined, NaN, Infinity];
				const specialNames = ['null', 'undefined', 'NaN', 'Infinity'];
				if (specialVals.includes(content)) {
					return specialNames[specialVals.indexOf(content)];
				}

				// Other
				return content;
			}
		}

		// Check and load all objFuncs that should be loaded
		function loadAll() {
			for (const objFunc of functions) {
				if (!objFunc.STOP) {
					// Execute objFunc.alwaysRun if exist.
					/* The name "alwaysRun" is given by AI. See these pics if u'r interested (Chinese conversation)
					https://p.sda1.dev/12/b991a6513bce3ae95180d6a5503f3694/截屏2023-07-26 下午5.16.53.jpg
					https://p.sda1.dev/12/cad9eaa7b265e21367a76c028fd6208c/截屏2023-07-26 下午5.17.08.jpg
					*/
					typeof objFunc.alwaysRun === 'function' && executeAs(objFunc.alwaysRun, objFunc);

					// Execute objFunc.func if checker passed
					check(objFunc) && load(objFunc);
				}
			}
		}

		function getUserStorageKey() {
			return getUserID() || 'Guest';
		}

		function getUserID() {
			const match = $URL.decode(document.cookie).match(/jieqiUserId=(\d+)/);
			const id = match && match[1] ? Math.floor(match[1]) : null;
			return isNaN(id) ? null : id;
		}

		function defineGetter(obj, prop, getter) {
			defineProperty(obj, prop, getter, v => true);
		}

		function reflectProperty(from, to, prop) {
			defineProperty(to, prop, () => from[prop], v => {from[prop] = v});
		}

		function defineProperty(obj, prop, getter, setter) {
			Object.defineProperty(obj, prop, {
				get: getter,
				set: setter,
				configurable: false,
				enumerable: true,
			});
		}

		function Messager() {
			const M = this;
			const listeners = Object.create(null);
			const history = {};

			M.recieveMessage = recieveMessage;
			M.postMessage = postMessage;

			function recieveMessage(name, func, poster_id=null, listener_id=null, read_history=false) {
				// Register listener
				const listener = {func, poster_id, listener_id};
				if (name in listeners) {
					listeners[name].push(listener);
				} else {
					listeners[name] = [listener];
				}

				// Read history messages
				if (read_history && history[name]) {
					const messages = history[name];
					for (const message of messages) {
						const correctPoster = !poster_id || message.poster_id === poster_id;
						const correctListener = !message.listener_id || message.listener_id === listener_id;
						correctPoster && correctListener && setTimeout(e => listener.func(message.data), 0);
					}
				}
			}

			function postMessage(name, data=null, poster_id=null, listener_id=null) {
				// Post to recievers
				if (name in listeners) {
					for (const listener of listeners[name]) {
						const correctPoster = !listener.poster_id || listener.poster_id === poster_id;
						const correctListener = !listener_id || listener.listener_id === listener_id;
						correctPoster && correctListener && listener.func(data);
					}
				}

				// Save in history messages
				!history[name] && (history[name] = []);
				history[name].push({name, data, poster_id, listener_id});
			}
		}
	}
})();