/* eslint-disable no-multi-spaces */
/* eslint-disable no-useless-call */
/* eslint-disable userscripts/no-invalid-headers */
// ==UserScript==
// @name 轻小说文库+
// @namespace Wenku8+
// @version 1.7.5
// @description 轻小说文库全方位体验改善,涵盖阅读、下载、书架、推荐、书评、账号、页面个性化等各种方面,你能想到的这里都有。没有?欢迎提出你的点子。
// @updateinfo <h3>v1.7.5</h3><ul><li>支持wenku8.cc</li><li>修复书评文字缩放行间距错误地随文字缩放放大缩小问题</li></ul>
// @author PY-DNG
// @license GPL-license
// @icon 
// @match http*://www.wenku8.net/*
// @match http*://www.wenku8.cc/*
// @connect wenku8.com
// @connect wenku8.net
// @connect greasyfork.org
// @connect image.kieng.cn
// @connect sm.ms
// @connect catbox.moe
// @connect liumingye.cn
// @connect p.sda1.dev
// @connect api.pandaimg.com
// @connect imagelol.com
// @connect pic.jitudisk.com
// @connect cdn.jsdelivr.net
// @connect cdnjs.cloudflare.com
// @connect bowercdn.net
// @connect unpkg.com
// @connect cdn.bootcdn.net
// @connect kit.fontawesome.com
// @grant GM_xmlhttpRequest
// @grant GM_getValue
// @grant GM_setValue
// @grant GM_deleteValue
// @grant GM_listValues
// @grant GM_openInTab
// @grant GM_getResourceText
// @grant GM_info
// @grant unsafeWindow
// @require https://greasyfork.org/scripts/427726-gbk-url-js/code/GBK_URLjs.js?version=953098
// @require https://greasyfork.org/scripts/431490-greasyforkscriptupdate/code/GreasyForkScriptUpdate.js?version=965063
// @require https://cdnjs.cloudflare.com/ajax/libs/AlertifyJS/1.13.1/alertify.min.js
// @require https://unpkg.com/@popperjs/core@2
// @require https://unpkg.com/tippy.js@6
// @resource alertify-css https://cdnjs.cloudflare.com/ajax/libs/AlertifyJS/1.13.1/css/alertify.min.css
// @resource alertify-theme https://cdnjs.cloudflare.com/ajax/libs/AlertifyJS/1.13.1/css/themes/default.min.css
// @noframes
// ==/UserScript==
/* 需求记录 [容易(优先级高) ➡️ 困难(优先级低)(我懒,一般而言优先做低难度的)]
** [已完成]{BK}书评页提供用户书评搜索
** {BK}图片大小(最大)限制
** [已完成]{BK}回复区插入@好友
** [已完成]全卷/分卷下载:文件重命名为书名,而不是书号
** · [已完成]添加单文件下载重命名
** {BK}回复区悬浮显示
** {热忱}[已完成]修复https引用问题
** [已完成]书评打开最后一页
** [待完善]书评实时更新
** · [待完善]新回复直接添加到当前页面
** · 主动回复内容直接添加到当前页面
** [待完善]引用回复
** · [已完成]引用楼层号和回复内容
** · [已完成]仅引用楼层号
** [已完成]支持preview版tag搜索
** [高优先级]备注功能
· [待完善]用户备注
· 小说备注
· [低优先级]阅读随笔(这真的可能实现吗??)
** [待完善]书评帖子收藏
** · [已完成]书评页面收藏
** · [高优先级]收藏的书评页面可以添加编辑备注
** [已完成]每日自动推书
** [待完善]{热忱}快速切换账号
** · [已完成]为每个账号储存单独的配置
** · [待完善]保存账号信息并快速自动切换
** [待完善]快速插入图片/表情
** · [已完成]直接插入本地图片
** · [持续进行]更多图床
** · [低优先级]保存常用图片/表情链接
** [部分完成]{BK}页面美化
** · [已完成]阅读页去除广告
** · [已完成]阅读页美化
** · [已完成]书评页美化
** · …
** [高优先级][施工中]脚本储存管理界面
** [高优先级][待完善]稍后再读(可以的话,请给我提出改进建议)
** {BK}类似ehunter的阅读模式
** 改进旧代码:
** · 每个page-addon内部要按照功能分模块,执行功能靠调用模块,不能直接写功能代码
** · 共性模块要写进脚本全局作用域,可以的话写成构造函数
** [低优先级]{RC}书评:@某人时通知他
** [待完善]{BK}书评:草稿箱功能
** {热忱}{s1h2}提供带文字和插图的epub整合下载
*/
/* API记录
** 阅读API:http://dl.wenku8.com/pack.php?aid=2478&vid=92914
** 回帖API:https://www.wenku8.net/modules/article/reviewshow.php?rid=209631&aid=2751
** 查人API:https://www.wenku8.net/modules/article/reviewslist.php?keyword=136877
** 读书API:https://www.wenku8.net/modules/article/reader.php?aid=2946
** 好友API:https://www.wenku8.net/myfriends.php // 好友名称选择器:content.querySelectorAll('tr>td.odd:nth-child(1)')
** 登录API:https://www.wenku8.net/login.php?do=submit&jumpurl=http%3A%2F%2Fwww.wenku8.net%2Findex.php
** 最新回复:https://www.wenku8.net/modules/article/reviewslist.php?t=1
** 检查更新:https://greasyfork.org/zh-CN/scripts/416310/code/script.meta.js
*/
/* 账号收藏
** wenku8高仿号(按照相似度排列):
** ** https://www.wenku8.net/userpage.php?uid=912148
** ** https://www.wenku8.net/userpage.php?uid=728810
** ** https://www.wenku8.net/userpage.php?uid=917768
** BK高仿号
** ** https://www.wenku8.net/userpage.php?uid=918609
** 热忱高仿号
** ** https://www.wenku8.net/userpage.php?uid=918764
** 隐身鱼高仿号
** ** https://www.wenku8.net/userpage.php?uid=918773
*/
(function FUNC_MAIN() {
'use strict';
// Polyfills
const script_name = '轻小说文库+';
const script_version = '1.7.4.3';
const NMonkey_Info = {
GM_info: {
script: {
name: script_name,
author: 'PY-DNG',
version: script_version,
}
},
mainFunc: FUNC_MAIN,
name: 'wenku8_plus',
requires: [
// GBK-URL
{
name: 'GBK-URL',
src: 'https://greasyfork.org/scripts/427726-gbk-url-js/code/GBK_URLjs.js?version=953098',
srcset: [
'https://cdn.jsdelivr.net/gh/PYUDNG/CDN@eed1fcf0e901348bc4e752fd483bcb571ebe0408/js/GBK_URL/GBK.js',
],
loaded: () => (typeof $URL === 'object'),
execmode: 'function'
},
// GreasyForkScriptUpdate
{
name: 'GreasyForkScriptUpdate',
src: 'https://greasyfork.org/scripts/431490-greasyforkscriptupdate/code/GreasyForkScriptUpdate.js?version=965063',
srcset: [
'https://cdn.jsdelivr.net/gh/PYUDNG/CDN@94fc2bdd313f7bf2af6db5b8699effee8dd0b18d/js/ajax/GreasyForkScriptUpdate.js',
],
loaded: () => (typeof GreasyForkUpdater === 'function'),
execmode: 'eval'
},
// Alertify
{
name: 'Alertify',
src: 'https://cdnjs.cloudflare.com/ajax/libs/AlertifyJS/1.13.1/alertify.min.js',
srcset: [
'https://cdn.jsdelivr.net/gh/MohammadYounes/AlertifyJS@3151fa0d65909936afcbb2f1665ed4f20767bee5/build/alertify.min.js',
'https://bowercdn.net/c/alertify-js-1.13.1/build/alertify.min.js',
'https://cdn.bootcdn.net/ajax/libs/AlertifyJS/1.9.0/alertify.min.js',
],
loaded: () => (typeof alertify === 'object'),
execmode: 'function'
},
// FontAwesome
/*
{
src: 'https://kit.fontawesome.com/1288cd6170.js',
loaded: () => (typeof(FontAwesomeKitConfig) === 'object')
}
*/
// Tippy.js
{
name: 'Tippy.js-Core',
src: 'https://unpkg.com/@popperjs/core@2',
loaded: () => (typeof tippy === 'function'),
execmode: 'function'
},
{
name: 'Tippy.js',
src: 'https://unpkg.com/tippy.js@6',
loaded: () => (typeof tippy === 'function'),
execmode: 'function'
},
],
resources: [
// Alertify css
{
src: 'https://cdnjs.cloudflare.com/ajax/libs/AlertifyJS/1.13.1/css/alertify.min.css',
srcset: [
'https://cdn.jsdelivr.net/gh/MohammadYounes/AlertifyJS@3151fa0d65909936afcbb2f1665ed4f20767bee5/build/css/alertify.min.css',
'https://bowercdn.net/c/alertify-js-1.13.1/build/css/alertify.min.css',
'https://cdn.bootcdn.net/ajax/libs/AlertifyJS/1.9.0/css/alertify.min.css',
],
name: 'alertify-css',
isCss: true
},
// Alertify theme
{
src: 'https://cdnjs.cloudflare.com/ajax/libs/AlertifyJS/1.13.1/css/themes/default.min.css',
srcset: [
'https://cdn.jsdelivr.net/gh/MohammadYounes/AlertifyJS@3151fa0d65909936afcbb2f1665ed4f20767bee5/build/css/themes/default.min.css',
'https://bowercdn.net/c/alertify-js-1.13.1/build/css/themes/default.min.css',
'https://cdn.bootcdn.net/ajax/libs/AlertifyJS/1.9.0/css/themes/default.min.css',
],
name: 'alertify-theme',
isCss: true
},
// tooltip
/*
{
src: 'https://cdn.jsdelivr.net/gh/PYUDNG/css-components@main/build/tooltip/tooltip.css',
srcset: [
'',
],
name: 'css-tooltip',
isCss: true
},
*/
// FontAwesome
/*
{
src: 'https://cdn.jsdelivr.net/npm/@fortawesome/fontawesome-free@6.1.1/css/all.min.css',
srcset: [
'https://bowercdn.net/c/fontAwesome-6.1.1/css/all.min.css',
'https://cdnjs.cloudflare.com/ajax/libs/font-awesome/6.1.1/css/all.min.css',
],
name: 'css-fontawesome',
isCss: true
}
*/
]
};
const NMonkey_Ready = NMonkey(NMonkey_Info);
if (!NMonkey_Ready) {return false;}
polyfill_replaceAll();
// CONSTS
const NUMBER_MAX_XHR = typeof mbrowser === 'object' ? 1 : 10;
const NUMBER_LOGSUCCESS_AFTER = NUMBER_MAX_XHR * 2;
const NUMBER_ELEMENT_LOADING_WAIT_INTERVAL = 500;
const KEY_CM = 'Config-Manager';
const KEY_CM_VERSION = 'version';
const VALUE_CM_VERSION = '0.3';
const KEY_DRAFT_DRAFTS = 'comment-drafts';
const KEY_DRAFT_VERSION = 'version';
const VALUE_DRAFT_VERSION = '0.2';
const KEY_REVIEW_PREFS = 'comment-preferences';
const KEY_REVIEW_VERSION = 'version';
const VALUE_REVIEW_VERSION = '0.9';
const KEY_BOOKCASES = 'book-cases';
const KEY_BOOKCASE_VERSION = 'version';
const VALUE_BOOKCASE_VERSION = '0.5';
const KEY_ATRCMMDS = 'auto-recommends';
const KEY_ATRCMMDS_VERSION = 'version';
const VALUE_ATRCMMDS_VERSION = '0.2';
const KEY_USRDETAIL = 'user-detail';
const KEY_USRDETAIL_VERSION = 'version';
const VALUE_USRDETAIL_VERSION = '0.2';
const KEY_BEAUTIFIER = 'beautifier';
const KEY_BEAUTIFIER_VERSION = 'version';
const VALUE_BEAUTIFIER_VERSION = '0.9';
const KEY_REMARKS = 'remarks';
const KEY_REMARKS_VERSION = 'version';
const VALUE_REMARKS_VERSION = '0.1';
const KEY_USERGLOBAL = 'user-global-config';
const KEY_USERGLOBAL_VERSION = 'version';
const VALUE_USERGLOBAL_VERSION = '0.1';
const VALUE_STR_NULL = 'null';
const URL_NOVELINDEX = `https://${location.host}/book/{I}.htm`;
const URL_REVIEWSEARCH = `https://${location.host}/modules/article/reviewslist.php?keyword={K}`;
const URL_REVIEWSHOW = `https://${location.host}/modules/article/reviewshow.php?rid={R}&aid={A}&page={P}`;
const URL_REVIEWSHOW_1 = `https://${location.host}/modules/article/reviewshow.php?rid={R}`;
const URL_REVIEWSHOW_2 = `https://${location.host}/modules/article/reviewshow.php?rid={R}&page={P}`;
const URL_REVIEWSHOW_3 = `https://${location.host}/modules/article/reviewshow.php?rid={R}&aid={A}`;
const URL_REVIEWSHOW_4 = `https://${location.host}/modules/article/reviewshow.php?rid={R}&page={P}#{Y}`;
const URL_REVIEWSHOW_5 = `https://${location.host}/modules/article/reviewshow.php?rid={R}&aid={A}&page={P}#{Y}`;
const URL_USERINFO = `https://${location.host}/userinfo.php?id={K}`;
const URL_DOWNLOAD1 = `http://${location.host.replace('www.', 'dl.')}/packtxt.php?aid={A}&vid={V}&charset={C}`;
const URL_DOWNLOAD2 = `http://${location.host.replace('www.', 'dl2.')}/packtxt.php?aid={A}&vid={V}&charset={C}`;
const URL_DOWNLOAD3 = `http://${location.host.replace('www.', 'dl3.')}/packtxt.php?aid={A}&vid={V}&charset={C}`;
const URL_PACKSHOW = `https://${location.host}/modules/article/packshow.php?id={A}&type={T}`;
const URL_BOOKINTRO = `https://${location.host}/book/{A}.htm`;
const URL_ADDBOOKCASE = `https://${location.host}/modules/article/addbookcase.php?bid={A}`;
const URL_RECOMMEND = `https://${location.host}/modules/article/uservote.php?id={B}`;
const URL_TAGSEARCH = `https://${location.host}/modules/article/tags.php?t={TU}`;
const URL_USRDETAIL = `https://${location.host}/userdetail.php`;
const URL_USRFRIEND = `https://${location.host}/myfriends.php`;
const URL_BOOKCASE = `https://${location.host}/modules/article/bookcase.php`;
const URL_USRLOGIN = `https://${location.host}/login.php?do=submit&jumpurl=http%3A%2F%2F${location.host}%2Findex.php`;
const URL_USRLOGOFF = `https://${location.host}/logout.php`;
const DATA_XHR_LOGIN = [
"username={U}",
"password={P}",
"usecookie={C}",
"action=login",
"submit=%26%23160%3B%B5%C7%26%23160%3B%26%23160%3B%C2%BC%26%23160%3B" // ' 登  录 '
].join('&');
const DATA_IMAGERS = {
default: 'SDAIDEV',
/* Imager Model
_IMAGER_KEY_: {
available: true,
name: '_IMAGER_DISPLAY_NAME_',
tip: '_IMAGER_DISPLAY_TIP_',
upload: {
request: {
url: '_UPLOAD_URL_',
data: {
'_FORM_NAME_FOR_FILE_': '$file$'
}
},
response: {
checksuccess: (json)=>{return json._SUCCESS_KEY_ === '_SUCCESS_VALUE_';},
geturl: (json)=>{return json._PATH_._SUCCESS_URL_KEY_;},
getname: (json)=>{return json._PATH_ ? json._PATH_._FILENAME_ : null;},
getsize: (json)=>{return json._PATH_._SIZE_},
getpage: (json)=>{return json._PATH_ ? json._PATH_._PAGE_ : null;},
gethash: (json)=>{return json._PATH_ ? json._PATH_._HASH_ : null;},
getdelete: (json)=>{return json._PATH_ ? json._PATH_._DELETE_ : null;}
}
},
isImager: true
},
*/
LIUMINGYE: {
available: true,
name: '刘明野-全能图床',
tip: '2021-12-04测试可用</br>理论无上传大小限制,实际测试图片过大会上传失败',
upload: {
request: {
url: 'https://tool.liumingye.cn/tuchuang/update.php',
data: {
'file': '$file$'
}
},
response: {
checksuccess: (json)=>{return json.code === 0;},
geturl: (json)=>{return json.msg;}
}
},
isImager: true
},
PANDAIMG: {
available: true,
name: '熊猫图床',
tip: '2022-01-16测试可用</br>单张图片最大5MB',
upload: {
request: {
url: 'https://api.pandaimg.com/upload',
data: {
'file': '$file$',
'classifications': '',
'day': '0'
},
headers: {
'usersOrigin': '5edd88d4dfe5d288518c0454d3ccdd2a'
}
},
response: {
checksuccess: (json)=>{return json.code === '200';},
geturl: (json)=>{return json.data.url;},
getname: (json)=>{return json.data.name;}
}
},
isImager: true
},
SDAIDEV: {
available: true,
name: '流浪图床',
tip: '2022-01-09测试可用</br>单张图片最大5MB',
upload: {
request: {
url: 'https://p.sda1.dev/api/v1/upload_external_noform',
urlargs: {
'filename': '$filename$',
'ts': '$time$',
'rand': '$random$'
}
},
response: {
checksuccess: (json)=>{return json.success;},
geturl: (json)=>{return json.data.url;},
getdelete: (json)=>{return json.data ? json.data.delete_url : null;},
getsize: (json)=>{return json.data ? json.data.size : null;}
}
},
isImager: true
},
JITUDISK: {
available: true,
name: '极兔兔床',
tip: '2022-02-02测试可用',
upload: {
request: {
url: 'https://pic.jitudisk.com/api/upload',
data: {
'image': '$file$'
}
},
response: {
checksuccess: (json)=>{return json.code === 200;},
geturl: (json)=>{return json.data.url;},
getname: (json)=>{return json.data.name;}
}
},
isImager: true
},
IMAGELOL: {
available: false,
name: '笑果图床',
tip: '2022-01-17测试可用</br>该图床不支持重复上传同一张图片,请注意</br>单张图片最大2MB',
upload: {
request: {
url: 'https://imagelol.com/json',
data: {
'source': '$file$',
'type': 'file',
'action': 'upload',
'timestamp': '$time$',
'auth_token': '4f6fb8d04525bae5a455f4f09e2b09aa750e60c3',
'nsfw': '0'
}
},
response: {
checksuccess: (json)=>{return json.status_code === 200 && json.success && json.success.code === 200;},
geturl: (json)=>{return json.image.url;},
getname: (json)=>{return json.image.original_filename;},
getsize: (json)=>{return json.image.size},
gethash: (json)=>{return json.image.md5;},
}
},
isImager: true
},
/*GEJIBA: {
available: true,
name: '老王图床',
tip: '2022-01-17测试可用</br>单张图片最大10MB</br>PS:此图床审核比较严格',
upload: {
request: {
url: '_UPLOAD_URL_',
data: {
'_FORM_NAME_FOR_FILE_': '$file$'
}
},
response: {
checksuccess: (json)=>{return json._SUCCESS_KEY_ === '_SUCCESS_VALUE_';},
geturl: (json)=>{return json._PATH_._SUCCESS_URL_KEY_;},
getname: (json)=>{return json._PATH_ ? json._PATH_._FILENAME_ : null;},
getsize: (json)=>{return json._PATH_._SIZE_},
getpage: (json)=>{return json._PATH_ ? json._PATH_._PAGE_ : null;},
gethash: (json)=>{return json._PATH_ ? json._PATH_._HASH_ : null;},
getdelete: (json)=>{return json._PATH_ ? json._PATH_._DELETE_ : null;}
}
}
},*/
KIENG_JD: {
available: false,
name: 'KIENG-JD',
tip: '默认图床</br>个人体验良好,推荐使用',
upload: {
request: {
url: 'https://image.kieng.cn/upload.html?type=jd',
data: {
'image': '$file$'
}
},
response: {
checksuccess: (json)=>{return json.code === 200;},
geturl: (json)=>{return json.data.url;},
getname: (json)=>{return json.data.name;}
}
},
isImager: true
},
KIENG_SG: {
available: false,
name: 'KIENG-SG',
upload: {
request: {
url: 'https://image.kieng.cn/upload.html?type=sg',
data: {
'image': '$file$'
}
},
response: {
checksuccess: (json)=>{return json.code === 200;},
geturl: (json)=>{return json.data.url;},
getname: (json)=>{return json.data.name;}
}
},
isImager: true
},
KIENG_58: {
available: false,
name: 'KIENG-58',
upload: {
request: {
url: 'https://image.kieng.cn/upload.html?type=c58',
data: {
'image': '$file$'
}
},
response: {
checksuccess: (json)=>{return json.code === 200;},
geturl: (json)=>{return json.data.url;},
getname: (json)=>{return json.data.name;}
}
},
isImager: true
},
KIENG_WY: {
available: false,
name: 'KIENG-WY',
upload: {
request: {
url: 'https://image.kieng.cn/upload.html?type=wy',
data: {
'image': '$file$'
}
},
response: {
checksuccess: (json)=>{return json.code === 200;},
geturl: (json)=>{return json.data.url;},
getname: (json)=>{return json.data.name;}
}
},
isImager: true
},
KIENG_QQ: {
available: false,
name: 'KIENG-QQ',
upload: {
request: {
url: 'https://image.kieng.cn/upload.html?type=qq',
data: {
'image': '$file$'
}
},
response: {
checksuccess: (json)=>{return json.code === 200;},
geturl: (json)=>{return json.data.url;},
getname: (json)=>{return json.data.name;}
}
},
isImager: true
},
KIENG_SN: {
available: false,
name: 'KIENG-SN',
upload: {
request: {
url: 'https://image.kieng.cn/upload.html?type=sn',
data: {
'image': '$file$'
}
},
response: {
checksuccess: (json)=>{return json.code === 200;},
geturl: (json)=>{return json.data.url;},
getname: (json)=>{return json.data.name;}
}
},
isImager: true
},
KIENG_HL: {
available: false,
name: 'KIENG-HLX',
upload: {
request: {
url: 'https://image.kieng.cn/upload.html?type=hl',
data: {
'image': '$file$'
}
},
response: {
checksuccess: (json)=>{return json.code === 200;},
geturl: (json)=>{return json.data.url;},
getname: (json)=>{return json.data.name;}
}
},
isImager: true
},
SMMS: {
available: true,
name: 'SM.MS',
tip: '注意:此图床跨域访问较不稳定,且有用户反映其被国内部分服务商屏蔽,请谨慎使用此图床',
warning: '注意:此图床跨域访问较不稳定,且有用户反映其被国内部分服务商屏蔽,请谨慎使用此图床</br>如出现上传错误/图片加载慢/无法加载图片等情况,请更换其他图床',
upload: {
request: {
url: 'https://sm.ms/api/v2/upload?inajax=1',
data: {
'smfile': '$file$'
}
},
response: {
checksuccess: (json)=>{return json.success === true || /^https?:\/\//.test(json.images);},
geturl: (json)=>{return json.data ? json.data.url : json.images;},
getname: (json)=>{return json.data ? json.data.filename : null;},
getpage: (json)=>{return json.data ? json.data.page : null;},
gethash: (json)=>{return json.data ? json.data.hash : null;},
getdelete: (json)=>{return json.data ? json.data.delete : null;}
}
},
isImager: true
},
CATBOX: {
available: true,
name: 'CatBox',
tip: '注意:此图床访问较不稳定,请谨慎使用此图床',
warning: '注意:此图床访问较不稳定,请谨慎使用此图床</br>如出现上传错误/图片加载慢/无法加载图片等情况,请更换其他图床',
upload: {
request: {
url: 'https://catbox.moe/user/api.php',
responseType: 'text',
data: {
'fileToUpload': '$file$',
'reqtype': 'fileupload'
}
},
response: {
checksuccess: (text)=>{return true;},
geturl: (text)=>{return text;}
}
},
isImager: true
}
};
const FUNC_LATERBOOK_SORTERS = {
'addTime_old2new': {
name: '由旧到新',
sorter: (a, b) => (a.addTime - b.addTime),
},
'addTime_new2old': {
name: '由新到旧',
sorter: (a, b) => (b.addTime - a.addTime),
},
'sort': {
name: '手动排序',
sorter: (a, b) => (a.sort - b.sort),
}
}
const CLASSNAME_BUTTON = 'plus_btn';
const CLASSNAME_TEXT = 'plus_text';
const CLASSNAME_DISABLED = 'plus_disabled';
const CLASSNAME_BOOKCASE_FORM = 'plus_bcform';
const CLASSNAME_LIST = 'plus_list';
const CLASSNAME_LIST_ITEM = 'plus_list_item';
const CLASSNAME_LIST_BUTTON = 'plus_list_input';
const CLASSNAME_MODIFIED = 'plus_modified';
const HTML_BOOK_COPY = '<span class="{C}">[复制]</span>'.replace('{C}', CLASSNAME_BUTTON);
const HTML_BOOK_META = '{K}:{V}<span class="{C}">[复制]</span>'.replace('{C}', CLASSNAME_BUTTON);
const HTML_BOOK_TAG = '<a class="{C}" href="{U}" target="_blank">{TN}</span>'.replace('{C}', CLASSNAME_BUTTON).replace('{U}', URL_TAGSEARCH);
const HTML_DOWNLOAD_CONTENER = '<div id="dctn" style=\"margin:0px auto;overflow:hidden;\">\n<fieldset style=\"width:820px;height:35px;margin:0px auto;padding:0px;\">\n<legend><b>《{BOOKNAME}》小说TXT简繁全本下载</b></legend>\n</fieldset>\n</div>';
const HTML_DOWNLOAD_LINKS_OLD = '<div id="txtfull" style="margin:0px auto;overflow:hidden;"><fieldset style="width:820px;height:35px;margin:0px auto;padding:0px;"><legend><b>《{ORIBOOKNAME}》小说TXT全本下载</b></legend><div style="width:16%; float:left; text-align:center;"><a href="http://dl.wenku8.com/down.php?type=txt&id={BOOKID}">G版原始下载</a></div><div style="width:16%; float:left; text-align:center;"><a href="http://dl.wenku8.com/down.php?type=txt&id={BOOKID}&fname={BOOKNAME}.txt">G版自动重命名</a></div><div style="width:16%; float:left; text-align:center;"><a href="http://dl.wenku8.com/down.php?type=utf8&id={BOOKID}">U版原始下载</a></div><div style="width:16%; float:left; text-align:center;"><a href="http://dl.wenku8.com/down.php?type=utf8&id={BOOKID}&fname={BOOKNAME}">U版自动重命名</a></div><div style="width:16%; float:left; text-align:center;"><a href="http://dl.wenku8.com/down.php?type=big5&id={BOOKID}">繁体原始下载</a></div><div style="width:16%; float:left; text-align:center;"><a href="http://dl.wenku8.com/down.php?type=big5&id={BOOKID}&fname={BOOKNAME}">繁体自动重命名</a></div></fieldset></div>'.replaceAll('{C}', CLASSNAME_BUTTON);
const HTML_DOWNLOAD_LINKS = `<div style="margin:0px auto;overflow:hidden;"><fieldset style="width:820px;height:35px;margin:0px auto;padding:0px;"><legend><b>《{ORIBOOKNAME}》小说TXT、UMD、JAR电子书下载</b></legend><div style="width:210px; float:left; text-align:center;"><a href="https://${location.host}/modules/article/packshow.php?id={BOOKID}&type=txt{CHARSET}">TXT简繁分卷</a></div><div style="width:210px; float:left; text-align:center;"><a href="https://${location.host}/modules/article/packshow.php?id={BOOKID}&type=txtfull{CHARSET}">TXT简繁全本</a></div><div style="width:210px; float:left; text-align:center;"><a href="https://${location.host}/modules/article/packshow.php?id={BOOKID}&type=umd{CHARSET}">UMD分卷下载</a></div><div style="width:190px; float:left; text-align:center;"><a href="https://${location.host}/modules/article/packshow.php?id={BOOKID}&type=jar{CHARSET}">JAR分卷下载</a></div></fieldset></div>`;
const HTML_DOWNLOAD_BOARD = '<span class="{C}">阅读与下载限制已解除</br>此功能仅供学习交流,请支持正版<span style="text-align: right;">——{N}</span></span>'.replace('{N}', GM_info.script.name).replace('{C}', CLASSNAME_TEXT);
const CSS_DOWNLOAD = '.even {display: grid; grid-template-columns: repeat(3, 1fr); text-align: center;} .dlink {text-align: center;}';
const CSS_PAGE_API = 'body>div {display: flex; align-items: center; justify-content: center;}';
const CSS_COLOR_BTN_NORMAL = 'rgb(0, 160, 0)', CSS_COLOR_BTN_HOVER = 'rgb(0, 100, 0)', CSS_COLOR_FLOOR_MODIFIED = '#CCCCFF';
const CSS_COMMON = '.{CT} {color: rgb(30, 100, 220) !important;} .{CB} {color: rgb(0, 160, 0) !important; cursor: pointer !important; user-select: none;} .{CB}:hover {color: rgb(0, 100, 0) !important;} .{CB}:focus {color: rgb(0, 100, 0) !important;} .{CB}.{CD} {color: rgba(150, 150, 150) !important; cursor: not-allowed !important;}'.replaceAll('{CB}', CLASSNAME_BUTTON).replaceAll('{CT}', CLASSNAME_TEXT).replaceAll('{CD}', CLASSNAME_DISABLED)
+ '.{CAT}>ul {list-style: none; text-align: center; padding: 0px; margin: 0px;} .{CAT} {position: absolute; zIndex: 999; backgroundColor: #f5f5f5; float: left; clear: both; height: 180px; overflow-y: auto; overflow-x: visible;} .{CLI} {display: block; list-style: outside none none; margin: 0px; border: 1px solid rgb(204, 204, 204);} .{CLB} {border: 0px; width: 100%; height: 100%; cursor: pointer; padding: 0 0.5em;}'.replaceAll('{CAT}', CLASSNAME_LIST).replaceAll('{CLI}', CLASSNAME_LIST_ITEM).replaceAll('{CLB}', CLASSNAME_LIST_BUTTON)
+ '.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;}';
const CSS_COMMONBEAUTIFIER = '.plus_cbty_image {position: fixed;top: 0;left: 0;z-index: -2;}.plus_cbty_cover {position: fixed;top: 0;left: calc((100vw - 960px) / 2);z-Index: -1;background-color: rgba(255,255,255,0.7);width: 960px;height: 100vh;}body {overflow: auto;}body>.main {position: relative;margin-left: 0;margin-right: 0;left: calc((100vw - 960px) / 2);}body.plus_cbty table.grid td, body.plus_cbty .odd, body.plus_cbty .even, body.plus_cbty .blockcontent {background-color: rgba(255,255,255,0) !important;}.textarea, .text {background-color: rgba(255,255,255,0.9);}#headlink{background-color: rgba(255,255,255,0.7);}';
const CSS_REVIEWSHOW ='body {overflow: auto;background-image: url({BGI});}#content > table > tbody > tr > td {background-color: rgba(255,255,255,0.7) !important;overflow: auto;}body.plus_cbty #content > table > tbody > tr > td {background-color: rgba(255,255,255,0) !important;overflow: auto;}#content {height: 100vh;overflow: auto;}.m_top, .m_head, .main.nav, .m_foot {display: none;}.main {margin-top: 0px;}#content table div[style*="width:100%"]{font-size: calc(1em * {S}/ 100);line-height: 100%;}.jieqiQuote, .jieqiCode, .jieqiNote {font-size: inherit;}.{M}{background-color: {C}}'.replace('{M}', CLASSNAME_MODIFIED).replace('{C}', CSS_COLOR_FLOOR_MODIFIED);
const CSS_NOVEL = 'html{background-image: url({BGI});}body {width: 100vw;height: 100vh;overflow: overlay;margin: 0px;background-color: rgba(255,255,255,0.7);}#contentmain {overflow-y: auto;height: calc(100vh - {H});max-width: 100%;min-width: 0px;max-width: 100vw;}#adv1, #adtop, #headlink, #footlink, #adbottom {overflow: overlay;min-width: 0px;max-width: 100vw;}#adv900, #adv5 {max-width: 100vw;}';
const CSS_SIDEPANEL = '#sidepanel-panel {background-color: #00000000;z-index: 4000;}.sidepanel-button {font-size: 1vmin;color: #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>i[class^="fa-"] {line-height: 3vmin;width: 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;}';
const ARR_GUI_BOOKCASE_WIDTH = ['3%', '19%', '9%', '25%', '20%', '9%', '5%', '10%'];
const TEXT_TIP_COPY = '点击复制';
const TEXT_TIP_COPIED = '已复制';
const TEXT_TIP_SERVERCHANGE = '点击切换线路';
const TEXT_TIP_API_PACKSHOW_LOADING = '正在初始化下载页面,请稍候...';
const TEXT_TIP_API_PACKSHOW_LOADED = '初始化下载页面成功';
const TEXT_TIP_INDEX_LATERREADS = '文库首页显示前六本稍后再读书目</br>您可以在书架页面管理稍后阅读书目和调整书籍顺序';
const TEXT_TIP_SEARCH_OPTION_TAG = '有关标签搜索</br></br>未完善-开发中…</br>官方尚未正式开放此功能</br>功能预览由[轻小说文库+]提供';
const TEXT_TIP_REVIEW_BEAUTIFUL = '背景图片可以在"用户面板"中设置</br>您可以从文库首页左侧点击进入用户面板';
const TEXT_TIP_REVIEW_IMG_INSERTURL = '直接插入网络图片的链接地址';
const TEXT_TIP_REVIEW_IMG_SELECTIMG = '选择本地图片上传到第三方图床,然后再插入图床提供的图片链接</br>您也可以直接拖拽图片到输入框,或者Ctrl+V直接粘贴您剪贴板里面的图片</br>您可以在用户面板中切换图床</br></br>上传图片请遵守法律以及图床使用规定</br>请不要上传违规图片';
const TEXT_TIP_IMAGE_FIT = '请选择适合您的屏幕宽高比的图片</br>您选择的图片将会被拉伸以适应屏幕的宽高比,图片宽高比与屏幕宽高比相差过大会导致图片扭曲</br>请避免选择文件大小过大的图片,以防止浏览器卡顿';
const TEXT_TIP_IMAGER_DEFAULT = '</br></br><span class=\'{CT}\'>{N} 默认图床</span>'.replace('{N}', GM_info.script.name).replace('{CT}', CLASSNAME_TEXT);
const TEXT_TIP_DOWNLOAD_BBCODE = 'BBCODE格式:</br>即文库评论的代码格式</br>相当于引用楼层时自动填入回复框的内容</br>保存为此格式可以保留排版及多媒体信息';
const TEXT_TIP_ACCOUNT_NOACCOUNT = '没有储存的账号信息</br>请在登录页面手动登录一次,相关帐号信息就会自动储存</br></br>所有储存的账号信息都自动保存在浏览器的本地存储中';
const TEXT_ALT_SCRIPT_ERROR_AJAX_FA = 'FontAwesome加载失败(自动重试也失败了),可能会影响一部分脚本界面图标和样式的展示,但基本不会影响功能</br>您可以将此消息<a href="https://greasyfork.org/scripts/416310/feedback" class=\'{CB}\'>反馈给开发者</a>以尝试解决问题'.replace('{CB}', CLASSNAME_BUTTON);
const TEXT_ALT_DOWNLOAD_BBCODE_NOCHANGE = '帖子正在下载中,请不要更改此设置!';
const TEXT_ALT_DOWNLOADFINISH_REVIEW = '{T}({I}) 已下载完毕</br>{N} 已保存';
const TEXT_ALT_DOWNLOADIMG_CONFIRM_TITLE = '确认下载';
const TEXT_ALT_DOWNLOADIMG_CONFIRM_MESSAGE = '是否要下载 {N} 的全部插图?';
const TEXT_ALT_DOWNLOADIMG_CONFIRM_OK = '下载';
const TEXT_ALT_DOWNLOADIMG_CONFIRM_CANCEL = '取消';
const TEXT_ALT_DOWNLOADIMG_STATUS_INDEX = '正在获取小说目录...';
const TEXT_ALT_DOWNLOADIMG_STATUS_LOADING = '正在下载: {CCUR}/{CALL}';
const TEXT_ALT_DOWNLOADIMG_STATUS_FINISH = '全部插图下载完毕:)';
const TEXT_ALT_BOOK_AFTERBOOKS_ADDED = '已添加到稍后再读';
const TEXT_ALT_BOOK_AFTERBOOKS_REMOVED = '已将其从稍后再读中移除';
const TEXT_ALT_BOOKCASE_AFTERBOOKS_MISSING = '看起来这本书并不在稍后再读的列表里呢</br>是不是已经在其他的标签页里把它从稍后再读中移除了?';
const TEXT_ALT_BOOKCASE_AFTERBOOKS_V4BUG = '由于历史版本脚本的一个bug,您的<i>稍后再读</i>列表的小说排序被打乱了(非常抱歉)</br>而现在这个bug已经修复,<i>稍后再读</i>列表的小说排序也许需要您重新调整一次</br><span class="{CB}">[我知道了]</span>'.replaceAll('{CB}', CLASSNAME_BUTTON);
const TEXT_ALT_AUTOREFRESH_ON = '页面自动刷新已开启';
const TEXT_ALT_AUTOREFRESH_OFF = '页面自动刷新已关闭';
const TEXT_ALT_AUTOREFRESH_NOTLAST = '请先翻到最后一页再开启页面自动刷新</br><span class="{CB}">[点击这里翻到最后一页]</span>'.replaceAll('{CB}', CLASSNAME_BUTTON);
const TEXT_ALT_AUTOREFRESH_WORKING = '正在获取新的回复...';
const TEXT_ALT_AUTOREFRESH_NOMORE = '木有新的回复';
const TEXT_ALT_AUTOREFRESH_APPLIED = '发现了新的回复,页面已更新~</br>'.replaceAll('{CB}', CLASSNAME_BUTTON);
const TEXT_ALT_AUTOREFRESH_MODIFIED = '发现已有楼层内容变更,已对其进行了颜色标记</br>点击标记区域即可恢复原来的颜色';
const TEXT_ALT_BEAUTIFUL_ON = '页面美化已开启</br>您可能需要刷新页面使其生效';
const TEXT_ALT_BEAUTIFUL_OFF = '页面美化已关闭</br>您可能需要刷新页面使其生效';
const TEXT_ALT_FAVORITE_LAST_ON = '将在点击收藏的帖子时打开最后一页';
const TEXT_ALT_FAVORITE_LAST_OFF = '将在点击收藏的帖子时打开第一页';
const TEXT_ALT_IMAGE_FORMATERROR = '很遗憾,您选择的图片格式无法识别</br>(建议选择jpeg,png)!';
const TEXT_ALT_IMAGE_UPLOAD_WORKING = '正在上传图片…';
const TEXT_ALT_IMAGE_DOWNLOAD_WORKING = '正在下载图片…';
const TEXT_ALT_IMAGE_UPLOAD_SUCCESS = '图片上传成功!</br>文件名: {NAME}</br>URL: {URL}';
const TEXT_ALT_IMAGE_DOWNLOAD_SUCCESS = '图片下载成功!</br>已经将背景图片 {NAME} 保存在本地';
const TEXT_ALT_IMAGE_RESPONSE_NONAME = '空(服务器没有返回文件名)';
const TEXT_ALT_IMAGE_UPLOAD_ERROR = '上传错误!';
const TEXT_ALT_TEXTSCALE_CHANGED = '字体缩放已保存:{S}%';
const TEXT_ALT_CONFIG_EXPORTED = '配置文件已导出</br>文件名:{N}';
const TEXT_ALT_CONFIG_IMPORTED = '配置文件已导入';
const TEXT_ALT_IMAGER_RESET = '由于{O}已失效,您的图床已自动切换到{N}';
const TEXT_ALT_IMAGER_NOAVAILBLE = '{O}已失效';
const TEXT_ALT_META_COPIED = '{M} 已复制';
const TEXT_ALT_ATRCMMDS_SAVED = '已保存:《{B}》</br>每日自动推荐{N}次</br>每日还可推荐{R}次';
const TEXT_ALT_ATRCMMDS_INVALID = '未保存:{N}不是非负整数';
const TEXT_ALT_ATRCMMDS_OVERFLOW = '注意:</br>您的用户信息显示您每天最多推荐{V}票</br>当前您已设置每日推荐合计{C}票</br><span class="{CB}">[单击此处以立即更新您的用户信息]</span>'.replaceAll('{CB}', CLASSNAME_BUTTON);
const TEXT_ALT_ATRCMMDS_AUTO = '已开启自动推书';
const TEXT_ALT_ATRCMMDS_NOAUTO = '已关闭自动推书';
const TEXT_ALT_ATRCMMDS_ALL_START = '{S}:正在自动推书...'.replaceAll('{S}', GM_info.script.name);
const TEXT_ALT_ATRCMMDS_RUNNING = '正在推荐书目:</br>{BN}({BID})';
const TEXT_ALT_ATRCMMDS_DONE = '推荐完成:</br>{BN}({BID})';
const TEXT_ALT_ATRCMMDS_ALL_DONE = '全部书目推荐完成:</br>{R}';
const TEXT_ALT_ATRCMMDS_NOTASK = '木有要推荐的书目╮( ̄▽ ̄)╭';
const TEXT_ALT_ATRCMMDS_NOTASK_OPENBC = '您还没有设置每日自动推荐的书目╮( ̄▽ ̄)╭</br><span class="{CB}">[点击此处打开书架页面进行设置]</span>'.replaceAll('{CB}', CLASSNAME_BUTTON);
const TEXT_ALT_ATRCMMDS_NOTASK_PLSSET = '请在\'自动推书\'一栏设置每日推荐的书目及推荐次数';
const TEXT_ALT_ATRCMMDS_MAXRCMMD = '根据您的头衔,您每日一共可以推荐{V}次';
const TEXT_ALT_USRDTL_REFRESH = '{S}:正在更新用户信息({T})...'.replaceAll('{S}', GM_info.script.name).replaceAll('{T}', getTime());
const TEXT_ALT_USRDTL_REFRESHED = '{S}:用户信息已更新</br><span class="{CB}">[点此查看详细信息]</span>'.replaceAll('{S}', GM_info.script.name).replaceAll('{CB}', CLASSNAME_BUTTON);
const TEXT_ALT_POLYFILL = '<span class="{CT}">提示:正在使用移动端适配模式</span>'.replaceAll('{CT}', CLASSNAME_TEXT);
const TEXT_ALT_LASTPAGE_LOADING = '正在获取最后一页,请稍候...';
const TEXT_ALT_ACCOUNT_SWITCHED = '帐号已切换到 <i>"<span class="{CT}">{N}</span>"</i></br>3s后自动刷新页面</br><span class="{CB}">点击这里取消刷新</span>'.replaceAll('{CT}', CLASSNAME_TEXT).replaceAll('{CB}', CLASSNAME_BUTTON);
const TEXT_ALT_ACCOUNT_WORKING_LOGOFF = '正在退出当前账号...';
const TEXT_ALT_ACCOUNT_WORKING_LOGIN = '正在登录...';
const TEXT_ALT_SCRIPT_UPDATE_CHECKING = '正在检查脚本更新...';
const TEXT_ALT_SCRIPT_UPDATE_GOT = '<div class="{CT}">{SN} 有新版本啦!</br>新版本:{NV}</br>当前版本:{CV}</br><span id="script_update_info" class="{CB}">[点击此处 查看 更新]</span></br><span id="script_update_install" class="{CB}">[点击此处 安装 更新]</span></div>'.replaceAll('{CT}', CLASSNAME_TEXT).replaceAll('{CB}', CLASSNAME_BUTTON);
const TEXT_ALT_SCRIPT_UPDATE_INFO = '更新信息';
const TEXT_ALT_SCRIPT_UPDATE_NOINFO = '没有发现更新日志。。';
const TEXT_ALT_SCRIPT_UPDATE_INSTALL = '安装';
const TEXT_ALT_SCRIPT_UPDATE_CLOSE = '朕知道了';
const TEXT_ALT_SCRIPT_UPDATE_NONE = '当前已是最新版本';
const TEXT_ALT_DETAIL_IMPORTED = '配置导入成功';
const TEXT_ALT_DETAIL_CONFIG_IMPORT_ERROR_SELECT = '您选择的文件不是配置文件,请检查后再试';
const TEXT_ALT_DETAIL_CONFIG_IMPORT_ERROR_READ = '配置文件读取出错,请检查是否粘贴了正确的配置文件,以及配置文件是否损坏';
const TEXT_ALT_DETAIL_MANAGE_NOTFOUND = '该记录已不存在,您是否已经在其他标签页删除它了呢?';
const TEXT_GUI_API_ADDBOOKCASE_TOBOOKCASE = '进入书架';
const TEXT_GUI_API_ADDBOOKCASE_REMOVE = '移出本书';
const TEXT_GUI_API_PACKSHOW_TITLE_LOADING = '初始化下载界面...';
const TEXT_GUI_API_PACKSHOW_TITLE = '{N} 轻小说TXT分卷下载 - 轻小说文库';
const TEXT_GUI_UNKNOWN = '未知';
const TEXT_GUI_DOWNLOAD_THISVOLUME = '下载本卷';
const TEXT_GUI_DOWNLOAD_THISCHAPTER = '下载本章';
const TEXT_GUI_NOVEL_FILLING = '</br><span class="{CT}">[轻小说文库+] 正在获取章节内容...</span>'.replaceAll('{CT}', CLASSNAME_TEXT);
const TEXT_GUI_BOOK_IMAGESDOWNLOAD = '全部插图下载';
const TEXT_GUI_BOOK_READITLATER = '稍后再读';
const TEXT_GUI_BOOK_DONTREADLATER = '移出稍后再读';
const TEXT_GUI_REVIEW_ADDFAVORITE = '收藏本帖:';
const TEXT_GUI_REVIEW_FAVORADDED = '已收藏 {N}';
const TEXT_GUI_REVIEW_FAVORDELED = '已从收藏中移除 {N}';
const TEXT_GUI_REVIEW_BEAUTIFUL = '页面美化:';
const TEXT_GUI_REVEIW_IMG_INSERTURL = '插入网图链接';
const TEXT_GUI_REVEIW_IMG_SELECTIMG = '选择本地图片';
const TEXT_GUI_REVIEW_UNLOCK_WARNING = '<span style="color: red;">仅供测试使用,请勿滥用此功能!</span>';
const TEXT_GUI_DOWNLOAD_REVIEW = '[下载本帖(共A页)]';
const TEXT_GUI_DOWNLOADING_REVIEW = '[下载中...(C/A)]';
const TEXT_GUI_DOWNLOAD_BBCODE = '保存为BBCODE格式:';
const TEXT_GUI_DOWNLOADFINISH_REVIEW = '[下载完毕]';
const TEXT_GUI_DOWNLOADALL = '下载全部分卷,请点击右边的按钮:';
const TEXT_GUI_WAITING = ' 等待中...';
const TEXT_GUI_DOWNLOADING = ' 下载中...';
const TEXT_GUI_DOWNLOADED = ' (下载完毕)';
const TEXT_GUI_NOTHINGHERE = '<span style="color:grey">-Nothing Here-</span>';
const TEXT_GUI_SDOWNLOAD = '地址三(程序重命名)';
const TEXT_GUI_SDOWNLOAD_FILENAME = '{NovelName} {VolumeName}.{Extension}';
const TEXT_GUI_DOWNLOADING_ALL = '下载中...(C/A)';
const TEXT_GUI_DOWNLOADED_ALL = '下载图片(已完成)';
const TEXT_GUI_AUTOREFRESH = '自动更新页面:';
const TEXT_GUI_AUTOREFRESH_PAUSED = '(回复编辑中,暂停刷新)';
const TEXT_GUI_AUTOSAVE = '(您输入的内容已保存到书评草稿中)';
const TEXT_GUI_AUTOSAVE_CLEAR = '(草稿为空)';
const TEXT_GUI_AUTOSAVE_RESTORE = '(已从书评草稿中恢复了您上次编辑的内容)';
const TEXT_GUI_AREAREPLY_AT = '想用@提到谁?';
const TEXT_GUI_INDEX_FAVORITES = '收藏的书评';
const TEXT_GUI_INDEX_STATUS = '{S} 正在运行,版本 {V}。'.replace('{S}', GM_info.script.name).replace('{V}', GM_info.script.version);
const TEXT_GUI_INDEX_LATERBOOKS = '稍后再读';
const TEXT_GUI_BOOKCASE_GETTING = '正在搬运书架...(C/A)';
const TEXT_GUI_BOOKCASE_TOPTITLE = '您的书架可收藏 A 本,已收藏 B 本';
const TEXT_GUI_BOOKCASE_MOVEBOOK = '移动到 [N]';
const TEXT_GUI_BOOKCASE_DBLCLICK = '双击/长按我,给我取一个好听的名字吧~';
const TEXT_GUI_BOOKCASE_WHATNAME = '呜呜呜~会是什么名字呢?';
const TEXT_GUI_BOOKCASE_ATRCMMD = '自动推书';
const TEXT_GUI_BOOKCASE_RCMMDAT = '<span>每日自动推书:</span>';
const TEXT_GUI_BOOKCASE_RCMMDNW = '立即推书';
const TEXT_GUI_BOOKCASE_RCMMDNW_DONE = '今日推书已完成';
const TEXT_GUI_BOOKCASE_RCMMDNW_NOTYET = '今日尚未推书';
const TEXT_GUI_BOOKCASE_RCMMDNW_NOTASK = '您还没有设置自动推书';
const TEXT_GUI_BOOKCASE_RCMMDNW_CONFIRM = '今天已经推过书了,是否要再推一遍?';
const TEXT_GUI_SEARCH_OPTION_TAG = '标签(preview)';
const TEXT_GUI_DETAIL_TITLE_SETTINGS = '脚本设置';
const TEXT_GUI_DETAIL_TITLE_BGI = '页面美化背景图片';
const TEXT_GUI_DETAIL_DEFAULT_BGI = '点击选择图片 / 拖拽图片到此处 / Ctrl+V粘贴剪贴板中的图片';
const TEXT_GUI_DETAIL_BGI = '当前图片:{N}';
const TEXT_GUI_DETAIL_BGI_WORKING = '处理中...';
const TEXT_GUI_DETAIL_BGI_UPLOADING = '正在上传: {NAME}';
const TEXT_GUI_DETAIL_BGI_UPLOADFAILED = '{NAME}(上传失败,已本地保存)';
const TEXT_GUI_DETAIL_BGI_DOWNLOADING = '正在下载: {NAME}';
const TEXT_GUI_DETAIL_BGI_UPLOAD = '上传图片到图床以防止卡顿';
const TEXT_GUI_DETAIL_BGI_LEGAL = '上传图片请遵守法律以及图床使用规定</br>请不要上传违规图片';
const TEXT_GUI_DETAIL_GUI_IMAGER = '图床选择';
const TEXT_GUI_DETAIL_GUI_SCALE = '书评字体缩放';
const TEXT_GUI_DETAIL_BTF_NOVEL = '阅读页面美化';
const TEXT_GUI_DETAIL_BTF_REVIEW = '书评页面美化';
const TEXT_GUI_DETAIL_BTF_COMMON = '其他页面美化';
const TEXT_GUI_DETAIL_FVR_LASTPAGE = '点击收藏的帖子时打开最后一页';
const TEXT_GUI_DETAIL_VERSION_CURVER = '当前版本';
const TEXT_GUI_DETAIL_VERSION_CHECKUPDATE = '检查更新';
const TEXT_GUI_DETAIL_VERSION_CHECK = '点击此处检查更新';
const TEXT_GUI_DETAIL_CONFIG_EXPORT = '导出所有脚本配置到文件(包含账号密码)';
const TEXT_GUI_DETAIL_CONFIG_EXPORT_NOPASS = '导出所有脚本配置到文件(不包含账号密码)';
const TEXT_GUI_DETAIL_EXPORT_CLICK = '点击导出';
const TEXT_GUI_DETAIL_CONFIG_IMPORT = '从文件导入脚本配置';
const TEXT_GUI_DETAIL_IMPORT_CLICK = '点击导入 / 拖拽配置文件到此处 / Ctrl+V粘贴剪贴板中的配置文件,并刷新页面';
const TEXT_GUI_DETAIL_FEEDBACK_TITLE = '提出反馈';
const TEXT_GUI_DETAIL_FEEDBACK = '点击打开反馈页面';
const TEXT_GUI_DETAIL_UPDATEINFO_TITLE = '更新日志';
const TEXT_GUI_DETAIL_UPDATEINFO = '点击去主页查看';
const TEXT_GUI_DETAIL_CONFIG_MANAGE = '管理存储的信息';
const TEXT_GUI_DETAIL_CONFIG_MANAGE_EMPTY = '<span style="color:grey;">没有内容</span>';
const TEXT_GUI_DETAIL_CONFIG_MANAGE_MORE = '<span style="color:grey;">…</span>';
const TEXT_GUI_DETAIL_MANAGE_CLICK = '点击打开管理页面';
const TEXT_GUI_DETAIL_MANAGE_HEADER = '脚本储存管理';
const TEXT_GUI_DETAIL_MANAGE_FAV_NOTE_BTN_OPEN = '打开';
const TEXT_GUI_DETAIL_MANAGE_FAV_NOTE_BTN_NOTE = '备注';
const TEXT_GUI_DETAIL_MANAGE_FAV_NOTE_BTN_DELETE = '删除';
const TEXT_GUI_DETAIL_MANAGE_FAV_NOTE_TIP = '为{TITLE}设置备注: </br>备注将在主页鼠标经过此帖子收藏的链接时悬浮显示';
const TEXT_GUI_DETAIL_MANAGE_FAV_NOTE_TITLE = '编辑备注';
const TEXT_GUI_DETAIL_MANAGE_FAV_DELETE_TIP = '确认将{TITLE}移除收藏?';
const TEXT_GUI_DETAIL_MANAGE_FAV_DELETE_TITLE = '移除收藏';
const TEXT_GUI_DETAIL_MANAGE_FAV_SAVED = '已保存';
const TEXT_GUI_DETAIL_MANAGE_FAV_DELETED = '已删除';
const TEXT_GUI_DETAIL_CONFIG_IMPORT_CONFIRM_SELECT = '是否要将您粘贴的图片({N})中设置为页面美化背景图片?';
const TEXT_GUI_DETAIL_CONFIG_IMPORT_CONFIRM_PASTE = '是否要从您粘贴的配置文件({N})中导入配置?\n建议先备份您当前的配置,再导入新配置';
const TEXT_GUI_BLOCK_TITLE_DEFULT = '操作区域';
const TEXT_GUI_USER_REVIEWSEARCH = '用户书评';
const TEXT_GUI_USER_USERINFO = '详细资料';
const TEXT_GUI_USER_USERREMARKEDIT = '编辑备注';
const TEXT_GUI_USER_USERREMARKSHOW = '用户备注:';
const TEXT_GUI_USER_USERREMARKEMPTY = '假装这里有个备注';
const TEXT_GUI_USER_USERREMARKEDIT_TITLE = '编辑备注';
const TEXT_GUI_USER_USERREMARKEDIT_MSG = '设置 [{N}] 的备注为:';
const TEXT_GUI_LINK_TOLASTPAGE = '[打开尾页]';
const TEXT_GUI_ACCOUNT_SWITCH = '切换账号:';
const TEXT_GUI_ACCOUNT_CONFIRM = '是否要切换到帐号 "{N}"?';
const TEXT_GUI_ACCOUNT_NOACCOUNT = '(帐号列表为空)';
const TEXT_GUI_ACCOUNT_NOTLOGGEDIN = '(没有登录信息)';
// Emoji smiles (not used in the script yet)
const SmList =
[{text:"/:O",id:"1",alt:"惊讶"}, {text:"/:~",id:"2",alt:"撇嘴"}, {text:"/:*",id:"3",alt:"色色"},
{text:"/:|",id:"4",alt:"发呆"}, {text:"/8-)",id:"5",alt:"得意"}, {text:"/:LL",id:"6",alt:"流泪"},
{text:"/:$",id:"7",alt:"害羞"}, {text:"/:X",id:"8",alt:"闭嘴"}, {text:"/:Z",id:"9",alt:"睡觉"},
{text:"/:`(",id:"10",alt:"大哭"}, {text:"/:-",id:"11",alt:"尴尬"}, {text:"/:@",id:"12",alt:"发怒"},
{text:"/:P",id:"13",alt:"调皮"}, {text:"/:D",id:"14",alt:"呲牙"}, {text:"/:)",id:"15",alt:"微笑"},
{text:"/:(",id:"16",alt:"难过"}, {text:"/:+",id:"17",alt:"耍酷"}, {text:"/:#",id:"18",alt:"禁言"},
{text:"/:Q",id:"19",alt:"抓狂"}, {text:"/:T",id:"20",alt:"呕吐"}]
/* \t
┌┬┐┌─┐┏┳┓┏━┓╭─╮
├┼┤│┼│┣╋┫┃╋┃│╳│
└┴┘└─┘┗┻┛┗━┛╰─╯
╲╱╭╮
╱╲╰╯
*/
/* **output format: Review Name.txt**
** 轻小说文库-帖子 [ID: reviewid]
** title
** 保存自: reviewlink
** 保存时间: savetime
** By scriptname Ver. version, author authorname
**
** ──────────────────────────────
** [用户: username userid]
** 用户名: username
** 用户ID: userid
** 加入日期: 1970-01-01
** 用户链接: userlink
** 最早出现: 1楼
** ──────────────────────────────
** ...
** ──────────────────────────────
** [#1 2021-04-26 17:53:49] [username userid]
** ──────────────────────────────
** content - line 1
** content - line 2
** content - line 3
** ──────────────────────────────
**
** ──────────────────────────────
** [#2 2021-04-26 19:28:08] [username userid]
** ──────────────────────────────
** content - line 1
** content - line 2
** content - line 3
** ──────────────────────────────
**
** ...
**
**
** [THE END]
*/
const TEXT_SPLIT_LINE_CHAR = '━'; const TEXT_SPLIT_LINE = TEXT_SPLIT_LINE_CHAR.repeat(20)
const TEXT_OUTPUT_REVIEW_HEAD =
'轻小说文库-帖子 [ID: {RWID}]\n{RWTT}\n保存自: {RWLK}\n保存时间: {SVTM}\nBy {SCNM} Ver. {VRSN}, author {ATNM}'
const TEXT_OUTPUT_REVIEW_USER =
'{LNSPLT}\n[用户: {USERNM} {USERID}]\n用户名: {USERNM}\n用户ID: {USERID}\n加入日期: {USERJT}\n用户链接: {USERLK}\n最早出现: {USERFL}楼\n{LNSPLT}'
const TEXT_OUTPUT_REVIEW_FLOOR =
'{LNSPLT}\n[#{RPNUMB} {RPTIME}] [{USERNM} {USERID}]\n{LNSPLT}\n{RPTEXT}\n{LNSPLT}';
const TEXT_OUTPUT_REVIEW_END = '\n[THE END]';
// Arguments: level=LogLevel.Info, logContent, asObject=false
// Needs one call "DoLog();" to get it initialized before using it!
function DoLog() {
// Global log levels set
unsafeWindow.LogLevel = {
None: 0,
Error: 1,
Success: 2,
Warning: 3,
Info: 4,
}
unsafeWindow.LogLevelMap = {};
unsafeWindow.LogLevelMap[LogLevel.None] = {prefix: '' , color: 'color:#ffffff'}
unsafeWindow.LogLevelMap[LogLevel.Error] = {prefix: '[Error]' , color: 'color:#ff0000'}
unsafeWindow.LogLevelMap[LogLevel.Success] = {prefix: '[Success]' , color: 'color:#00aa00'}
unsafeWindow.LogLevelMap[LogLevel.Warning] = {prefix: '[Warning]' , color: 'color:#ffa500'}
unsafeWindow.LogLevelMap[LogLevel.Info] = {prefix: '[Info]' , color: 'color:#888888'}
unsafeWindow.LogLevelMap[LogLevel.Elements] = {prefix: '[Elements]', color: 'color:#000000'}
// Current log level
DoLog.logLevel = (unsafeWindow ? unsafeWindow.isPY_DNG : window.isPY_DNG) ? LogLevel.Info : LogLevel.Warning; // Info Warning Success Error
// Log counter
DoLog.logCount === undefined && (DoLog.logCount = 0);
if (++DoLog.logCount > 512) {
console.clear();
DoLog.logCount = 0;
}
// Get args
let level, logContent, asObject;
switch (arguments.length) {
case 1:
level = LogLevel.Info;
logContent = arguments[0];
asObject = false;
break;
case 2:
level = arguments[0];
logContent = arguments[1];
asObject = false;
break;
case 3:
level = arguments[0];
logContent = arguments[1];
asObject = arguments[2];
break;
default:
level = LogLevel.Info;
logContent = 'DoLog initialized.';
asObject = false;
break;
}
// Log when log level permits
if (level <= DoLog.logLevel) {
let msg = '%c' + LogLevelMap[level].prefix;
let subst = LogLevelMap[level].color;
if (asObject) {
msg += ' %o';
} else {
switch(typeof(logContent)) {
case 'string': msg += ' %s'; break;
case 'number': msg += ' %d'; break;
case 'object': msg += ' %o'; break;
}
}
console.log(msg, subst, logContent);
}
}
DoLog();
let tipready, CONFIG, TASK, DMode, SPanel, AndAPI
let API
main();
// Main
function main() {
// Get tab url api part
API = window.location.href.replace(/https?:\/\/www\.wenku8\.(net|cc)\//, '').replace(/\?.*/, '').replace(/#.*/, '')
.replace(/^book\/\d+\.html?/, 'book').replace(/novel\/(\d+\/?)+\.html?$/, 'novel')
.replace(/^novel[\/\d]+index\.html?$/, 'novelindex');
// Common actions
loadinResourceCSS();
loadinFontAwesome();
polyfillAlert();
tipready = tipcheck();
tipscroll();
addStyle(CSS_COMMON);
GMXHRHook(NUMBER_MAX_XHR);
CONFIG = new configManager();
TASK = new taskManager();
AndAPI = new AndroidAPI();
//DMode = new Darkmode({autoMatchOsTheme: false});
formSearch();
linkReview();
multiAccount();
commonBeautify(API);
SPanel = sideFunctions();
unsafeWindow.alertify = alertify;
alertify.set('notifier','position', 'top-right');
if (isAPIPage()) {
if (!pageAPI(API)) {
return;
}
}
if (!API) {
location.href = `https://${location.host}/index.php`;
return;
};
switch (API) {
// Dwonload page
case 'modules/article/packshow.php':
pageDownload();
break;
// ReviewList page
case 'modules/article/reviews.php':
areaReply();
break;
// Review page
case 'modules/article/reviewshow.php':
areaReply();
pageReview();
break;
// ReviewEdit page
case 'modules/article/reviewedit.php':
areaReply();
pageReviewedit();
break;
// Bookcase page
case 'modules/article/bookcase.php':
pageBookcase();
break;
// Tags page
case 'modules/article/tags.php':
pageTags();
break;
// Mylink page
case 'mylink.php':
pageMylink();
break;
case 'userpage.php':
pageUser();
break;
// Detail page
case 'userdetail.php':
pageDetail();
break;
// Index page
case 'index.php':
pageIndex();
break;
// Book page
// Also: https://www.wenku8.net/modules/article/articleinfo.php?id={ID}&charset=gbk
case 'modules/article/articleinfo.php':
case 'book':
pageBook();
break;
// Novel index page
case 'novelindex':
pageNovelIndex();
break;
// Novel page
case 'novel':
pageNovel();
break;
// Novel index page & novel page
case 'modules/article/reader.php':
chapter_id === '0' ? pageNovelIndex() : pageNovel();
break;
// Login page
case 'login.php':
pageLogin();
break;
// Other pages
default:
DoLog(LogLevel.Info, API);
}
}
// Autorun tasks
// use 'new' keyword
function taskManager() {
const TM = this;
// UserDetail refresh
TM.UserDetail = {
// Refresh userDetail storage everyday
refresh: function() {
// Time check: whether recommend has done today
if (getMyUserDetail().lasttime === getTime('-', false)) {return false;};
refreshMyUserDetail();
}
}
// Auto-recommend
TM.AutoRecommend = {
// Check if recommend has done
checkRcmmd: function() {
const arConfig = CONFIG.AutoRecommend.getConfig();
return arConfig.lasttime === getTime('-', false);
},
// Auto recommend main function
run: function(recommendAnyway=false) {
let i;
// Get config
const arConfig = CONFIG.AutoRecommend.getConfig();
// Time check: whether all recommends has done today
if (TM.AutoRecommend.checkRcmmd() && !recommendAnyway) {return false;};
// Config check: whether we need to auto-recommend
if (!arConfig.auto && !recommendAnyway) {return false;}
// Config check: whether the recommend list is empty
if (arConfig.allCount === 0) {
const altBox = alertify.notify(
/modules\/article\/bookcase\.php$/.test(location.href) ?
TEXT_ALT_ATRCMMDS_NOTASK_PLSSET + (getMyUserDetail().userDetail ? '</br>'+TEXT_ALT_ATRCMMDS_MAXRCMMD.replace('{V}', String(getMyUserDetail().userDetail.vote)) : '') :
TEXT_ALT_ATRCMMDS_NOTASK_OPENBC
);
altBox.callback = (isClicked) => {
isClicked && window.open(URL_BOOKCASE);
}
return false;
};
// Recommend for each
let recommended = {}, AM = new AsyncManager();
AM.onfinish = allFinish;
alertify.notify(TEXT_ALT_ATRCMMDS_ALL_START);
for (const strBookID in arConfig.books) {
// Only when inherited properties exists must we use hasOwnProperty()
// here we know there is no inherited properties
const book = arConfig.books[strBookID]
const number = book.number;
const bookID = book.id;
const bookName = book.name;
// Time check: whether this book's recommend has done today
if (book.lasttime === getTime('-', false) && !recommendAnyway) {continue;};
// Soft alert
//alertify.notify(TEXT_ALT_ATRCMMDS_RUNNING.replaceAll('{BN}', bookName).replaceAll('{BID}', strBookID));
// Go work
for (i = 0; i < number; i++) {
AM.add();
getDocument(URL_RECOMMEND.replaceAll('{B}', strBookID), bookFinish,[book, strBookID, bookName]);
}
// Soft alert
//alertify.notify(TEXT_ALT_ATRCMMDS_DONE.replaceAll('{BN}', bookName).replaceAll('{BID}', strBookID));
}
AM.finishEvent = true;
return true;
function bookFinish(oDoc, book, strBookID, bookName) {
// title: "处理成功"
const statusText = $(oDoc, '.blocktitle').innerText;
// success: "我们已经记录了本次推荐,感谢您的参与!\n\n您每天拥有 5 次推荐权利,这是您今天第 1 次推荐。"
// overflow: "\n错误原因:对不起,您今天已经用完了推荐的权利!\n\n您每天可以推荐 20 次。\n\n请 返 回 并修正"
const returnText = $(oDoc, '.blockcontent').innerText.replace(/\s*\[.+\]\s*$/, '');
// Save book
book.lasttime = getTime('-', false);
CONFIG.AutoRecommend.saveConfig(arConfig);
// Log
DoLog(statusText + '\n' + returnText);
/*
// Check status
const success = /我们已经记录了本次推荐,感谢您的参与!\s*您每天拥有\s*(\d+)\s*次推荐权利,这是您今天第\s*(\d+)\s*次推荐。/;
const overflow = /\s*错误原因:对不起,您今天已经用完了推荐的权利!\s*您每天可以推荐\s*(\d+)\s*次。\s*请\s*返\s*回\s*并修正/;
*/
const b = recommended[strBookID] = recommended[strBookID] || {name: bookName, strID: strBookID, count: 0};
b.count++;
AM.finish();
}
function allFinish() {
// Save config
arConfig.lasttime = getTime('-', false);
CONFIG.AutoRecommend.saveConfig(arConfig);
// Soft alert
let text = [];
for (const strBookID of Object.keys(recommended)) {
const book = recommended[strBookID];
text.push('[{BID}]{BN} 推荐了{C}次'.replaceAll('{C}', book.count).replaceAll('{BID}', book.strID).replaceAll('{BN}', book.name));
}
alertify.success(TEXT_ALT_ATRCMMDS_ALL_DONE.replaceAll('{R}', text.join('</br>')));
}
}
}
// Config Maintainer
TM.Cleaner = {
cleanPageStatus: function() {
const config = CONFIG.BkReviewPrefs.getConfig();
const history = config.history;
let count = 0;
for (const [rid, his] of Object.entries(history)) {
if (!his.time || (new Date()).getTime() - his.time > 30*1000) {
delete history[rid];
count++;
}
}
CONFIG.BkReviewPrefs.saveConfig(config);
DoLog(count > 0 ? LogLevel.Success : LogLevel.Info, 'Review page status cleaned ({C})'.replace('{C}', count.toString()));
},
imagerFix: function() {
const config = CONFIG.UserGlobalCfg.getConfig();
const curimager = config.imager;
// If imager does not exist or imager disabled, change it to default
if (!DATA_IMAGERS[curimager] || !DATA_IMAGERS[curimager].available) {
DoLog(LogLevel.Warning, 'Current imager unavailable, changing to default.');
if (curimager !== DATA_IMAGERS.default && DATA_IMAGERS[DATA_IMAGERS.default].available) {
// Default available
config.imager = DATA_IMAGERS.default;
DoLog(LogLevel.Success, 'Changed to default.');
} else {
// Default not available
DoLog(LogLevel.Warning, 'Default imager unavailable, trying to find another imager for use. ')
for (const [key, imager] of Object.entries(DATA_IMAGERS)) {
if (imager.available) {
config.imager = key;
DoLog(LogLevel.Success, 'Changed to {K}.'.replace('{K}', key));
break;
}
}
if (config.imager === curimager) {
// OMG, There's NO IMAGER AVAILABLE!!
DoLog(LogLevel.Error, 'OMG, There\'s NO IMAGER AVAILABLE!!');
}
}
CONFIG.UserGlobalCfg.saveConfig(config);
alertify.warning((config.imager !== curimager ? TEXT_ALT_IMAGER_RESET : TEXT_ALT_IMAGER_NOAVAILBLE).replace('{O}', DATA_IMAGERS[curimager].name).replace('{N}', DATA_IMAGERS[config.imager].name));
}
},
}
// Script
TM.Script = {
// Check & Update to latest version of script
update: function(force=false) {
// Check for update once a day
const scriptID = 416310;
const config = CONFIG.GlobalConfig.getConfig();
if (!force && config.scriptUpdate.lasttime === getTime('-', false)) {return false;}
const GFU = new GreasyForkUpdater();
alertify.notify(TEXT_ALT_SCRIPT_UPDATE_CHECKING);
GFU.checkUpdate(scriptID, GM_info.script.version, function(update, updateurl, metaData) {
if (update) {
const box = alertify.notify(TEXT_ALT_SCRIPT_UPDATE_GOT.replaceAll('{SN}', metaData.name).replaceAll('{NV}', metaData.version).replaceAll('{CV}', GM_info.script.version));
const btnInfo = $(box.element, '#script_update_info');
const btnInstall = $(box.element, '#script_update_install');
btnInfo.addEventListener('click', show);
btnInstall.addEventListener('click', install);
} else {
alertify.message(TEXT_ALT_SCRIPT_UPDATE_NONE);
}
config.scriptUpdate.lasttime = getTime('-', false);
CONFIG.GlobalConfig.saveConfig(config);
function install(e) {
location.href = updateurl;
}
function show(e) {
const info = metaData.updateinfo;
const box = alertify.confirm(info ? info : TEXT_ALT_SCRIPT_UPDATE_NOINFO, install);
box.setHeader(TEXT_ALT_SCRIPT_UPDATE_INFO);
box.set('labels', {ok: TEXT_ALT_SCRIPT_UPDATE_INSTALL, cancel: TEXT_ALT_SCRIPT_UPDATE_CLOSE});
box.set('overflow', true);
}
});
return true;
}
}
TM.Script.update();
TM.Cleaner.cleanPageStatus();
TM.Cleaner.imagerFix();
TM.UserDetail.refresh();
TM.AutoRecommend.run();
}
// Config Manager
// use 'new' keyword
function configManager() {
const CM = this;
const [getValue, setValue, deleteValue, listValues] = [
window.getValue ? window.getValue : GM_getValue,
window.setValue ? window.setValue : GM_setValue,
window.deleteValue ? window.deleteValue : GM_deleteValue,
window.listValues ? window.listValues : GM_listValues,
]
CM.GlobalConfig = {
saveConfig: function(config) {
config ? config[KEY_CM_VERSION] = VALUE_CM_VERSION : function() {};
setValue(KEY_CM, config);
},
initConfig: function(save=true, func) {
let config = {
users: {},
scriptUpdate: {
lasttime: ''
}
};
config = func ? func(config) : config;
save ? CM.GlobalConfig.saveConfig(config) : function() {};
return config;
},
getConfig: function(init) {
let config = getValue(KEY_CM, null);
config = config ? config : (init ? CM.GlobalConfig.initConfig(true, init) : CM.GlobalConfig.initConfig());
return config;
},
// Review config upgrade (Uses GM_functions)
upgradeConfig: function() {
// Get version
const default_self = {}; default_self[KEY_CM_VERSION] = '0.1'; // v0.1 has no self object
const self = GM_getValue(KEY_CM, default_self);
const version = self[KEY_CM_VERSION];
// Upgrade by version
if (self[KEY_CM_VERSION] === VALUE_CM_VERSION) {DoLog(LogLevel.Info, 'Config Manager self config is in latest version. ');};
switch(version) {
case '0.1':
v01_To_v02();
v02_To_v03();
logUpgrade();
break;
case '0.2':
v02_To_v03();
logUpgrade();
break;
}
// Save to global gm_storage
self[KEY_CM_VERSION] = VALUE_CM_VERSION;
setValue(KEY_CM, self);
function logUpgrade() {
DoLog(LogLevel.Success, 'Config Manager self config successfully upgraded From v{V1} to v{V2}. '.replaceAll('{V1}', version).replaceAll('{V2}', VALUE_CM_VERSION));
}
function v01_To_v02() {
const props = GM_listValues();
const userStorage = {};
for (const prop of props) {
userStorage[prop] = GM_getValue(prop);
}
const userID = getUserID();
userID ? GM_setValue(userID, userStorage) : GM_setValue('temp', userStorage);
for (const prop of props) {
GM_deleteValue(prop);
}
}
function v02_To_v03() {
self.scriptUpdate = self.scriptUpdate ? self.scriptUpdate : {lasttime: ''};
}
},
// Redirect global gm_storage to user's storage area (Uses GM_functions)
// callback(key)
redirectToUser: function (callback) {
// Get userID from cookies
const userID = getUserID();
if (userID) {
// delete temp data if exist
GM_deleteValue('temp');
// Save lastUserID
const config = CM.GlobalConfig.getConfig();
config.lastUserID = userID;
CM.GlobalConfig.saveConfig(config);
// Redirect to user storage area
redirectGMStorage(userID);
DoLog(LogLevel.Info, 'GM_storage redirected to ' + String(userID));
} else {
// Redirect to temp storage area before request finish
const lastUserID = CM.GlobalConfig.getConfig().lastUserID;
redirectTemp(lastUserID);
// Request userID
getMyUserDetail((userDetail)=>{
const key = userDetail.userDetail.userID;
// Move temp data to user storage area
redirectGMStorage();
const tempStorage = GM_getValue('temp');
GM_setValue(lastUserID ? lastUserID : key, tempStorage);
GM_deleteValue('temp');
// Save lastUserID
const config = CM.GlobalConfig.getConfig();
config.lastUserID = key;
CM.GlobalConfig.saveConfig(config);
// Redirect to user storage area
redirectGMStorage(key);
DoLog(LogLevel.Info, 'GM_storage redirected to ' + String(key));
// callback
callback ? callback(key) : function() {};
})
}
// When userID request not finished, use 'temp' as gm_storage key
function redirectTemp(lastUserID) {
if (lastUserID) {
// Copy config of the user we use last time to 'temp' storage area
const lastUser = GM_getValue(lastUserID, {});
GM_setValue('temp', lastUser);
}
redirectGMStorage('temp');
DoLog(LogLevel.Info, 'GM_storage redirected to temp');
}
}
}
CM.GlobalConfig.upgradeConfig();
CM.GlobalConfig.redirectToUser();
CM.AutoRecommend = {
saveConfig: function(config) {
config ? config[KEY_ATRCMMDS_VERSION] = VALUE_ATRCMMDS_VERSION : function() {};
GM_setValue(KEY_ATRCMMDS, config);
},
initConfig: function(save=true, func) {
let config = {};
config[KEY_ATRCMMDS_VERSION] = VALUE_ATRCMMDS_VERSION;
config.allCount = 0;
config.books = {};
config.auto = true;
config = func ? func(config) : config;
save ? CM.AutoRecommend.saveConfig(config) : function() {};
return config;
},
getConfig: function(init) {
let config = GM_getValue(KEY_ATRCMMDS, null);
config = config ? config : (init ? CM.AutoRecommend.initConfig(true, init) : CM.AutoRecommend.initConfig());
return config;
},
// Auto-recommend config upgrade
upgradeConfig: function() {
// Get config
const config = CM.AutoRecommend.getConfig();
// if not inited
if (!config) {return;};
switch (config[KEY_ATRCMMDS_VERSION]) {
case '0.1':
config.auto = true;
logUpgrade();
break;
case VALUE_ATRCMMDS_VERSION:
DoLog(LogLevel.Info, 'Auto-recommend config is in latest version. ');
break;
default:
DoLog(LogLevel.Error, 'configCheckUpgrade: Invalid config version({V}) for Auto-recommend. '.replace('{V}', config[KEY_ATRCMMDS_VERSION]));
}
// Save to gm_storage
CM.AutoRecommend.saveConfig(config);
function logUpgrade() {
DoLog(LogLevel.Success, 'Auto-recommend config successfully upgraded From v{V1} to {V2}. '.replaceAll('{V1}', config[KEY_ATRCMMDS_VERSION]).replaceAll('{V2}', VALUE_ATRCMMDS_VERSION));
}
}
}
CM.commentDrafts = {
saveConfig: function(config) {
config ? config[KEY_DRAFT_VERSION] = VALUE_DRAFT_VERSION : function() {};
GM_setValue(KEY_DRAFT_DRAFTS, config);
},
initConfig: function(save=true, func) {
let config = {};
config = func ? func(config) : config;
save ? CM.commentDrafts.saveConfig(config) : function() {};
return config;
},
getConfig: function(init) {
let config = GM_getValue(KEY_DRAFT_DRAFTS, null);
config = config ? config : (init ? CM.commentDrafts.initConfig(true, init) : CM.commentDrafts.initConfig());
return config;
},
// Comment-drafts config upgrade
upgradeConfig: function() {
// Get config
let config = CM.commentDrafts.getConfig();
// if not inited
if (!config) {return;};
switch (config[KEY_DRAFT_VERSION]) {
case '0.1':
case undefined:
v01_To_v02();
logUpgrade();
break;
case VALUE_DRAFT_VERSION:
DoLog(LogLevel.Info, 'comment-drafts config is in latest version. ');
break;
default:
DoLog(LogLevel.Error, 'configCheckUpgrade: Invalid config version({V}) for comment-drafts. '.replace('{V}', config[KEY_DRAFT_VERSION]));
}
// Save to gm_storage
CM.commentDrafts.saveConfig(config);
function logUpgrade() {
DoLog(LogLevel.Success, 'comment-drafts config successfully upgraded From v{V1} to {V2}. '.replaceAll('{V1}', config[KEY_DRAFT_VERSION]).replaceAll('{V2}', VALUE_DRAFT_VERSION));
}
function v01_To_v02() {
// Fix bug caused bookcase's config overwriting comment-drafts' config
if (config instanceof Array) {
config = {};
}
}
}
}
CM.bookcasePrefs = {
saveConfig: function(config) {
config ? config[KEY_BOOKCASE_VERSION] = VALUE_BOOKCASE_VERSION : function() {};
GM_setValue(KEY_BOOKCASES, config);
},
initConfig: function(save=true, func) {
let config = {
bookcases: [],
laterbooks: {
sortby: 'addTime_old2new',
books: {}
}
};
config = func ? func(config) : config;
save ? CM.bookcasePrefs.saveConfig(config) : function() {};
return config;
},
getConfig: function(init) {
let config = GM_getValue(KEY_BOOKCASES, null);
config = config ? config : (init ? CM.bookcasePrefs.initConfig(true, init) : CM.bookcasePrefs.initConfig());
return config;
},
// Bookcase config upgrade
upgradeConfig: function() {
// Get config
let config = CM.bookcasePrefs.getConfig();
// if not inited
if (!config) {return;};
// Original version
let V = config && config[KEY_BOOKCASE_VERSION] ? config[KEY_BOOKCASE_VERSION] : '0';
switch (V) {
case '0.1':
case undefined:
case '0':
v01_To_v02();
v02_To_v03();
v03_To_v04();
v04_To_v05();
logUpgrade();
break;
case '0.2':
v02_To_v03();
v03_To_v04();
v04_To_v05();
logUpgrade();
break;
case '0.3':
v03_To_v04();
v04_To_v05();
logUpgrade();
break;
case '0.4':
v04_To_v05();
logUpgrade();
break;
case VALUE_BOOKCASE_VERSION:
DoLog(LogLevel.Info, 'Bookcase config is in latest version. ');
break;
default:
DoLog(LogLevel.Error, 'configCheckUpgrade: Invalid config version({V}) for Bookcase. '.replace('{V}', config[KEY_BOOKCASE_VERSION]));
}
// Save to gm_storage
CM.bookcasePrefs.saveConfig(config);
function logUpgrade() {
DoLog(LogLevel.Success, 'Bookcase config successfully upgraded From v{V1} to v{V2}. '.replaceAll('{V1}', V).replaceAll('{V2}', VALUE_BOOKCASE_VERSION));
}
function v01_To_v02() {
// Clear useless key added falsely
delete config.bbcode;
// Convert array to an object
if (Array.isArray(config)) {
const newConfig = {bookcases: []};
for (let i = 0; i < config.length; i++) {
newConfig.bookcases[i] = config[i];
}
config = newConfig;
}
}
function v02_To_v03() {
// Fix bug caused config.bookcases equals to []
if (config && config.bookcases && config.bookcases.length === 0) {
config = CM.bookcasePrefs.initConfig();
}
}
function v03_To_v04() {
if (config.laterbooks) {return false;}
config.laterbooks = {
sortby: 'addTime_old2new',
books: {}
};
}
function v04_To_v05() {
const books = config.laterbooks.books;
const sorts = [];
let err = false;
for (const book of Object.values(books)) {
if (sorts.includes(book.sort)) {
err = true;
break;
}
sorts.push(book.sort);
}
Math.max.apply(null, sorts) > books.length && (err = true);
if (err) {
let i = 0;
for (const book of Object.values(books)) {
book.sort = ++i;
}
alertify.notify(TEXT_ALT_BOOKCASE_AFTERBOOKS_V4BUG, '', 0);
}
}
}
}
CM.userDtlePrefs = {
saveConfig: function(config) {
config ? config[KEY_USRDETAIL_VERSION] = VALUE_USRDETAIL_VERSION : function() {};
GM_setValue(KEY_USRDETAIL, config);
},
initConfig: function(save=true, func) {
let config = {userDetail: null};
config = func ? func(config) : config;
save ? CM.userDtlePrefs.saveConfig(config) : function() {};
return config;
},
getConfig: function(init) {
let config = GM_getValue(KEY_USRDETAIL, null);
config = config ? config : (init ? CM.userDtlePrefs.initConfig(true, init) : CM.userDtlePrefs.initConfig());
return config;
},
// userDetail config upgrade
upgradeConfig: function() {
// Get config
const config = CM.userDtlePrefs.getConfig();
// if not inited
if (!config) {return;};
// Original version
let V = config && config[KEY_BOOKCASE_VERSION] ? config[KEY_BOOKCASE_VERSION] : '0';
switch (V) {
case '0.1':
refreshMyUserDetail(logUpgrade);
break;
case VALUE_USRDETAIL_VERSION:
DoLog(LogLevel.Info, 'User-detail config is in latest version. ');
break;
default:
DoLog(LogLevel.Error, 'configCheckUpgrade: Invalid config version({V}) for User-detail. '.replace('{V}', V));
}
// Save to gm_storage
CM.userDtlePrefs.saveConfig(config);
function logUpgrade() {
DoLog(LogLevel.Success, 'User-detail config successfully upgraded From v{V1} to v{V2}. '.replaceAll('{V1}', V).replaceAll('{V2}', VALUE_USRDETAIL_VERSION));
}
}
}
CM.BkReviewPrefs = {
saveConfig: function(config) {
config ? config[KEY_REVIEW_VERSION] = VALUE_REVIEW_VERSION : function() {};
GM_setValue(KEY_REVIEW_PREFS, config);
},
initConfig: function(save=true, func) {
let config = {
bbcode: false,
autoRefresh: false,
beautiful: true,
backgroundImage: 'https://img12.360buyimg.com/ddimg/jfs/t1/197476/22/6462/3478996/613227a8E03e8ffc3/99970183ddb9f896.jpg',
favorites: {
228884: {
name: '文库导航姬',
href: `https://${location.host}/modules/article/reviewshow.php?rid=228884`,
tiptitle: '梦想成为书评区大水怪的可以来康康'
}
},
favorlast: false,
history: {}
};
config = func ? func(config) : config;
save ? CM.BkReviewPrefs.saveConfig(config) : function() {};
return config;
},
getConfig: function(init) {
let config = GM_getValue(KEY_REVIEW_PREFS, null);
config = config ? config : (init ? CM.BkReviewPrefs.initConfig(true, init) : CM.BkReviewPrefs.initConfig());
return config;
},
// Review config upgrade
upgradeConfig: function() {
// Get config
const config = CM.BkReviewPrefs.getConfig();
// if not inited
if (!config) {return;};
switch (config[KEY_REVIEW_VERSION]) {
case '0.1':
v01_To_v02();
v02_To_v03();
v03_To_v04();
v04_To_v05();
v05_To_v06();
v06_To_v07();
v07_To_v08();
v08_To_v09();
logUpgrade();
break;
case '0.2':
v02_To_v03();
v03_To_v04();
v04_To_v05();
v05_To_v06();
v06_To_v07();
v07_To_v08();
v08_To_v09();
logUpgrade();
break;
case '0.3':
v03_To_v04();
v04_To_v05();
v05_To_v06();
v06_To_v07();
v07_To_v08();
v08_To_v09();
logUpgrade();
break;
case '0.4':
v04_To_v05();
v05_To_v06();
v06_To_v07();
v07_To_v08();
v08_To_v09();
logUpgrade();
break;
case '0.5':
v05_To_v06();
v06_To_v07();
v07_To_v08();
v08_To_v09();
logUpgrade();
break;
case '0.6':
v06_To_v07();
v07_To_v08();
v08_To_v09();
logUpgrade();
break;
case '0.7':
v07_To_v08();
v08_To_v09();
logUpgrade();
break;
case '0.8':
v08_To_v09();
logUpgrade();
break;
case VALUE_REVIEW_VERSION:
DoLog(LogLevel.Info, 'Review config is in latest version. ');
break;
default:
DoLog(LogLevel.Error, 'configCheckUpgrade: Invalid config version({V}) for Review. '.replace('{V}', config[KEY_REVIEW_VERSION]));
}
// Save to gm_storage
CM.BkReviewPrefs.saveConfig(config);
function logUpgrade() {
DoLog(LogLevel.Success, 'Review config successfully upgraded From v{V1} to v{V2}. '.replaceAll('{V1}', config[KEY_REVIEW_VERSION]).replaceAll('{V2}', VALUE_REVIEW_VERSION));
}
function v01_To_v02() {
config.autoRefresh = false;
delete config.downloading;
}
function v02_To_v03() {
config.favorites = {
228884: {
name: '文库导航姬',
href: `https://${location.host}/modules/article/reviewshow.php?rid=228884`,
tiptitle: '梦想成为书评区大水怪的可以来康康'
}
}
}
function v03_To_v04() {
if (config.favorites) {return;};
config.favorites = {
228884: {
name: '文库导航姬',
href: `https://${location.host}/modules/article/reviewshow.php?rid=228884`,
tiptitle: '梦想成为书评区大水怪的可以来康康'
}
};
}
function v04_To_v05() {
if (config.history) {return;};
config.history = {};
}
function v05_To_v06() {
if (config.beautiful !== undefined) {return;};
config.beautiful = true;
config.backgroundImage = 'https://img12.360buyimg.com/ddimg/jfs/t1/197476/22/6462/3478996/613227a8E03e8ffc3/99970183ddb9f896.jpg';
}
function v06_To_v07() {
// Move CM.BkReviewPrefs.upgradeConfig.beautiful to CM.BeautifierCfg
if (config.beautiful === undefined) {return;};
const beautifierConfig = {
reviewshow: {
beautiful: config.beautiful,
backgroundImage: config.backgroundImage
}
}
CM.BeautifierCfg.saveConfig(beautifierConfig);
delete config.beautiful;
delete config.backgroundImage;
}
function v07_To_v08() {
// Move CM.BkReviewPrefs.upgradeConfig.beautiful to CM.BeautifierCfg
if (config.favorlast !== undefined) {return;};
config.favorlast = false;
for (const [rid, favorite] of Object.entries(config.favorites)) {
config.favorites[rid] = {
name: favorite.name,
href: favorite.href.replace(/&page=1$/, ''),
tiptitle: favorite.tiptitle
};
}
}
function v08_To_v09() {
// Fill all favorite bookreviews' tiptitle using null for those don't have
config.favorlast = false;
for (const [rid, favorite] of Object.entries(config.favorites)) {
!favorite.tiptitle && (favorite.tiptitle = null);
}
}
}
}
CM.BeautifierCfg = {
saveConfig: function(config) {
config ? config[KEY_BEAUTIFIER_VERSION] = VALUE_BEAUTIFIER_VERSION : function() {};
GM_setValue(KEY_BEAUTIFIER, config);
},
initConfig: function(save=true, func) {
let config = {
upload: false,
reviewshow: {
beautiful: true,
},
novel: {
beautiful: true,
},
common: {
beautiful: false,
},
backgroundImage: 'https://img12.360buyimg.com/ddimg/jfs/t1/197476/22/6462/3478996/613227a8E03e8ffc3/99970183ddb9f896.jpg',
bgiName: '默认背景图片 - Pixiv ID: 88913164',
textScale: 100
};
config = func ? func(config) : config;
save ? CM.BeautifierCfg.saveConfig(config) : function() {};
return config;
},
getConfig: function(init) {
let config = GM_getValue(KEY_BEAUTIFIER, null);
config = config ? config : (init ? CM.BeautifierCfg.initConfig(true, init) : CM.BeautifierCfg.initConfig());
return config;
},
// Beautifier config upgrade
upgradeConfig: function() {
// Get config
const config = CM.BeautifierCfg.getConfig();
// if not inited
if (!config) {return;};
switch (config[KEY_BEAUTIFIER_VERSION]) {
/*case '0.1':
v01_To_v02();
break;*/
case VALUE_BEAUTIFIER_VERSION:
DoLog(LogLevel.Info, 'Beautifier config is in latest version. ');
break;
case '0.1':
v01_To_v02();
v02_To_v03();
v03_To_v04();
v04_To_v05();
v05_To_v06();
v06_To_v07();
v07_To_v08();
v08_To_v09();
logUpgrade();
break;
case '0.2':
v02_To_v03();
v03_To_v04();
v04_To_v05();
v05_To_v06();
v06_To_v07();
v07_To_v08();
v08_To_v09();
logUpgrade();
break;
case '0.3':
v03_To_v04();
v04_To_v05();
v05_To_v06();
v06_To_v07();
v07_To_v08();
v08_To_v09();
logUpgrade();
break;
case '0.4':
v04_To_v05();
v05_To_v06();
v06_To_v07();
v07_To_v08();
v08_To_v09();
logUpgrade();
break;
case '0.5':
v05_To_v06();
v06_To_v07();
v07_To_v08();
v08_To_v09();
logUpgrade();
break;
case '0.6':
v06_To_v07();
v07_To_v08();
v08_To_v09();
logUpgrade();
break;
case '0.7':
v07_To_v08();
v08_To_v09();
logUpgrade();
break;
case '0.8':
v08_To_v09();
logUpgrade();
break;
default:
DoLog(LogLevel.Error, 'configCheckUpgrade: Invalid config version({V}) for Beautifier. '.replace('{V}', config[KEY_BEAUTIFIER_VERSION]));
}
// Save to gm_storage
CM.BeautifierCfg.saveConfig(config);
function logUpgrade() {
DoLog(LogLevel.Success, 'Beautifier config successfully upgraded From v{V1} to v{V2}. '.replaceAll('{V1}', config[KEY_BEAUTIFIER_VERSION]).replaceAll('{V2}', VALUE_BEAUTIFIER_VERSION));
}
function v01_To_v02() {
if (config.upload !== undefined) {return false;};
config.upload = false;
}
function v02_To_v03() {
if (config.reviewshow.bgiName !== undefined) {return false;};
config.reviewshow.bgiName = 'image.jpeg';
}
function v03_To_v04() {
if (config.textScale !== undefined) {return false;};
config.textScale = 100;
}
function v04_To_v05() {
if (config.novel !== undefined) {return false;};
config.novel = {
beautiful: true
};
}
function v05_To_v06() {
if (!config.textScale) {config.textScale = 100;};
if (!config.novel) {config.novel = {beautiful: true};};
}
function v06_To_v07() {
config.backgroundImage = config.reviewshow.backgroundImage;
config.bgiName = config.reviewshow.bgiName;
delete config.reviewshow.backgroundImage;
delete config.reviewshow.bgiName;
}
function v07_To_v08() {
if (config.common) {return false;}
config.common = {
beautiful: false
};
}
function v08_To_v09() {
if (config.common) {return false;}
config.common = {
beautiful: false
};
}
}
}
CM.RemarksConfig = {
saveConfig: function(config) {
config ? config[KEY_REMARKS_VERSION] = VALUE_REMARKS_VERSION : function() {};
GM_setValue(KEY_REMARKS, config);
},
initConfig: function(save=true, func) {
let config = {
user: {}
};
config = func ? func(config) : config;
save ? CM.RemarksConfig.saveConfig(config) : function() {};
return config;
},
getConfig: function(init) {
let config = GM_getValue(KEY_REMARKS, null);
config = config ? config : (init ? CM.RemarksConfig.initConfig(true, init) : CM.RemarksConfig.initConfig());
return config;
},
// Beautifier config upgrade
upgradeConfig: function() {
// Get config
const config = CM.RemarksConfig.getConfig();
// if not inited
if (!config) {return;};
switch (config[KEY_REMARKS_VERSION]) {
//case '0.1':
// v01_To_v02();
// logUpgrade();
// break;
case VALUE_REMARKS_VERSION:
DoLog(LogLevel.Info, 'RemarksConfig config is in latest version. ');
break;
default:
DoLog(LogLevel.Error, 'configCheckUpgrade: Invalid config version({V}) for RemarksConfig. '.replace('{V}', config[KEY_REMARKS_VERSION]));
}
// Save to gm_storage
CM.RemarksConfig.saveConfig(config);
function logUpgrade() {
DoLog(LogLevel.Success, 'RemarksConfig config successfully upgraded From v{V1} to v{V2}. '.replaceAll('{V1}', config[KEY_REMARKS_VERSION]).replaceAll('{V2}', VALUE_REMARKS_VERSION));
}
//function v#BEFORE_To_v#AFTER() {
// if (config.#NEWPROP !== undefined) {return false;};
// config.#NEWPROP = #DEFAULTVALUE;
//}
}
}
CM.UserGlobalCfg = {
saveConfig: function(config) {
config ? config[KEY_USERGLOBAL_VERSION] = VALUE_USERGLOBAL_VERSION : function() {};
GM_setValue(KEY_USERGLOBAL, config);
},
initConfig: function(save=true, func) {
let config = {
imager: DATA_IMAGERS.default
};
config = func ? func(config) : config;
save ? CM.UserGlobalCfg.saveConfig(config) : function() {};
return config;
},
getConfig: function(init) {
let config = GM_getValue(KEY_USERGLOBAL, null);
config = config ? config : (init ? CM.UserGlobalCfg.initConfig(true, init) : CM.UserGlobalCfg.initConfig());
return config;
},
// Beautifier config upgrade
upgradeConfig: function() {
// Get config
const config = CM.UserGlobalCfg.getConfig();
// if not inited
if (!config) {return;};
switch (config[KEY_USERGLOBAL_VERSION]) {
//case '0.1':
// v01_To_v02();
// logUpgrade();
// break;
case VALUE_USERGLOBAL_VERSION:
DoLog(LogLevel.Info, 'UserGlobal config is in latest version. ');
break;
default:
DoLog(LogLevel.Error, 'configCheckUpgrade: Invalid config version({V}) for UserGlobalCfg. '.replace('{V}', config[KEY_USERGLOBAL_VERSION]));
}
// Save to gm_storage
CM.UserGlobalCfg.saveConfig(config);
function logUpgrade() {
DoLog(LogLevel.Success, 'UserGlobal config successfully upgraded From v{V1} to v{V2}. '.replaceAll('{V1}', config[KEY_USERGLOBAL_VERSION]).replaceAll('{V2}', VALUE_USERGLOBAL_VERSION));
}
//function v#BEFORE_To_v#AFTER() {
// if (config.#NEWPROP !== undefined) {return false;};
// config.#NEWPROP = #DEFAULTVALUE;
//}
}
}
// New Config Item Template
/*CM.#NEWCONFIGNAME = {
saveConfig: function(config) {
config ? config[#KEY_NEWCONFIG_VERSION] = #VALUE_NEWCONFIG_VERSION : function() {};
GM_setValue(#KEY_NEWCONFIG, config);
},
initConfig: function(save=true, func) {
let config = {
#key: #value,
#key: #value
};
config = func ? func(config) : config;
save ? CM.#NEWCONFIGNAME.saveConfig(config) : function() {};
return config;
},
getConfig: function(init) {
let config = GM_getValue(#KEY_NEWCONFIG, null);
config = config ? config : (init ? CM.#NEWCONFIGNAME.initConfig(true, init) : CM.#NEWCONFIGNAME.initConfig());
return config;
},
// Beautifier config upgrade
upgradeConfig: function() {
// Get config
const config = CM.#NEWCONFIGNAME.getConfig();
// if not inited
if (!config) {return;};
switch (config[#KEY_NEWCONFIG_VERSION]) {
//case '0.1':
// v01_To_v02();
// logUpgrade();
// break;
case #VALUE_NEWCONFIG_VERSION:
DoLog(LogLevel.Info, '#NEWCONFIGNAME config is in latest version. ');
break;
default:
DoLog(LogLevel.Error, 'configCheckUpgrade: Invalid config version({V}) for #NEWCONFIGNAME. '.replace('{V}', config[#KEY_NEWCONFIG_VERSION]));
}
// Save to gm_storage
CM.#NEWCONFIGNAME.saveConfig(config);
function logUpgrade() {
DoLog(LogLevel.Success, '#NEWCONFIGNAME config successfully upgraded From v{V1} to v{V2}. '.replaceAll('{V1}', config[#KEY_NEWCONFIG_VERSION]).replaceAll('{V2}', #VALUE_NEWCONFIG_VERSION));
}
//function v#BEFORE_To_v#AFTER() {
// if (config.#NEWPROP !== undefined) {return false;};
// config.#NEWPROP = #DEFAULTVALUE;
//}
}
}*/
CM.AutoRecommend.upgradeConfig();
CM.commentDrafts.upgradeConfig();
CM.bookcasePrefs.upgradeConfig();
CM.userDtlePrefs.upgradeConfig();
CM.BkReviewPrefs.upgradeConfig();
CM.BeautifierCfg.upgradeConfig();
CM.RemarksConfig.upgradeConfig();
CM.UserGlobalCfg.upgradeConfig();
//CM.#NEWCONFIGNAME.upgradeConfig();
}
// Beautifier for all wenku pages
function commonBeautify(API) {
// No beautifier on exluded pages
const excludes = ['novel']
if (excludes.includes(API)) {return false;}
// No beatifier if user does not want
if (!CONFIG.BeautifierCfg.getConfig().common.beautiful) {return false;}
const img = $CrE('img');
img.src = CONFIG.BeautifierCfg.getConfig().backgroundImage;
img.classList.add('plus_cbty_image');
document.body.appendChild(img);
const cover = $CrE('div');
cover.classList.add('plus_cbty_cover');
document.body.appendChild(cover);
document.body.classList.add('plus_cbty');
addStyle(CSS_COMMONBEAUTIFIER, 'plus_commonbeautifier')
return true;
}
// Book page add-on
function pageBook() {
// Resource
const pageResource = {
elements: {},
info: {}
}
collectPageResources();
DoLog(LogLevel.Info, pageResource, true)
// Provide meta info copy
metaCopy();
// Provide read-later button
laterReads();
// Provide txtfull download for copyright book
enableDownload();
// Provide images download
imagesDownload();
// Provide tag search
tagOption();
// Ctrl+Enter comment submit
areaReply();
// Get page resources
function collectPageResources() {
collectElements();
collectInfos();
function collectElements() {
const elements = pageResource.elements;
elements.content = $('#content');
elements.bookMain = $(elements.content, 'div');
elements.header = $(elements.content, 'div>table');
elements.titleContainer = $(elements.header, 'table td>span');
elements.bookName = $(elements.header, 'b');
elements.recommend = $(elements.content, `a[href^="https://${location.host}/modules/article/uservote.php"]`);
elements.metaContainer = $(elements.header, 'tr+tr');
elements.metas = $All(elements.metaContainer, 'td');
elements.info = $(elements.bookMain, 'div+table');
elements.cover = $(elements.info, 'img');
elements.infoText = $(elements.info, 'td+td');
elements.notice = $All(elements.infoText, 'span.hottext>b');
elements.tags = elements.notice.length > 1 ? elements.notice[0] : null;
elements.notice = elements.notice[elements.notice.length-1];
elements.introduce = $All(elements.infoText, 'span');
elements.introduce = elements.introduce[elements.introduce.length-1];
elements.downloadContainer = $(pageResource.elements.bookMain, 'div>fieldset');
elements.downloadPanel = elements.downloadContainer ? elements.downloadContainer.parentElement : null;
}
function collectInfos() {
const info = pageResource.info;
const elements = pageResource.elements;
info.bookName = elements.bookName.innerText;
info.BID = Number(getUrlArgv('id') || location.href.match(/book\/(\d+).htm/)[1]);
info.metas = []; elements.metas.forEach(function(meta){this.push(getKeyValue(meta.innerText));}, info.metas);
info.notice = elements.notice.innerText;
info.tags = elements.tags ? getKeyValue(elements.tags.innerText).VALUE.split(' ') : null;
info.introduce = elements.introduce.innerText;
info.cover = elements.cover.src;
info.dlEnabled = $(elements.content, 'legend>b');
info.dlEnabled = info.dlEnabled ? info.dlEnabled.innerText : false;
info.dlEnabled = info.dlEnabled ? (info.dlEnabled.indexOf('TXT') !== -1 && info.dlEnabled.indexOf('UMD') !== -1 && info.dlEnabled.indexOf('JAR') !== -1) : false;
}
}
// Copy meta info
function metaCopy() {
let tip = TEXT_TIP_COPY;
for (let i = -1; i < pageResource.elements.metas.length; i++) {
const meta = i !== -1 ? pageResource.elements.metas[i] : pageResource.elements.bookName;
const info = i !== -1 ? pageResource.info.metas[i] : pageResource.info.bookName;
const value = i !== -1 ? info.VALUE : info;
meta.innerHTML += HTML_BOOK_COPY;
const copyBtn = $(meta, '.'+CLASSNAME_BUTTON);
copyBtn.addEventListener('click', function() {
copyText(value);
showtip(TEXT_TIP_COPIED);
alertify.message(TEXT_ALT_META_COPIED.replaceAll('{M}', value));
});
settip(copyBtn, TEXT_TIP_COPY);
}
}
// Add to later-reads
function laterReads() {
// Make button
let btn = installBtn(makeBtn(inAfterbooks() ? 'remove' : 'add'));
// Update book info if in list
inAfterbooks() && add(false);
function add(alt=true) {
// Add to config
const config = CONFIG.bookcasePrefs.getConfig();
config.laterbooks.books[pageResource.info.BID] = {
sort: Object.keys(config.laterbooks.books).length + 1,
addTime: new Date().getTime(),
name: pageResource.info.bookName,
aid: pageResource.info.BID,
metas: pageResource.info.metas,
tags: pageResource.info.tags,
introduce: pageResource.info.introduce,
cover: pageResource.info.cover
};
CONFIG.bookcasePrefs.saveConfig(config);
// New button
removeBtn(btn);
btn = installBtn(makeBtn('remove'));
// Soft alert
alt && alertify.success(TEXT_ALT_BOOK_AFTERBOOKS_ADDED);
}
function remove() {
// Remove from config
const config = CONFIG.bookcasePrefs.getConfig();
const books = config.laterbooks.books;
const book = books[pageResource.info.BID];
if (!book) {return false;}
delete books[pageResource.info.BID];
Array.prototype.forEach.call(Object.values(books), (b) => (b.sort > book.sort && b.sort--));
CONFIG.bookcasePrefs.saveConfig(config);
// New button
removeBtn(btn);
btn = installBtn(makeBtn('add'));
// Soft alert
alertify.success(TEXT_ALT_BOOK_AFTERBOOKS_REMOVED);
}
function makeBtn(type='add') {
const btn = $CrE('span');
btn.classList.add(CLASSNAME_BUTTON);
switch (type) {
case 'add':
btn.innerHTML = TEXT_GUI_BOOK_READITLATER;
btn.addEventListener('click', add);
break;
case 'remove':
btn.innerHTML = TEXT_GUI_BOOK_DONTREADLATER;
btn.addEventListener('click', remove);
break;
}
return btn;
}
function installBtn(btn) {
pageResource.elements.recommend.previousElementSibling.insertAdjacentElement('afterend', btn);
btn.insertAdjacentText('beforebegin', '[');
btn.insertAdjacentText('afterend', ']');
return btn;
}
function removeBtn(btn) {
const parent = btn.parentElement;
for (const node of [btn.previousSibling, btn, btn.nextSibling]) {
parent.removeChild(node);
}
return btn;
}
function inAfterbooks() {
return CONFIG.bookcasePrefs.getConfig().laterbooks.books[pageResource.info.BID] ? true : false;
}
}
// Download copyright book
function enableDownload() {
if (pageResource.info.dlEnabled) {return false;};
// Download panel
// Create panel
let div = $CrE('div');
pageResource.elements.bookMain.appendChild(div);
div.outerHTML = HTML_DOWNLOAD_LINKS
.replaceAll('{ORIBOOKNAME}', pageResource.info.bookName)
.replaceAll('{BOOKID}', String(pageResource.info.BID))
.replaceAll('{CHARSET}', getUrlArgv('charset') ? '&charset=' + getUrlArgv('charset') : '')
// Use about:blank instead of direct url; aims to aviod unnecessary web requests
const container = pageResource.elements.downloadContainer = $(pageResource.elements.bookMain, 'div>fieldset');
div = pageResource.elements.downloadPanel = container.parentElement;
for (const a of $All(container, 'div>a')) {
//a.addEventListener('click', openDlPage);
}
// Notice board
pageResource.elements.notice.innerHTML = HTML_DOWNLOAD_BOARD
.replaceAll('{ORIBOOKNAME}', pageResource.info.bookName);
function openDlPage(e) {
e.preventDefault();
const url = e.target.href;
const win = window.open(`https://${location.host}/`);
win.history.replaceState({...win.history.state}, '', url);
}
}
// All images downloader
function imagesDownload() {
const container = pageResource.elements.downloadContainer;
const divImage = $CrE('div'), a = $CrE('a');
divImage.setAttribute('style', 'width:164px; float:left; text-align:center');
a.href = 'javascript:void(0);';
a.innerHTML = TEXT_GUI_BOOK_IMAGESDOWNLOAD;
a.addEventListener('click', confirm);
divImage.appendChild(a);
container.appendChild(divImage);
for (const div of $All(container, 'div')) {
div.style.width = '164px';
}
function confirm() {
const title = TEXT_ALT_DOWNLOADIMG_CONFIRM_TITLE;
const message = TEXT_ALT_DOWNLOADIMG_CONFIRM_MESSAGE.replace('{N}', pageResource.info.bookName);
const ok = TEXT_ALT_DOWNLOADIMG_CONFIRM_OK;
const cancel = TEXT_ALT_DOWNLOADIMG_CONFIRM_CANCEL;
alertify.confirm(title, message, download, function() {/* oncancel */}).set('labels', {ok: ok, cancel: cancel});
}
function download() {
// GUI
const delay = alertify.get('notifier','delay');
alertify.set('notifier','delay', 0);
let finished = false, CAll, CCur = 0;
const AM = new AsyncManager();
AM.onfinish = downloadFinish;
const box = alertify.message(TEXT_ALT_DOWNLOADIMG_STATUS_INDEX);
box.ondismiss = function() {return finished;}
// Start download
AM.add()
AndAPI.getNovelIndex({
aid: pageResource.info.BID,
lang: 0,
callback: function(xml) {
const allChapters = $All(xml, 'chapter');
const chapters = Array.prototype.filter.call(allChapters, (c) => (c.firstChild.nodeValue.includes('插图')));
CAll = chapters.length;
box.setContent(TEXT_ALT_DOWNLOADIMG_STATUS_LOADING.replace('{CCUR}', CCur).replace('{CALL}', CAll));
for (const chapter of chapters) {
AM.add();
getChapter(chapter.getAttribute('cid'), chapter.parentNode);
}
AM.finish();
}
});
AM.finishEvent = true;
function getChapter(cid, volume) {
AndAPI.getNovelContent({
aid: pageResource.info.BID,
cid: cid,
lang: 0,
callback: getImgs,
args: [volume]
});
function getImgs(str, volume) {
const imgs = str.match(/<!--image-->https?:[^<>]+<!--image-->/g);
const len = imgs.length.toString().length;
const CAM = new AsyncManager();
CAM.onfinish = chapterFinish;
for (let i = 0; i < imgs.length; i++) {
const img = imgs[i];
const src = img.match(/<!--image-->(https?:[^<>]+)<!--image-->/)[1];
const ext = src.match(/\.(\w+)$/) ? src.match(/\.(\w+)$/)[1] : 'jpg';
const filename = pageResource.info.bookName + '_' + volume.firstChild.nodeValue + ' ' + ['插图', '插圖'][getLang()] + '_' + fillNumber(i+1, len) + '.' + ext;
CAM.add();
downloadFile({
url: src,
name: filename,
onload: function() {
CAM.finish();
}
});
}
CAM.finishEvent = true;
function chapterFinish() {
AM.finish();
box.setContent(TEXT_ALT_DOWNLOADIMG_STATUS_LOADING.replace('{CCUR}', ++CCur).replace('{CALL}', CAll));
}
}
}
function downloadFinish() {
finished = true;
alertify.set('notifier','delay', delay);
box.dismiss();
alertify.success(TEXT_ALT_DOWNLOADIMG_STATUS_FINISH);
}
}
}
// Download copyright book full txt
function enableDownload_old() {
if (pageResource.info.dlEnabled) {return false;};
let div = $CrE('div');
pageResource.elements.bookMain.appendChild(div);
div.outerHTML = HTML_DOWNLOAD_LINKS_OLD
.replaceAll('{ORIBOOKNAME}', pageResource.info.bookName)
.replaceAll('{BOOKID}', String(pageResource.info.BID))
.replaceAll('{BOOKNAME}', encodeURIComponent(pageResource.info.bookName));
div = $('#txtfull');
pageResource.elements.txtfull = div;
pageResource.elements.notice.innerHTML = HTML_DOWNLOAD_BOARD
.replaceAll('{ORIBOOKNAME}', pageResource.info.bookName);
}
// Tag Search
function tagOption() {
const tagsEle = pageResource.elements.tags;
const tags = pageResource.info.tags;
if (!tags) {return false;}
let html = getKeyValue(tagsEle.innerText).KEY + ':';
for (const tag of tags) {
html += HTML_BOOK_TAG.replace('{TU}', $URL.encode(tag)).replace('{TN}', tag) + ' ';
}
tagsEle.innerHTML = html;
}
}
// Reply area add-on
function areaReply() {
/* ## Release title area ## */
if ($('td > input[name="Submit"]') && !$('#ptitle')) {
const table = $('form>table');
const titleText = table.innerHTML.match(/<!--[\s\S]+id="ptitle"[\s\S]+-->/)[0];
const titleHTML = titleText.replace(/^<!--\s*/, '').replace(/\s*-->$/, '');
const titleEle = $CrE('tr');
const caption = $(table, 'caption');
table.insertBefore(titleEle, caption);
titleEle.outerHTML = titleHTML;
}
const commentArea = $('#pcontent'); if (!commentArea) {return false;};
const commentForm = $(`form[action^="https://${location.host}/modules/article/review"]`);
const commentSbmt = $('td > input[name="Submit"]');
const commenttitl = $('#ptitle');
const commentbttm = commentSbmt.parentElement;
/* ## Ctrl+Enter comment submit ## */
let btnSbmtValue = commentSbmt.value;
if (commentSbmt) {
commentSbmt.value = '发表书评(Ctrl+Enter)';
commentSbmt.style.padding = '0.3em 0.4em 0.3em 0.4em';
commentSbmt.style.height= 'auto';
commentArea.addEventListener('keydown', hotkeyReply);
commenttitl.addEventListener('keydown', hotkeyReply);
}
// Enable https protocol for inserted url
fixHTTPS();
// Provide image upload & insert
imageplus();
// At user
atUser();
// Comment auto-save
// GUI
const asTip = $CrE('span');
commentbttm.appendChild(asTip);
// Review-Page: Same rid, same savekey - 'rid123456'
// Book-Page & Book-Review-List-Page: Same bookid, same savekey - 'bid1234'
const rid = getUrlArgv({url: commentForm.action, name: 'rid', dealFunc: Number});
const aid = getUrlArgv({url: commentForm.action, name: 'aid', dealFunc: Number});
const bid = location.href.match(/\/book\/(\d+).htm/) ? Number(location.href.match(/\/book\/(\d+).htm/)[1]) : 0;
const key = rid ? 'rid' + String(rid) : 'bid' + String(bid);
let commentData = CONFIG.commentDrafts.getConfig()[key] || {
key : key,
rid : rid,
aid : aid,
bid : bid,
page : getUrlArgv({name: 'rid', dealFunc: Number, defaultValue: 1}),
time : (new Date()).getTime()
};
restoreDraft();
submitHook();
const events = ['focus', 'blur', 'mousedown', 'keydown', 'keyup', 'change'];
const eventEles = [commentArea, commenttitl];
for (const eventEle of eventEles) {
for (const event of events) {
eventEle.addEventListener(event, saveDraft);
}
}
function saveDraft() {
const content = commentArea.value;
const title = commenttitl.value;
if (!content && !title) {
clearDraft();
return;
} else if (commentData.content === content && commentData.title === title) {
return;
}
commentData.content = content;
commentData.title = title;
const allCData = CONFIG.commentDrafts.getConfig();
allCData[commentData.key] = commentData;
CONFIG.commentDrafts.saveConfig(allCData);
asTip.innerHTML = TEXT_GUI_AUTOSAVE;
}
function restoreDraft() {
const allCData = CONFIG.commentDrafts.getConfig();
if (!allCData[commentData.key]) {return false;};
if (!commenttitl.value && !commentArea.value) {
commentData = allCData[commentData.key];
commenttitl.value = commentData.title;
commentArea.value = commentData.content;
asTip.innerHTML = TEXT_GUI_AUTOSAVE_RESTORE;
}
return true;
}
function clearDraft() {
const allCData = CONFIG.commentDrafts.getConfig();
if (!allCData[commentData.key]) {return false;};
delete allCData[commentData.key];
CONFIG.commentDrafts.saveConfig(allCData);
asTip.innerHTML = TEXT_GUI_AUTOSAVE_CLEAR;
return true;
}
function hotkeyReply() {
let keycode = event.keyCode;
if (keycode === 13 && event.ctrlKey && !event.altKey) {
// Do not submit directly like this; we need to submit with onsubmit executed
//commentForm.submit();
commentSbmt.click();
}
}
function fixHTTPS() {
if (typeof(UBBEditor) === 'undefined') {
fixHTTPS.wait = fixHTTPS.wait ? fixHTTPS.wait : 0;
if (++fixHTTPS.wait > 50) {return false;}
DoLog('fixHTTPS: UBBEditor not loaded, waiting...');
setTimeout(fixHTTPS, NUMBER_ELEMENT_LOADING_WAIT_INTERVAL);
return false;
}
const eid = 'pcontent';
const menuItemInsertUrl = $(commentForm, '#menuItemInsertUrl');
const menuItemInsertImage = $(commentForm, '#menuItemInsertImage');
// Wait until menuItemInsertUrl and menuItemInsertImage is loaded
if (!menuItemInsertUrl || !menuItemInsertImage) {
DoLog(LogLevel.Info, 'fixHTTPS: element not loaded, waiting...');
setTimeout(fixHTTPS, NUMBER_ELEMENT_LOADING_WAIT_INTERVAL);
return false;
}
// Wait until original onclick function is set
if (!menuItemInsertUrl.onclick || !menuItemInsertImage.onclick) {
DoLog(LogLevel.Info, 'fixHTTPS: defult onclick not loaded, waiting...');
setTimeout(fixHTTPS, NUMBER_ELEMENT_LOADING_WAIT_INTERVAL);
return false;
}
menuItemInsertUrl.onclick = function () {
var url = prompt("请输入超链接地址", "http://");
if (url != null && url.indexOf("http://") < 0 && url.indexOf("https://") < 0) {
alert("请输入完整的超链接地址!");
return;
}
if (url != null) {
if ((document.selection && document.selection.type == "Text") ||
(window.getSelection &&
document.getElementById(eid).selectionStart > -1 && document.getElementById(eid).selectionEnd >
document.getElementById(eid).selectionStart)) {UBBEditor.InsertTag(eid, "url", url,'');}
else {UBBEditor.InsertTag(eid, "url", url, url);}
}
};
menuItemInsertImage.onclick = function () {
var imgurl = prompt("请输入图片路径", "http://");
if (imgurl != null && imgurl.indexOf("http://") < 0 && imgurl.indexOf("https://") < 0) {
alert("请输入完整的图片路径!");
return;
}
if (imgurl != null) {
UBBEditor.InsertTag(eid, "img", "", imgurl);
}
};
return true;
}
function imageplus() {
if (typeof(UBBEditor) === 'undefined') {
DoLog('imageplus: UBBEditor not loaded, waiting...');
setTimeout(imageplus, NUMBER_ELEMENT_LOADING_WAIT_INTERVAL);
return false;
}
// Imager menu
const menu = $('#UBB_Menu');
const elmImage = $(commentForm, '#menuItemInsertImage');
const onclick = elmImage.onclick;
const imagers = new PlusList({
id: 'plus_imager',
list: [
{value: TEXT_GUI_REVEIW_IMG_INSERTURL, tip: TEXT_TIP_REVIEW_IMG_INSERTURL, onclick: onclick},
{value: TEXT_GUI_REVEIW_IMG_SELECTIMG, tip: TEXT_TIP_REVIEW_IMG_SELECTIMG, onclick: pickfile}
],
parentElement: menu.parentElement,
insertBefore: $('#SmileListTable'),
visible: false,
onshow: onshow
});
elmImage.onclick = (e) => {
e.stopPropagation();
imagers.show();
};
document.addEventListener('click', imagers.hide);
// drag-drop & copy-paste
commentArea.addEventListener('paste', pictureGot);
commentArea.addEventListener('dragenter', destroyEvent);
commentArea.addEventListener('dragover', destroyEvent);
commentArea.addEventListener('drop', pictureGot);
function onshow() {
imagers.div.style.left = String(UBBEditor.GetPosition(elmImage).x) + 'px';
imagers.div.style.top = String(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(TEXT_ALT_IMAGE_FORMATERROR);
}
return false;
} else {
destroyEvent(e);
}
// Insert picture marker
const marker = '[image_uploading={ID} name={NAME}]'.replace('{ID}', randstr(16, true, commentArea.value)).replace('{NAME}', name);
insertText(marker);
// Upload
alertify.notify(TEXT_ALT_IMAGE_UPLOAD_WORKING);
uploadImage({
file: file,
onerror: (e) => {
alertify.error(TEXT_ALT_IMAGE_UPLOAD_ERROR);
DoLog(LogLevel.Error, ['Upload error at imageplus>upload:', e]);
},
onload: (json) => {
const name = json.name;
const url = json.url;
commentArea.value = commentArea.value.replace(marker, url);
alertify.success(TEXT_ALT_IMAGE_UPLOAD_SUCCESS.replaceAll('{NAME}', name).replaceAll('{URL}', url));
}
});
}
}
function submitHook() {
const onsubmit = commentForm.onsubmit;
commentForm.onsubmit = onsubmitForm;
function onsubmitForm(e) {
// Cancel submit while content empty
if (commentArea.value === '' && commenttitl.value === '') {return false;};
// Clear Draft
clearDraft();
// Restore original submit button value
if (commentSbmt.value !== btnSbmtValue) {
commentSbmt.value = btnSbmtValue;
setTimeout(()=>{commentSbmt.click.call(commentSbmt);}, 0);
return false;
}
// Continue submit
return onsubmit ? onsubmit() : function() {return true;};
}
}
function atUser() {
if (typeof(UBBEditor) === 'undefined') {
DoLog(LogLevel.Info, 'atUser: UBBEditor not loaded, waiting...');
setTimeout(atUser, NUMBER_ELEMENT_LOADING_WAIT_INTERVAL);
return false;
}
const menu = $('#UBB_Menu');
const list = new PlusList({
id: 'plus_AtTable',
list: [],
parentElement: menu.parentElement,
insertBefore: $('#FontSizeTable'),
visible: false,
onshow: showlist
});
list.onhide = list.clear;
document.addEventListener('click', list.hide);
const firstBtn = menu.children[0];
const atBtn = $CrE('input');
atBtn.type = 'button';
atBtn.style.backgroundImage = 'none';
atBtn.value = '@';
atBtn.title = TEXT_GUI_AREAREPLY_AT;
atBtn.id = 'plus_At';
atBtn.classList.add(CLASSNAME_BUTTON);
atBtn.classList.add('UBB_MenuItem');
atBtn.addEventListener('click', (e) => {
e.stopPropagation();
list.show();
});
menu.insertBefore(atBtn, firstBtn);
function showlist(shown) {
if (shown) {return false;};
if (typeof(ubb_subdiv) === 'string' && typeof(hideeve) === 'function') {
hideeve(ubb_subdiv);
ubb_subdiv = 'plus_AtTable';
}
makelist();
list.ul.focus();
return true;
}
function makelist() {
// Get users
const allUsers = getAllUsers();
// Make list
for (const user of allUsers) {
const item = list.append({
value: user.userName,
tip: ()=>{return 'uid: ' + String(user.userID);},
onclick: btnClick
});
item.li.user = user;
item.button.user = user;
}
// Style
list.div.style.left = String(UBBEditor.GetPosition(atBtn).x) + 'px';
list.div.style.top = String(UBBEditor.GetPosition(atBtn).y + 20) + 'px';
return true;
function getAllUsers() {
const pageUsers = $All(`#content table strong>a[href^="https://${location.host}/userpage.php"]`);
const friends = getMyUserDetail().userFriends;
if (!friends) {
refreshMyUserDetail(refreshList);
return false;
}
// concat to one array
const allUsers = [];
for (const pageUser of pageUsers) {
// Valid check
if (isNaN(Number(pageUser.href.match(/\?uid=(\d+)/)[1]))) {continue;};
const user = {
userName: pageUser.innerText,
userID: Number(pageUser.href.match(/\?uid=(\d+)/)[1]),
referred: 0
}
if (!userExist(allUsers, user)) {
const userAsFriend = userExist(friends, user);
allUsers.push(userAsFriend ? userAsFriend : user);
}
}
for (const friend of friends) {
if (!userExist(allUsers, friend)) {
allUsers.push(friend);
}
}
// Sort by referred
allUsers.sort((a,b)=>{return (b.referred?b.referred:0) - (a.referred?a.referred:0);});
return allUsers;
// returns the exist user object found in users, or false if not found
function userExist(users, user) {
for (const u of users) {
if (u.userID === user.userID) {return u;};
}
return false;
}
}
function btnClick() {
const btn = this;
const user = btn.user;
const name = btn.user.userName;
const insertValue = '@' + name;
insertText(insertValue);
// referred increase
const userDetail = getMyUserDetail();
const friends = userDetail.userFriends;
user.referred = user.referred ? user.referred+1 : 1;
for (let i = 0; i < friends.length; i++) {
if (friends[i].userID === user.userID) {
friends[i] = user;
break;
}
}
CONFIG.userDtlePrefs.saveConfig(userDetail);
}
}
}
function insertText(insertValue) {
const insertPosition = commentArea.selectionEnd;
const text = commentArea.value;
const leftText = text.substr(0, insertPosition);
const rightText = text.substr(insertPosition);
// if not at the beginning of a line then insert a whitespace before the link
insertValue = ((leftText.length === 0 || /[ \r\n]$/.test(leftText)) ? '' : ' ') + insertValue;
// if not at the end of a line then insert a whitespace after the link
insertValue += (rightText.length === 0 || /^[ \r\n]/.test(rightText)) ? '' : ' ';
commentArea.value = leftText + insertValue + rightText;
const position = insertPosition + insertValue.length;
commentForm.scrollIntoView(); commentArea.focus(); commentArea.setSelectionRange(position, position);
}
}
// Review link add-on
function linkReview() {
// Get all review links and apply add-on functions
const allRLinks = $All(`td>a[href^="https://${location.host}/modules/article/reviewshow.php?"]`);
for (const RLink of allRLinks) {
lastPage(RLink);
}
// Provide button direct to review last page
// New version. Uses '&page=last' keyword.
function lastPage(a) {
const p = a.parentElement;
const lastpg = $CrE('a');
const strrid = getUrlArgv({url: a.href, name: 'rid'});
lastpg.href = URL_REVIEWSHOW_2.replace('{R}', strrid).replace('{P}', 'last');
lastpg.classList.add(CLASSNAME_BUTTON);
lastpg.target = '_blank';
lastpg.innerText = TEXT_GUI_LINK_TOLASTPAGE;
p.insertBefore(lastpg, a);
}
}
// Side functions area
function sideFunctions() {
const SPanel = new SidePanel();
SPanel.usercss = CSS_SIDEPANEL;
SPanel.create();
SPanel.setPosition('bottom-right');
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');
btnShowHide.setAttribute('aria-label', (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
/*
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) => {
reloadPage();
}
});
}
}
// Reviewedit page add-on
function pageReviewedit() {
redirectToCorrectPage();
function redirectToCorrectPage() {
// Get redirect target rid
const refreshMeta = $('meta[http-equiv="refresh"]');
const metaurl = refreshMeta.content.match(/url=(.+)/)[1];
if (!refreshMeta) {return false;};
if (getUrlArgv({url: metaurl, name: 'page'})) {return false;};
// Read correct redirect location
const rid = Number(getUrlArgv({url: metaurl, name: 'rid'}));
const config = CONFIG.BkReviewPrefs.getConfig();
const history = config.history;
const pageHist = history[rid];
if (!pageHist) {return false;}
const url = pageHist.href;
// Check if time expired (Expire time: 30 seconds)
if ((new Date()).getTime() - pageHist.time > 30*1000) {
// Delete expired record
delete history[rid];
CONFIG.BkReviewPrefs.saveConfig(config);
}
// Redirect link
$('a').href = url;
// Redirect
setTimeout(() => {location.href = url;}, 1500);
}
}
// Review page add-on
function pageReview() {
// Elements
const main = $('#content');
const headBars = $All(main, 'tr>td[align]');
// Page Info
const rid = Number(getUrlArgv('rid'));
const aid = getUrlArgv('aid') ? Number(getUrlArgv('aid')) : Number($(main, 'td[width]>a').href.match(/(\d+)\.html?$/)[1]);
const page = Number($('#pagelink strong').innerText);
const title = $(main, 'th>strong').textContent;
// URL correction
correctURL();
// Enhancements
pageStatus();
downloader();
sideButtons();
beautifier();
floorEnhance();
autoRefresh();
addFavorite();
addUnlock();
function correctURL() {
(getUrlArgv('page') === 'last' || !getUrlArgv('page')) && setPageUrl(URL_REVIEWSHOW.replace('{A}', aid).replace('{R}', rid).replace('{P}', page));
}
function sideButtons() {
// Last page
SPanel.add({
faicon: 'fa-solid fa-angles-right',
tip: '最后一页',
onclick: (e) => {findclick('#pagelink>.last');}
});
// Next page
SPanel.add({
faicon: 'fa-solid fa-angle-right',
tip: '下一页',
onclick: (e) => {findclick('#pagelink>.next');}
});
// Previous page
SPanel.add({
faicon: 'fa-solid fa-angle-left',
tip: '上一页',
onclick: (e) => {findclick('#pagelink>.prev');}
});
// First page
SPanel.add({
faicon: 'fa-solid fa-angles-left',
tip: '第一页',
onclick: (e) => {findclick('#pagelink>.first');}
});
function findclick(selector) {return $(selector) && $(selector).click();}
}
function beautifier() {
// GUI
const span = $CrE('span');
const check = $CrE('input');
check.type = 'checkbox';
check.checked = CONFIG.BeautifierCfg.getConfig().reviewshow.beautiful;
span.innerHTML = TEXT_GUI_REVIEW_BEAUTIFUL;
span.classList.add(CLASSNAME_BUTTON);
span.style.marginLeft = '0.5em';
span.addEventListener('click', toggleBeautiful);
check.addEventListener('click', toggleBeautiful);
settip(span, TEXT_TIP_REVIEW_BEAUTIFUL);
settip(check, TEXT_TIP_REVIEW_BEAUTIFUL);
headBars[0].appendChild(span);
headBars[0].appendChild(check);
CONFIG.BeautifierCfg.getConfig().reviewshow.beautiful && beautiful();
function toggleBeautiful(e) {
// stop event
destroyEvent(e);
// Togle & save to config
const config = CONFIG.BeautifierCfg.getConfig();
config.reviewshow.beautiful = !config.reviewshow.beautiful;
CONFIG.BeautifierCfg.saveConfig(config);
setTimeout(() => {check.checked = config.reviewshow.beautiful;}, 0);
alertify.notify(config.reviewshow.beautiful ? TEXT_ALT_BEAUTIFUL_ON : TEXT_ALT_BEAUTIFUL_OFF);
// beautifier
config.reviewshow.beautiful ? beautiful() : recover();
}
function beautiful() {
const config = CONFIG.BeautifierCfg.getConfig();
addStyle(CSS_REVIEWSHOW
.replaceAll('{BGI}', config.backgroundImage)
.replaceAll('{S}', config.textScale)
, 'beautifier');
scaleimgs();
hookPosition();
function scaleimgs() {
const imgs = $All('.divimage>img');
const w = main.clientWidth * 0.8 - 3; // td.width = "80%", .even {padding: 3px;}
for (const img of imgs) {
img.width = w;
}
}
}
function recover() {
addStyle('', 'beautifier');
restorePosition();
}
function hookPosition() {
if (!CONFIG.BeautifierCfg.getConfig().reviewshow.beautiful) {return false;};
if (typeof(UBBEditor) !== 'object') {
hookPosition.wait = hookPosition.wait ? hookPosition.wait : 0;
if (++hookPosition.wait > 50) {return false;}
DoLog('beautiful/hookPosition: UBBEditor not loaded, waiting...');
setTimeout(hookPosition, NUMBER_ELEMENT_LOADING_WAIT_INTERVAL);
return false;
}
UBBEditor.GetPosition_BK = UBBEditor.GetPosition;
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 restorePosition() {
if (typeof(UBBEditor) !== 'object') {return false;};
if (!UBBEditor.GetPosition_BK) {return false;};
UBBEditor.GetPosition = UBBEditor.GetPosition_BK;
}
}
function pageStatus() {
window.addEventListener('load', () => {
// Recover page status
applyPageStatus();
// Record the current page status of current review
setInterval(recordPage, 1000);
});
}
// Apply page status sored in history record
function applyPageStatus() {
const config = CONFIG.BkReviewPrefs.getConfig();
const history = config.history;
const pageHist = history[rid];
// Scroll to the last position
if (pageHist && pageHist.page === page) {
// Check if time expired
if (pageHist.time && (new Date()).getTime() - pageHist.time < 30*1000) {
// Do not scroll when opening a positioned link(http[s]://.../...#yidxxxxxx)
if (/#yid\d+$/.test(location.href)) {return;}
// Scroll
pageHist.scrollX !== undefined && window.scrollTo(pageHist.scrollX, pageHist.scrollY);
pageHist.contentsclX !== undefined && main.scrollTo(pageHist.contentsclX, pageHist.contentsclY);
} else {
// Delete expired record
delete history[rid];
CONFIG.BkReviewPrefs.saveConfig(config);
}
}
}
function recordPage() {
const config = CONFIG.BkReviewPrefs.getConfig();
const history = config.history;
// Save page history
config.history[rid] = {
rid: rid,
aid: aid,
page: page,
href: URL_REVIEWSHOW.replace('{R}', String(rid)).replace('{A}', String(aid)).replace('{P}', String(page)),
scrollX: window.pageXOffset,
scrollY: window.pageYOffset,
contentsclX: main.scrollLeft,
contentsclY: main.scrollTop,
time: (new Date()).getTime()
}
CONFIG.BkReviewPrefs.saveConfig(config);
}
function floorEnhance() {
const floors = getAllFloors();
floors.forEach((f)=>(correctFloorLink(f)));
for (const floor of floors) {
alinkEdit(floor);
addQuoteBtn(floor);
addQueryBtn(floor);
addRemark(floor);
alinktofloor(floor.table);
}
}
function alinktofloor(parent=main) {
const floorLinks = $All(main, `a[name][href^="https://${location.host}/modules/article/reviewshow.php"][href*="#yid"]`);
for (const a of $All(parent, 'a')) {
if (!a.href.match(/^https?:\/\/www\.wenku8\.(net|cc)\/modules\/article\/reviewshow\.php\?(&?rid=\d+|&?aid=\d+|&?page=\d+){1,4}#yid\d+$/)) {continue;};
for (const flink of floorLinks) {
if (isSameReply(a, flink)) {
// Set scroll target
a.targetNode = flink;
while (a.targetNode.nodeName !== 'TABLE') {
a.targetNode = a.targetNode.parentElement;
}
// Scroll when clicked
a.addEventListener('click', (e) => {
destroyEvent(e);
e.currentTarget.targetNode.scrollIntoView();
})
};
}
}
function isSameReply(link1, link2) {
const url1 = link1.href.toLowerCase().replace('http://', 'https://');
const url2 = link2.href.toLowerCase().replace('http://', 'https://');
const rid1 = getUrlArgv({url: url1, name: 'rid', defaultValue: null});
const yid1 = url1.match(/#yid(\d+)/) ? url1.match(/#yid(\d+)/)[1] : null;
const rid2 = getUrlArgv({url: url2, name: 'rid', defaultValue: null});
const yid2 = url2.match(/#yid(\d+)/) ? url2.match(/#yid(\d+)/)[1] : null;
return rid1 === rid2 && yid1 === yid2;
}
}
function alinkEdit(parent=document) {
const eLinks = $All(`a[href^="https://${location.host}/modules/article/reviewedit.php?yid="]`);
for (const eLink of eLinks) {
eLink.addEventListener('click', (e) => {
// NO e.stopPropagation() here. Just hooks the open action.
e.preventDefault();
// Open editor dialog
openDialog(e.target.href + '&ajax_gets=jieqi_contents');
// Show mask if mask not shown
!document.getElementById("mask") && showMask();
})
}
}
function autoRefresh() {
let working=false, interval=0;
const pagelink = $('#pagelink');
const tdLink = pagelink.parentElement;
const trContainer = tdLink.parentElement;
const tdAutoRefresh = $CrE('td');
const chkAutoRefresh = $CrE('input');
const txtAutoRefresh = $CrE('span');
const txtPaused = $CrE('span');
const ptitle = $('#ptitle');
const pcontent = $('#pcontent');
txtAutoRefresh.innerText = TEXT_GUI_AUTOREFRESH;
txtAutoRefresh.classList.add(CLASSNAME_BUTTON);
txtAutoRefresh.addEventListener('click', toggleRefresh);
chkAutoRefresh.addEventListener('click', toggleRefresh);
chkAutoRefresh.type = 'checkbox';
chkAutoRefresh.checked = false;
txtPaused.innerText = '';
txtPaused.classList.add(CLASSNAME_TEXT);
txtPaused.style.marginLeft = '0.5em';
tdAutoRefresh.style.align = 'left';
tdAutoRefresh.appendChild(txtAutoRefresh);
tdAutoRefresh.appendChild(chkAutoRefresh);
tdAutoRefresh.appendChild(txtPaused);
trContainer.insertBefore(tdAutoRefresh, tdLink);
// Apply config
CONFIG.BkReviewPrefs.getConfig().autoRefresh ? toggleRefresh() : function() {};
/* No pauses after v1.5.7
// Show pause
// Note: Blur event triggers after Focus event was triggered
for (const editElm of [ptitle, pcontent]) {
if (!editElm) {continue;};
editElm.addEventListener('blur', (e) => {
txtPaused.innerText = '';
});
editElm.addEventListener('focus', (e) => {
txtPaused.innerText = TEXT_GUI_AUTOREFRESH_PAUSED;
});
}
*/
function toggleRefresh(e) {
// stop event
destroyEvent(e);
// Not in last Page, no auto refresh
if (!isCurLastPage() && !working) {
const box = alertify.notify(TEXT_ALT_AUTOREFRESH_NOTLAST);
box.callback = (isClicked) => {isClicked && (location.href = $('#pagelink>a.last').href);};
return false;
}
// toggle
working = !working;
working ? interval = setInterval(refresh, 20*1000) : clearInterval(interval);
working && refresh();
// Save to config
const review = CONFIG.BkReviewPrefs.getConfig();
review.autoRefresh = working;
CONFIG.BkReviewPrefs.saveConfig(review);
setTimeout(() => {chkAutoRefresh.checked = working;}, 0);
alertify.notify(working ? TEXT_ALT_AUTOREFRESH_ON : TEXT_ALT_AUTOREFRESH_OFF);
}
function refresh() {
const box = alertify.notify(TEXT_ALT_AUTOREFRESH_WORKING);
const url = URL_REVIEWSHOW.replace('{R}', String(rid)).replace('{A}', String(aid)).replace('{P}', 'last');
getDocument(url, refreshLoaded, url);
function refreshLoaded(oDoc, pageurl) {
// Clost alert box
box.exist ? box.close.apply(box) : function() {};
// Update all existing floor content (and title)
const nowfloors = $All('#content>table[class="grid"]');
const newfloors = $All(oDoc, '#content>table[class="grid"]');
let i, modified = false;
for (i = 1; i < Math.min(nowfloors.length, newfloors.length); i++) {
isFloorTable(nowfloors[i]) && isFloorTable(newfloors[i]) && getFloorNumber(nowfloors[i]) === getFloorNumber(newfloors[i]) && updatefloor(i);
}
modified && alertify.notify(TEXT_ALT_AUTOREFRESH_MODIFIED);
const newtop = getTopFloorNumber(oDoc);
const nowtop = getTopFloorNumber(document);
if (unsafeWindow.isPY_DNG && newtop === 9899) {
sendReviewReply({rid: rid, title: '测试标题', content: '测试内容'});
}
if (newtop > nowtop) {
const newmain = $(oDoc, '#content');
const eleLastPage = $(oDoc, '#pagelink a.last');
const urlLastPage = newmain.url = eleLastPage.href;
const newpage = Number(getUrlArgv({url: urlLastPage, name: 'page'}));
const newfloors = getAllFloors(newmain);
const nowfloors = getAllFloors();
if (newpage === page) {
// In same page, append floors
for (let i = nowfloors.length; i < newfloors.length; i++) {
const floor = newfloors[i];
appendfloor(floor);
}
} else {
// In New page, remake floors
let box = alertify.notify(TEXT_ALT_AUTOREFRESH_APPLIED);
// Remove old floors
for (const oldfloor of nowfloors) {
oldfloor.table.parentElement.removeChild(oldfloor.table);
}
// Append new floors
for (const newfloor of newfloors) {
appendfloor(newfloor);
}
// Remake #pagelink
$(main, '#pagelink').innerHTML = $(newmain, '#pagelink').innerHTML;
// Reset location.href
page !== 'last' && setPageUrl(urlLastPage);
return true;
}
} else {
alertify.message(TEXT_ALT_AUTOREFRESH_NOMORE);
return false;
}
function updatefloor(i) {
const nowfloor = nowfloors[i];
const newfloor = newfloors[i];
const nowTitle = getEleFloorTitle(nowfloor);
const newTitle = getEleFloorTitle(newfloor);
const nowContent = getEleFloorContent(nowfloor);
const newContent = getEleFloorContent(newfloor);
if (nowTitle.innerHTML !== newTitle.innerHTML) {
nowTitle.innerHTML = newTitle.innerHTML;
nowTitle.classList.add(CLASSNAME_MODIFIED);
nowTitle.addEventListener('click', (e) => {e.currentTarget.classList.remove(CLASSNAME_MODIFIED);});
modified = true;
}
if (getFloorContent(nowContent) !== getFloorContent(newContent)) {
nowContent.innerHTML = newContent.innerHTML;
nowContent.classList.add(CLASSNAME_MODIFIED);
nowContent.addEventListener('click', (e) => {e.currentTarget.classList.remove(CLASSNAME_MODIFIED);});
modified = true;
}
if (modified) {
alinktofloor(nowfloor);
}
}
}
}
function isCurLastPage() {
return $('#pagelink>strong').innerText === $('#pagelink>a.last').innerText;
}
function getTopFloorNumber(oDoc) {
const tblfloors = $All(oDoc, '#content>table[class="grid"]');
for (let i = tblfloors.length-1; i >= 0; i--) {
const tbllast = tblfloors[i];
if (isFloorTable(tbllast)) {return getFloorNumber(tbllast);}
}
return null;
}
}
function correctFloorLink(floor) {
floor.hrefa.href = floor.href;
}
function addFavorite() {
// Create GUI
const spliter = $CrE('span');
const favorBtn = $CrE('span');
const favorChk = $CrE('input');
spliter.style.marginLeft = '1em';
favorBtn.innerText = TEXT_GUI_REVIEW_ADDFAVORITE;
favorBtn.classList.add(CLASSNAME_BUTTON);
favorChk.type = 'checkbox';
favorChk.checked = CONFIG.BkReviewPrefs.getConfig().favorites.hasOwnProperty(rid);
favorBtn.addEventListener('click', checkChange);
favorChk.addEventListener('change', checkChange);
headBars[0].appendChild(spliter);
headBars[0].appendChild(favorBtn);
headBars[0].appendChild(favorChk);
function checkChange(e) {
if (e && e.target === favorChk) {
destroyEvent(e);
}
let inFavorites;
const config = CONFIG.BkReviewPrefs.getConfig();
if (config.favorites.hasOwnProperty(rid)) {
delete config.favorites[rid];
inFavorites = false;
} else {
config.favorites[rid] = {
rid: rid,
name: title,
href: URL_REVIEWSHOW_3.replace('{R}', rid).replace('{A}', aid),
time: (new Date()).getTime(), // time added in version 1.6.7
tiptitle: null
};
inFavorites = true;
}
CONFIG.BkReviewPrefs.saveConfig(config);
setTimeout(() => {favorChk.checked = inFavorites;}, 0);
alertify.notify((inFavorites ? TEXT_GUI_REVIEW_FAVORADDED : TEXT_GUI_REVIEW_FAVORDELED).replace('{N}', title));
}
function updateFavorite() {
const config = CONFIG.BkReviewPrefs.getConfig();
if (config.favorites.hasOwnProperty(rid)) {
config.favorites[rid] = {
rid: rid,
name: title,
href: URL_REVIEWSHOW_3.replace('{R}', rid).replace('{A}', aid)
};
}
}
}
function addQuoteBtn(floor) {
const table = floor.table;
const numberEle = $(table, 'td.even div a');
const attr = numberEle.parentElement;
const btn = createQuoteBtn(attr);
const spliter = document.createTextNode(' | ');
attr.insertBefore(spliter, numberEle);
attr.insertBefore(btn, spliter);
function createQuoteBtn() {
// Get content textarea
const pcontent = $('#pcontent');
const form = $(`form[action^="https://${location.host}/modules/article/review"]`);
// Create button
const btn = $CrE('span');
btn.classList.add(CLASSNAME_BUTTON);
btn.addEventListener('click', quoteThisFloor);
btn.innerHTML = '引用';
const tip_panel = $CrE('div');
tip_panel.insertAdjacentText('afterbegin', '或者,');
const btn_qtnum = $CrE('span');
btn_qtnum.classList.add(CLASSNAME_BUTTON);
btn_qtnum.addEventListener('click', quoteFloorNum);
btn_qtnum.innerHTML = '仅引用序号';
tip_panel.appendChild(btn_qtnum);
const panel = tippy(btn, {
content: tip_panel,
theme: 'wenku_tip',
placement: 'top',
interactive: true,
});
return btn;
function quoteThisFloor() {
// In DOM Events, <this> keyword points to the Event Element.
const numberEle = $(this.parentElement, 'a[name]');
const numberText = numberEle.innerText;
const url = URL_REVIEWSHOW_4.replace('{R}', rid).replace('{P}', page).replace('{Y}', numberEle.name);
const contentEle = $(this.parentElement.parentElement, 'hr+div');
const content = getFloorContent(contentEle);
const insertPosition = pcontent.selectionEnd;
const text = pcontent.value;
const leftText = text.substr(0, insertPosition);
const rightText = text.substr(insertPosition);
// Create insert value
let insertValue = '[url=U]N[/url] [quote]Q[/quote]';
insertValue = insertValue.replace('U', url).replace('N', numberText).replace('Q', content);
// if not at the beginning of a line then insert a whitespace before the link
insertValue = ((leftText.length === 0 || /[ \r\n]$/.test(leftText)) ? '' : ' ') + insertValue;
// if not at the end of a line then insert a whitespace after the link
insertValue += (rightText.length === 0 || /^[ \r\n]/.test(rightText)) ? '' : ' ';
pcontent.value = leftText + insertValue + rightText;
const position = insertPosition + (pcontent.value.length - text.length);
form.scrollIntoView(); pcontent.focus(); pcontent.setSelectionRange(position, position);
}
function quoteFloorNum() {
// In DOM Events, <this> keyword points to the Event Element.
const numberEle = $(this.parentElement.parentElement.parentElement.parentElement.parentElement, 'a[name]');
const numberText = numberEle.innerText;
const url = URL_REVIEWSHOW_4.replace('{R}', rid).replace('{P}', page).replace('{Y}', numberEle.name);
const contentEle = $(this.parentElement.parentElement.parentElement.parentElement.parentElement.parentElement, 'hr+div');
const insertPosition = pcontent.selectionEnd;
const text = pcontent.value;
const leftText = text.substr(0, insertPosition);
const rightText = text.substr(insertPosition);
// Create insert value
let insertValue = '[url=U]N[/url]';
insertValue = insertValue.replace('U', url).replace('N', numberText);
// if not at the beginning of a line then insert a whitespace before the link
insertValue = ((leftText.length === 0 || /[ \r\n]$/.test(leftText)) ? '' : ' ') + insertValue;
// if not at the end of a line then insert a whitespace after the link
insertValue += (rightText.length === 0 || /^[ \r\n]/.test(rightText)) ? '' : ' ';
pcontent.value = leftText + insertValue + rightText;
const position = insertPosition + (pcontent.value.length - text.length);
form.scrollIntoView(); pcontent.focus(); pcontent.setSelectionRange(position, position);
}
}
}
function addQueryBtn(floor) {
// Get container div
const div = floor.leftdiv;
// Create buttons
const qBtn = $CrE('a'); // Button for query reviews
const iBtn = $CrE('a'); // Button for query userinfo
const mBtn = $CrE('a'); // Button for edit user remark
// Get UID
const user = $(div, 'a');
const name = user.innerText;
const UID = Math.floor(user.href.match(/uid=(\d+)/)[1]);
// Create text spliter
const spliter = document.createTextNode(' | ');
// Config buttons
qBtn.href = URL_REVIEWSEARCH.replaceAll('{K}', String(UID));
iBtn.href = URL_USERINFO .replaceAll('{K}', String(UID));
mBtn.href = 'javascript: void(0);'
qBtn.target = '_blank';
iBtn.target = '_blank';
mBtn.addEventListener('click', editUserRemark.bind(null, UID, name, reloadRemarks));
qBtn.innerText = TEXT_GUI_USER_REVIEWSEARCH;
iBtn.innerText = TEXT_GUI_USER_USERINFO;
mBtn.innerText = TEXT_GUI_USER_USERREMARKEDIT;
// Append to GUI
div.appendChild($CrE('br'));
div.appendChild(iBtn);
div.appendChild(qBtn);
div.insertBefore(spliter, qBtn);
div.appendChild($CrE('br'));
div.appendChild(mBtn);
function reloadRemarks() {
const floors = getAllFloors();
floors.forEach((f) => (addRemark(f)));
}
}
function addRemark(floor) {
// Get container div
const div = floor.leftdiv;
const strong = $(div, 'strong');
// Get config
const config = CONFIG.RemarksConfig.getConfig();
const uid = Math.floor($(div, 'strong>a').href.match(/\?uid=(\d+)/)[1]);
const user = (config.user[uid] || {});
if ($(div, '.user-remark')) {
// Edit remark displayer
const name = $(div, '.user-remark-remark');
name.innerText = user.remark || TEXT_GUI_USER_USERREMARKEMPTY;
name.style.color = user.remark ? 'black' : 'grey';
} else {
// Add remark displayer
const container = $CrE('span');
const br = $CrE('br');
const name = $CrE('span');
container.classList.add('user-remark');
container.classList.add(CLASSNAME_TEXT);
container.innerText = TEXT_GUI_USER_USERREMARKSHOW;
name.innerText = user.remark || TEXT_GUI_USER_USERREMARKEMPTY;
name.style.color = user.remark ? 'black' : 'grey';
name.classList.add('user-remark-remark');
container.appendChild(name);
strong.insertAdjacentElement('afterend', br);
br.insertAdjacentElement('afterend', container);
}
}
// Provide a hidden function to reply overtime book-reviews
function addUnlock() {
listen();
function listen() {
if ($('#pcontent')) {return;}
const target = $('#content>table>caption+tbody>tr>td:nth-child(2)');
let count = 0;
document.addEventListener('click', function hidden_unlocker(e) {
e.target === target ? count++ : (count = 0);
count >= 10 && add();
count >= 10 && document.removeEventListener('click', hidden_unlocker);
count >= 10 && (target.innerHTML = TEXT_GUI_REVIEW_UNLOCK_WARNING);
});
}
function add() {
const container = $CrE('div');
$('#content').appendChild(container);
makeEditor(container, rid, aid);
}
}
// Reply without refreshing the document
function hookReply() {
const form = $('form[name="frmreview"]');
const onsubmit = form.onsubmit;
form.onsubmit = function() {
const title = $(form, '#ptitle').value;
const content = $(form, '#pcontent').value;
(onsubmit ? onsubmit() : true) && sendReviewReply({
rid: rid, title: title, content: content,
onload: function(oDoc) {
// Make floor(s)
},
onerror: function(e) {
DoLog(LogLevel.Error, 'pageReview/hookReply: submit onerror.');
}
});
};
}
function downloader() {
// GUI
const pageCountText = $('#pagelink>.last').href.match(/page=(\d+)/)[1];
const lefta = $(headBars[0], 'a');
const lefttext = document.createTextNode('书评回复');
clearChildnodes(headBars[0]);
headBars[0].appendChild(lefta);
headBars[0].appendChild(lefttext);
headBars[0].width = '45%';
headBars[1].width = '55%';
const saveBtn = $CrE('span');
saveBtn.innerText = TEXT_GUI_DOWNLOAD_REVIEW.replaceAll('A', pageCountText);
saveBtn.classList.add(CLASSNAME_BUTTON);
saveBtn.addEventListener('click', downloadWholePost);
headBars[1].appendChild(saveBtn);
const spliter = $CrE('span');
const bbcdTxt = $CrE('span');
const bbcdChk = $CrE('input');
spliter.style.marginLeft = '1em';
bbcdTxt.innerText = TEXT_GUI_DOWNLOAD_BBCODE;
bbcdChk.type = 'checkbox';
bbcdChk.checked = CONFIG.BkReviewPrefs.getConfig().bbcode;
bbcdTxt.addEventListener('click', bbcodeOnclick);
bbcdChk.addEventListener('click', bbcodeOnclick);
settip(bbcdTxt, TEXT_TIP_DOWNLOAD_BBCODE);
settip(bbcdChk, TEXT_TIP_DOWNLOAD_BBCODE);
bbcdTxt.classList.add(CLASSNAME_BUTTON);
headBars[1].appendChild(spliter);
headBars[1].appendChild(bbcdTxt);
headBars[1].appendChild(bbcdChk);
function bbcodeOnclick(e) {
destroyEvent(e);
if (downloadWholePost.working) {
alertify.warning(TEXT_ALT_DOWNLOAD_BBCODE_NOCHANGE);
return false;
}
const cmConfig = CONFIG.BkReviewPrefs.getConfig();
cmConfig.bbcode = !cmConfig.bbcode;
setTimeout(() => {bbcdChk.checked = cmConfig.bbcode;}, 0);
CONFIG.BkReviewPrefs.saveConfig(cmConfig);
}
// ## Function: Get data from page document or join it into the given data variable ##
function getDataFromPage(document, data) {
let i;
DoLog(LogLevel.Info, document, true);
// Get Floors; avatars uses for element locating
const main = $(document, '#content');
const avatars = $All(main, 'table div img.avatar');
// init data, floors and users if need
let floors = {}, users = {};
if (data) {
floors = data.floors;
users = data.users;
} else {
data = {};
initData(data, floors, users);
}
for (i = 0; i < avatars.length; i++) {
const floor = newFloor(floors, avatars, i);
const elements = getFloorElements(floor);
const reply = getFloorReply(floor);
const user = getFloorUser(floor);
appendFloor(floors, floor);
}
return data;
function initData(data, floors, users) {
// data vars
data.floors = floors; floors.data = data;
data.users = users; users.data = data;
// review info
data.link = location.href;
data.id = getUrlArgv({name: 'rid', dealFunc: Number, defaultValue: 0});
data.page = getUrlArgv({name: 'page', dealFunc: Number, defaultValue: 1});
data.title = $(main, 'th strong').innerText;
return data;
}
function newFloor(floors, avatars, i) {
const floor = {};
floor.avatar = avatars[i];
floor.floors = floors;
return floor;
}
function getFloorElements(floor) {
const elements = {}; floor.elements = elements;
elements.avatar = floor.avatar;
elements.table = elements.avatar.parentElement.parentElement.parentElement.parentElement.parentElement;
elements.tr = $(elements.table, 'tr');
elements.tdUser = $(elements.table, 'td.odd');
elements.tdReply = $(elements.table, 'td.even');
elements.divUser = $(elements.tdUser, 'div');
elements.aUser = $(elements.divUser, 'a');
elements.attr = $(elements.tdReply, 'div a').parentElement;
elements.time = elements.attr.childNodes[0];
elements.number = $(elements.attr, 'a[name]');
elements.title = $(elements.tdReply, 'div>strong');
elements.content = $(elements.tdReply, 'hr+div');
return elements;
}
function getFloorReply(floor) {
const elements = floor.elements;
const reply = {}; floor.reply = reply;
reply.time = elements.time.nodeValue.match(/\d{4}-\d{2}-\d{2} \d{2}:\d{2}:\d{2}/)[0];
reply.number = Number(elements.number.innerText.match(/\d+/)[0]);
reply.value = CONFIG.BkReviewPrefs.getConfig().bbcode ? getFloorContent(elements.content, true) : elements.content.innerText;
reply.title = elements.title.innerText;
return reply;
}
function getFloorUser(floor) {
const elements = floor.elements;
const user = {}; floor.user = user;
user.id = elements.aUser.href.match(/uid=(\d+)/)[1];
user.name = elements.aUser.innerText;
user.avatar = elements.avatar.src;
user.link = elements.aUser.href;
user.jointime = elements.divUser.innerText.match(/\d{4}-\d{2}-\d{2}/)[0];
const data = floor.floors.data; const users = data.users;
if (!users.hasOwnProperty(user.id)) {
users[user.id] = user;
user.floors = [floor];
} else {
const uFloors = users[user.id].floors;
uFloors.push(floor);
sortUserFloors(uFloors);
}
return user;
}
function sortUserFloors(uFloors) {
uFloors.sort(function(F1, F2) {
return F1.reply.number - F2.reply.number;
})
}
function appendFloor(floors, floor) {
floors[floor.reply.number-1] = floor;
}
}
// ## Function: Get pages and parse each page to a data, returns data ##
// callback(data, gotcount, finished) is called when xhr and parsing completed
function getAllPages(callback) {
let i, data, gotcount = 0;
const ridMatcher = /rid=(\d+)/, pageMatcher = /page=(\d+)/;
const lastpageUrl = $('#pagelink>.last').href;
const rid = Number(lastpageUrl.match(ridMatcher)[1]);
const pageCount = Number(lastpageUrl.match(pageMatcher)[1]);
const curPageNum = location.href.match(pageMatcher) ? Number(location.href.match(pageMatcher)[1]) : 1;
for (i = 1; i <= pageCount; i++) {
const url = lastpageUrl.replace(pageMatcher, 'page='+String(i));
getDocument(url, joinPageData, callback);
}
function joinPageData(pageDocument, callback) {
data = getDataFromPage(pageDocument, data);
gotcount++;
// log
const level = gotcount % NUMBER_LOGSUCCESS_AFTER ? LogLevel.Info : LogLevel.Success;
DoLog(level, 'got ' + String(gotcount) + ' pages.');
if (gotcount === pageCount) {
DoLog(LogLevel.Success, 'All pages xhr and parsing completed.');
DoLog(LogLevel.Success, data, true);
}
// callback
if (callback) {callback(data, gotcount, gotcount === pageCount);};
}
}
// Function output
function joinTXT(data, noSpliter=true) {
const floors = data.floors; const users = data.users;
// HEAD META DATA
const saveTime = getTime();
const head = TEXT_OUTPUT_REVIEW_HEAD
.replaceAll('{RWID}', data.id).replaceAll('{RWTT}', data.title).replaceAll('{RWLK}', data.link)
.replaceAll('{SVTM}', saveTime).replaceAll('{SCNM}', GM_info.script.name)
.replaceAll('{VRSN}', GM_info.script.version).replaceAll('{ATNM}', GM_info.script.author);
// join userinfos
let userText = '';
for (const [pname, user] of Object.entries(users)) {
if (!isNumeric(pname)) {continue;};
userText += TEXT_OUTPUT_REVIEW_USER
.replaceAll('{LNSPLT}', noSpliter ? '' : TEXT_SPLIT_LINE).replaceAll('{USERNM}', user.name)
.replaceAll('{USERID}', user.id).replaceAll('{USERJT}', user.jointime)
.replaceAll('{USERLK}', user.link).replaceAll('{USERFL}', user.floors[0].reply.number);
userText += '\n'.repeat(2);
}
// join floors
let floorText = '';
for (const [pname, floor] of Object.entries(floors)) {
if (!isNumeric(pname)) {continue;};
const avatar = floor.avatar; const elements = floor.elements; const user = floor.user; const reply = floor.reply;
floorText += TEXT_OUTPUT_REVIEW_FLOOR
.replaceAll('{LNSPLT}', noSpliter ? '' : TEXT_SPLIT_LINE).replaceAll('{RPNUMB}', String(reply.number))
.replaceAll('{RPTIME}', reply.time).replaceAll('{USERNM}', user.name)
.replaceAll('{USERID}', user.id).replaceAll('{RPTEXT}', reply.value);
floorText += '\n'.repeat(2);
}
// End
const foot = TEXT_OUTPUT_REVIEW_END;
// return
const txt = head + '\n'.repeat(2) + userText + '\n'.repeat(2) + floorText + '\n'.repeat(2) + foot;
return txt;
}
// ## Function: Download the whole post ##
function downloadWholePost() {
// Continues only if not working
if (downloadWholePost.working) {return;};
downloadWholePost.working = true;
bbcdTxt.classList.add(CLASSNAME_DISABLED);
// GUI
saveBtn.innerText = TEXT_GUI_DOWNLOADING_REVIEW
.replaceAll('C', '0').replaceAll('A', pageCountText);
// go work!
getAllPages(function(data, gotCount, finished) {
// GUI
saveBtn.innerText = TEXT_GUI_DOWNLOADING_REVIEW
.replaceAll('C', String(gotCount)).replaceAll('A', pageCountText);
// Stop here if not completed
if (!finished) {return;};
// Join text
const TXT = joinTXT(data);
// Download
const blob = new Blob([TXT],{type:"text/plain;charset=utf-8"});
const url = URL.createObjectURL(blob);
const name = '文库贴 - ' + data.title + ' - ' + data.id.toString() + '.txt';
const a = $CrE('a');
a.href = url;
a.download = name;
a.click();
// GUI
saveBtn.innerText = TEXT_GUI_DOWNLOADFINISH_REVIEW;
alertify.success(TEXT_ALT_DOWNLOADFINISH_REVIEW.replaceAll('{T}', data.title).replaceAll('{I}', data.id).replaceAll('{N}', name));
// Work finish
downloadWholePost.working = false;
bbcdTxt.classList.remove(CLASSNAME_DISABLED);
})
}
}
// Get all floor object
/* Contains:
** floor.table
** floor.tbody
** floor.tr
** floor.lefttd
** floor.righttd
** floor.leftdiv
** floor.titlediv
** floor.titlestrong
** floor.metadiv
** floor.replydiv
*/
function getAllFloors(parent=main) {
const avatars = $All(parent, 'table div img.avatar');
const floors = [];
for (const avt of avatars) {
const floor = {};
floor.leftdiv = avt.parentElement;
floor.lefttd = floor.leftdiv.parentElement;
floor.tr = floor.lefttd.parentElement
floor.righttd = floor.tr.children[1];
floor.titlediv = floor.righttd.children[0];
floor.titlestrong = floor.titlediv.children[0];
floor.metadiv = floor.righttd.children[1];
floor.replydiv = floor.righttd.children[3];
floor.hrefa = $(floor.metadiv, 'a[name]');
floor.tbody = floor.tr.parentElement;
floor.table = floor.tbody.parentElement;
floor.rid = Number(getUrlArgv({url: parent.url || location.href, name: 'rid'}));
floor.aid = Number($(parent, 'td[width]>a').href.match(/(\d+)\.html?$/)[1]);
floor.page = Number($(avt.ownerDocument, '#pagelink strong').innerText);
floor.pagehref = URL_REVIEWSHOW.replace('{R}', floor.rid.toString()).replace('{A}', floor.aid.toString()).replace('{P}', floor.page.toString());
floor.href = URL_REVIEWSHOW_5.replace('{R}', floor.rid.toString()).replace('{A}', floor.aid.toString()).replace('{P}', floor.page.toString()).replace('{Y}', floor.hrefa.name);
floors.push(floor);
}
return floors;
}
// Validate a <table> element whether is a floor
function isFloorTable(tbl) {
return $(tbl, 'a[href*="#yid"][name^="yid"]') ? true : false;
}
// Get floor title element (<strong>)
// Argv: <table> element of the floor
function getEleFloorTitle(tblfloor) {
return $(tblfloor, 'td.even>div:first-child>strong'); // or :nth-child(1)
}
// Get floor content element (<div>)
// Argv: <table> element of the floor
function getEleFloorContent(tblfloor) {
return $(tblfloor, 'td.even>hr+div');
}
// Get the floor number
// Argv: <table> element of the floor
function getFloorNumber(tblfloor) {
const eleNumber = $(tblfloor, 'a[name^="yid"]');
return eleNumber ? Number(eleNumber.innerText.match(/\d+/)[0]) : false;
}
// 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;
}
}
}
// Append floor to #content
function appendfloor(floor) {
// Append
const table = floor.table;
const elmafter = $(main, 'table.grid+table[border]');
main.insertBefore(table, elmafter);
// Enhances
correctFloorLink(floor);
alinkEdit(floor);
addQuoteBtn(floor);
addQueryBtn(floor);
addRemark(floor);
alinktofloor(floor.table);
}
}
// Bookcase page add-on
function pageBookcase() {
// Get auto-recommend config
let arConfig = CONFIG.AutoRecommend.getConfig();
// Get bookcase lists
const bookCaseURL = `https://${location.host}/modules/article/bookcase.php?classid={CID}`;
const content = $('#content');
const selector = $('[name="classlist"]');
const options = selector.children;
// Current bookcase
const curForm = $(content, '#checkform');
const curClassid = Number($('[name="clsssid"]').value);
// Init bookcase config if need
initPreferences();
const bookcases = CONFIG.bookcasePrefs.getConfig().bookcases;
addTopTitle();
decorateForm(curForm, bookcases[curClassid]);
// gowork
laterReads();
showBookcases();
recommendAllGUI();
function recommendAllGUI() {
const block = createWenkuBlock({
type: 'mypage',
parent: '#left',
title: TEXT_GUI_BOOKCASE_ATRCMMD,
items: [
{innerHTML: arConfig.allCount === 0 ? TEXT_GUI_BOOKCASE_RCMMDNW_NOTASK : (TASK.AutoRecommend.checkRcmmd() ? TEXT_GUI_BOOKCASE_RCMMDNW_DONE : TEXT_GUI_BOOKCASE_RCMMDNW_NOTYET), id: 'arstatus'},
{innerHTML: TEXT_GUI_BOOKCASE_RCMMDAT, id: 'autorcmmd'},
{innerHTML: TEXT_GUI_BOOKCASE_RCMMDNW, id: 'rcmmdnow'}
]
});
// Configure buttons
const ulitm = $(block, '.ulitem');
const txtst = $(block, '#arstatus');
const btnAR = $(block, '#autorcmmd');
const btnRN = $(block, '#rcmmdnow');
const txtAR = $(block, 'span');
const checkbox = $CrE('input');
txtst.classList.add(CLASSNAME_TEXT);
btnAR.classList.add(CLASSNAME_BUTTON);
btnRN.classList.add(CLASSNAME_BUTTON);
checkbox.type = 'checkbox';
checkbox.checked = arConfig.auto;
checkbox.addEventListener('click', onclick);
btnAR.addEventListener('click', onclick);
btnAR.appendChild(checkbox);
btnRN.addEventListener('click', rcmmdnow);
function onclick(e) {
destroyEvent(e);
arConfig.auto = !arConfig.auto;
setTimeout(function() {checkbox.checked = arConfig.auto;}, 0);
CONFIG.AutoRecommend.saveConfig(arConfig);
alertify.notify(arConfig.auto ? TEXT_ALT_ATRCMMDS_AUTO : TEXT_ALT_ATRCMMDS_NOAUTO);
}
function rcmmdnow() {
if (TASK.AutoRecommend.checkRcmmd() && !confirm(TEXT_GUI_BOOKCASE_RCMMDNW_CONFIRM)) {return false;}
if (arConfig.allCount === 0) {alertify.warning(TEXT_ALT_ATRCMMDS_NOTASK); return false;};
TASK.AutoRecommend.run(true);
}
}
function initPreferences() {
const config = CONFIG.bookcasePrefs.getConfig();
if (config.bookcases.length === 0) {
for (const option of options) {
config.bookcases.push({
classid: Number(option.value),
url: bookCaseURL.replace('{CID}', String(option.value)),
name: option.innerText
});
}
CONFIG.bookcasePrefs.saveConfig(config);
}
}
function addTopTitle() {
// Clone title bar
const checkform = $('#checkform') ? $('#checkform') : $('.'+CLASSNAME_BOOKCASE_FORM);
const oriTitle = $(checkform, 'div.gridtop');
const topTitle = oriTitle.cloneNode(true);
content.insertBefore(topTitle, checkform);
// Hide bookcase selector
const bcSelector = $(topTitle, '[name="classlist"]');
bcSelector.style.display = 'none';
// Write title text
const textNode = topTitle.childNodes[0];
const numMatch = textNode.nodeValue.match(/\d+/g);
const text = TEXT_GUI_BOOKCASE_TOPTITLE.replace('A', numMatch[0]).replace('B', numMatch[1]);
textNode.nodeValue = text;
}
function showBookcases() {
// GUI
const topTitle = $(content, 'script+div.gridtop');
const textNode = topTitle.childNodes[0];
const oriTitleText = textNode.nodeValue;
const allCount = bookcases.length;
let finished = 1;
textNode.nodeValue = TEXT_GUI_BOOKCASE_GETTING.replace('C', String(finished)).replace('A', String(allCount));
// Get all bookcase pages
for (const bookcase of bookcases) {
if (bookcase.classid === curClassid) {continue;};
getDocument(bookcase.url, appendBookcase, [bookcase]);
}
function appendBookcase(mDOM, bookcase) {
const classid = bookcase.classid;
// Get bookcase form and modify it
const form = $(mDOM, '#checkform');
form.parentElement.removeChild(form);
// Find the right place to insert it in
const forms = $All(content, '.'+CLASSNAME_BOOKCASE_FORM);
for (let i = 0; i < forms.length; i++) {
const thisForm = forms[i];
const cid = typeof thisForm.classid === 'number' ? thisForm.classid : curClassid;
if (cid > classid) {
content.insertBefore(form, thisForm);
break;
}
}
if(!form.parentElement) {$('#laterbooks').insertAdjacentElement('beforebegin', form);};
// Decorate
decorateForm(form, bookcase);
// finished increase
finished++;
textNode.nodeValue = finished < allCount ?
TEXT_GUI_BOOKCASE_GETTING.replace('C', String(finished)).replace('A', String(allCount)) :
oriTitleText;
}
}
function decorateForm(form, bookcase) {
const classid = bookcase.classid;
let name = bookcase.name;
// Provide auto-recommand button
arBtn();
// Modify properties
form.classList.add(CLASSNAME_BOOKCASE_FORM);
form.id += String(classid);
form.classid = classid;
form.onsubmit = my_check_confirm;
// Hide bookcase selector
const bcSelector = $(form, '[name="classlist"]');
bcSelector.style.display = 'none';
// Dblclick Change title
const titleBar = bcSelector.parentElement;
titleBar.childNodes[0].nodeValue = name;
titleBar.addEventListener('dblclick', editName);
// Longpress Change title for mobile
let touchTimer;
titleBar.addEventListener('touchstart', () => {touchTimer = setTimeout(editName, 500);});
titleBar.addEventListener('touchmove', () => {clearTimeout(touchTimer);});
titleBar.addEventListener('touchend', () => {clearTimeout(touchTimer);});
titleBar.addEventListener('mousedown', () => {touchTimer = setTimeout(editName, 500);});
titleBar.addEventListener('mouseup', () => {clearTimeout(touchTimer);});
// Show tips
let tip = TEXT_GUI_BOOKCASE_DBLCLICK;
if (tipready) {
// tipshow and tiphide is coded inside wenku8 itself, its function is to show a text tip besides the mouse
titleBar.addEventListener('mouseover', function() {tipshow(tip);});
titleBar.addEventListener('mouseout' , tiphide);
} else {
titleBar.title = tip;
}
// Change selector names
renameSelectors(false);
// Replaces the original check_confirm() function
function my_check_confirm() {
const checkform = this;
let checknum = 0;
for (let i = 0; i < checkform.elements.length; i++){
if (checkform.elements[i].name == 'checkid[]' && checkform.elements[i].checked == true) checknum++;
}
if (checknum === 0){
alert('请先选择要操作的书目!');
return false;
}
const newclassid = $(checkform, '#newclassid');
if(newclassid.value == -1){
if (confirm('确实要将选中书目移出书架么?')) {return true;} else {return false;};
} else {
return true;
}
}
// Selector name refresh
function renameSelectors(renameAll) {
if (renameAll) {
const forms = $All(content, '.'+CLASSNAME_BOOKCASE_FORM);
for (const form of forms) {
renameFormSlctr(form);
}
} else {
renameFormSlctr(form);
}
function renameFormSlctr(form) {
const newclassid = $(form, '#newclassid');
const options = newclassid.children;
for (let i = 0; i < options.length; i++) {
const option = options[i];
const value = Number(option.value);
const bc = bookcases[value];
bc ? option.innerText = TEXT_GUI_BOOKCASE_MOVEBOOK.replace('N', bc.name) : function(){};
}
}
}
// Provide <input> GUI to edit bookcase name
function editName() {
const nameInput = $CrE('input');
const form = this;
tip = TEXT_GUI_BOOKCASE_WHATNAME;
tipready ? tipshow(tip) : function(){};
titleBar.childNodes[0].nodeValue = '';
titleBar.appendChild(nameInput);
nameInput.value = name;
nameInput.addEventListener('blur', onblur);
nameInput.addEventListener('keydown', onkeydown)
nameInput.focus();
nameInput.setSelectionRange(0, name.length);
function onblur() {
tip = TEXT_GUI_BOOKCASE_DBLCLICK;
tipready ? tipobj.innerHTML = tip : function(){};
const value = nameInput.value.trim();
if (value) {
name = value;
bookcase.name = name;
CONFIG.bookcasePrefs.saveConfig(bookcases);
}
titleBar.childNodes[0].nodeValue = name;
try {titleBar.removeChild(nameInput)} catch (DOMException) {};
renameSelectors(true);
}
function onkeydown(e) {
if (e.keyCode === 13) {
e.preventDefault();
onblur();
}
}
}
// Provide auto-recommend option
function arBtn() {
const table = $(form, 'table');
for (const tr of $All(table, 'tr')) {
$(tr, '.odd') ? decorateRow(tr) : function() {};
$(tr, 'th') ? decorateHeader(tr) : function() {};
$(tr, 'td.foot') ? decorateFooter(tr) : function() {};
}
// Insert auto-recommend option for given row
function decorateRow(tr) {
const eleBookLink = $(tr, 'td:nth-child(2)>a');
const strBookID = eleBookLink.href.match(/aid=(\d+)/)[1];
const strBookName = eleBookLink.innerText;
const newTd = $CrE('td');
const input = $CrE('input');
newTd.classList.add('odd');
input.type = 'number';
input.inputmode = 'numeric';
input.style.width = '85%';
input.value = arConfig.books[strBookID] ? String(arConfig.books[strBookID].number) : '0';
input.addEventListener('change', onvaluechange);
input.strBookID = strBookID; input.strBookName = strBookName;
newTd.appendChild(input); tr.appendChild(newTd);
}
// Insert a new row for auto-recommend options
function decorateHeader(tr) {
const allTh = $All(tr, 'th');
const width = ARR_GUI_BOOKCASE_WIDTH;
const newTh = $CrE('th');
newTh.innerText = TEXT_GUI_BOOKCASE_ATRCMMD;
newTh.classList.add(CLASSNAME_TEXT);
tr.appendChild(newTh);
for (let i = 0; i < allTh.length; i++) {
const th = allTh[i];
th.style.width = width[i];
}
}
// Fit the width
function decorateFooter(tr) {
const td = $(tr, 'td.foot');
td.colSpan = ARR_GUI_BOOKCASE_WIDTH.length;
}
// auto-recommend onvaluechange
function onvaluechange(e) {
arConfig = CONFIG.AutoRecommend.getConfig();
const input = e.target;
const value = input.value;
const strBookID = input.strBookID;
const strBookName = input.strBookName;
const bookID = Number(strBookID);
const userDetail = getMyUserDetail() ? getMyUserDetail().userDetail : refreshMyUserDetail();
if (isNumeric(value, true) && Number(value) >= 0) {
// allCount increase
const oriNum = arConfig.books[strBookID] ? arConfig.books[strBookID].number : 0;
const number = Number(value);
arConfig.allCount += number - oriNum;
// save to config
number > 0 ? arConfig.books[strBookID] = {number: number, name: strBookName, id: bookID} : delete arConfig.books[strBookID];
CONFIG.AutoRecommend.saveConfig(arConfig);
// alert
alertify.notify(
TEXT_ALT_ATRCMMDS_SAVED
.replaceAll('{B}', strBookName)
.replaceAll('{N}', value)
.replaceAll('{R}', userDetail.vote-arConfig.allCount)
);
if (userDetail && arConfig.allCount > userDetail.vote) {
const alertBox = alertify.warning(
TEXT_ALT_ATRCMMDS_OVERFLOW
.replace('{V}', String(userDetail.vote))
.replace('{C}', String(arConfig.allCount))
);
alertBox.callback = function(isClicked) {
isClicked && refreshMyUserDetail();
}
};
} else {
// invalid input value, alert
alertify.error(TEXT_ALT_ATRCMMDS_INVALID.replaceAll('{N}', value));
}
}
}
}
function laterReads() {
// Container
const container = $CrE('div');
container.id = 'laterbooks';
content.appendChild(container);
// Title div
const titlebar = $CrE('div');
titlebar.classList.add('gridtop');
titlebar.style.display = 'grid';
titlebar.style['grid-template-columns'] = '1fr 1fr 1fr';
container.appendChild(titlebar);
const title = $CrE('span');
title.innerHTML = '稍后再读';
title.style['grid-column'] = '2/3';
titlebar.appendChild(title);
// Sorter select container
const sortContainer = $CrE('span');
sortContainer.style['grid-column'] = '3/4';
sortContainer.style.textAlign = 'right';
titlebar.appendChild(sortContainer);
// Sorter select
const sltsort = $CrE('select');
sltsort.style.width = 'max-content';
sltsort.addEventListener('change', function() {
const config = CONFIG.bookcasePrefs.getConfig();
config.laterbooks.sortby = sltsort.value;
CONFIG.bookcasePrefs.saveConfig(config);
showBooks();
});
sortContainer.appendChild(sltsort);
// Sorter select options
const sorttypes = Object.keys(FUNC_LATERBOOK_SORTERS);
for (const type of sorttypes) {
const sort = FUNC_LATERBOOK_SORTERS[type];
const option = $CrE('option');
option.innerHTML = sort.name;
option.value = type;
sltsort.appendChild(option);
}
sltsort.selectedIndex = sorttypes.indexOf(CONFIG.bookcasePrefs.getConfig().laterbooks.sortby);
// Body table
const body = $CrE('table');
setAttributes(body, {
'class': 'grid',
'width': '100%',
'align': 'center'
});
const tbody = $CrE('tbody');
body.appendChild(tbody);
container.appendChild(body);
// Header & Rows
showBooks();
function showBooks() {
const config = CONFIG.bookcasePrefs.getConfig().laterbooks;
clearChildnodes(body);
// headers
const headtr = $CrE('tr');
headtr.setAttribute('align', 'center');
const headers = [{
name: '名称',
width: '22%'
},{
name: '简介',
width: '60%'
},{
name: '操作',
width: '18%'
}];
for (const head of headers) {
const th = $CrE('th');
th.innerHTML = head.name;
th.style.width = head.width;
headtr.appendChild(th);
}
body.appendChild(headtr);
// Book rows
const books = sortLaterReads(config.books, config.sortby);
for (const book of books) {
makeRow(book);
}
function makeRow(book) {
const config = CONFIG.bookcasePrefs.getConfig().laterbooks;
// row
const row = $CrE('tr');
// cover & name
const tdName = $CrE('td');
tdName.classList.add('odd');
tdName.style.textAlign = 'center';
const clink = $CrE('a');
clink.href = URL_NOVELINDEX.replace('{I}', book.aid);
clink.target = '_blank';
tdName.appendChild(clink);
const cover = $CrE('img');
cover.src = book.cover;
cover.style.width = '100px';
clink.appendChild(cover);
clink.insertAdjacentHTML('beforeend', '</br>');
clink.insertAdjacentText('beforeend', book.name);
row.appendChild(tdName);
// info
const tdInfo = $CrE('td');
tdInfo.classList.add('even');
tdInfo.insertAdjacentHTML('afterbegin', '<span class="hottext">作品Tags:</span></br>');
for (const tag of book.tags) {
const a = $CrE('a');
a.target = '_blank';
a.href = URL_TAGSEARCH.replace('{TU}', $URL.encode(tag));
a.classList.add(CLASSNAME_BUTTON);
a.innerText = tag + ' ';
tdInfo.appendChild(a);
}
tdInfo.insertAdjacentHTML('beforeend', '</br></br><span class="hottext">内容简介:</span></br>');
tdInfo.insertAdjacentText('beforeend', book.introduce);
row.appendChild(tdInfo);
// operator
const tdOprt = $CrE('td');
tdOprt.classList.add('odd');
tdOprt.style.textAlign = 'center';
const btnDel = makeBtn();
btnDel.innerHTML = '删除';
btnDel.addEventListener('click', del);
tdOprt.appendChild(btnDel);
const btnAbc = makeBtn('a'); // Abc ==> AddBookCase
btnAbc.innerHTML = '加入书架';
btnAbc.href = URL_ADDBOOKCASE.replace('{A}', book.aid);
btnAbc.target = '_blank';
tdOprt.appendChild(btnAbc);
if (config.sortby === 'sort') {
tdOprt.appendChild($CrE('br'));
const btnUp = makeBtn();
btnUp.innerHTML = '上移';
btnUp.addEventListener('click', function () {
const config = CONFIG.bookcasePrefs.getConfig();
const books = Object.values(config.laterbooks.books);
const cur = books.filter((b) => (b.sort === book.sort));
const previous = books.filter((b) => (b.sort === book.sort-1));
if (cur) {
if (previous.length > 0) {
previous[0].sort++;
cur[0].sort--;
CONFIG.bookcasePrefs.saveConfig(config);
showBooks();
}
} else {
alertify.warning(TEXT_ALT_BOOKCASE_AFTERBOOKS_MISSING);
}
});
tdOprt.appendChild(btnUp);
const btnDown = makeBtn();
btnDown.innerHTML = '下移';
btnDown.addEventListener('click', function () {
const config = CONFIG.bookcasePrefs.getConfig();
const books = Object.values(config.laterbooks.books);
const cur = books.filter((b) => (b.sort === book.sort));
const after = books.filter((b) => (b.sort === book.sort+1));
if (cur) {
if (after.length > 0) {
after[0].sort--;
cur[0].sort++;
CONFIG.bookcasePrefs.saveConfig(config);
showBooks();
}
} else {
alertify.warning(TEXT_ALT_BOOKCASE_AFTERBOOKS_MISSING);
}
});
tdOprt.appendChild(btnDown);
}
row.appendChild(tdOprt);
body.appendChild(row);
function del() {
const config = CONFIG.bookcasePrefs.getConfig();
const books = config.laterbooks.books;
const bk = books[book.aid];
if (!bk) {
body.removeChild(row);
return false;
}
delete config.laterbooks.books[book.aid];
Array.prototype.forEach.call(Object.values(books), (b) => (b.sort > bk.sort && b.sort--));
CONFIG.bookcasePrefs.saveConfig(config);
body.removeChild(row);
}
function makeBtn(tagName='span') {
const btn = $CrE(tagName);
btn.classList.add(CLASSNAME_BUTTON);
btn.style.margin = '0 1em';
return btn;
}
}
}
}
// Set attributes to an element
function setAttributes(elm, attributes) {
for (const [name, attr] of Object.entries(attributes)) {
elm.setAttribute(name, attr);
}
}
}
// Novel ads remover
function removeTopAds() {
const ads = []; $All('div>script+script+a').forEach(function(a) {ads.push(a.parentElement);});
for (const ad of ads) {
ad.parentElement.removeChild(ad);
}
}
// Novel index page add-on
function pageNovelIndex() {
removeTopAds();
//downloader();
function downloader() {
AndAPI.getNovelIndex({
aid: unsafeWindow.article_id,
lang: 0,
callback: indexGot
});
function indexGot(xml) {
const volumes = $All(xml, 'volume');
const vtitles = $All('.vcss');
if (volumes.length !== vtitles.length) {return false;}
for (let i = 0; i < volumes.length; i++) {
const volume = volumes[i];
const vtitle = vtitles[i];
const vname = volume.childNodes[0].nodeValue;
// Title element
const elmTitle = $CrE('span');
elmTitle.innerText = vname;
// Spliter element
const elmSpliter = $CrE('span');
elmSpliter.style.margin = '0 0.5em';
// Download button
const elmDlBtn = $CrE('span');
elmDlBtn.classList.add(CLASSNAME_BUTTON);
elmDlBtn.innerHTML = TEXT_GUI_DOWNLOAD_THISVOLUME;
elmDlBtn.addEventListener('click', function() {
// getAttribute returns string rather than number,
// but downloadVolume accepts both string and number as vid
downloadVolume(volume.getAttribute('vid'), vname, ['utf-8', 'big5'][getLang()]);
});
clearChildnodes(vtitle);
vtitle.appendChild(elmTitle);
vtitle.appendChild(elmSpliter);
vtitle.appendChild(elmDlBtn);
}
}
function downloadVolume(vid, vname, charset='utf-8') {
const url = URL_DOWNLOAD1.replace('{A}', unsafeWindow.article_id).replace('{V}', vid).replace('{C}', charset);
downloadFile({
url: url,
name: TEXT_GUI_SDOWNLOAD_FILENAME
.replace('{NovelName}', $('#title').innerText)
.replace('{VolumeName}', vname)
.replace('{Extension}', 'txt')
});
}
}
}
// Novel page add-on
function pageNovel() {
const CSM = new ConfigSetManager();
CSM.install();
const pageResource = {elements: {}, infos: {}, download: {}};
collectPageResources();
// Remove ads
removeTopAds();
// Side-Panel buttons
sideButtons();
// Provide download GUI
downloadGUI();
// Prevent URL.revokeObjectURL in script 轻小说文库下载
revokeObjectURLHOOK();
// Font changer
fontChanger();
// More font-sizes
moreFontSizes();
// Fill content if need
fillContent();
// Beautifier page
beautifier();
function collectPageResources() {
collectElements();
collectInfos();
initDownload();
function collectElements() {
const elements = pageResource.elements;
elements.title = $('#title');
elements.images = $All('.imagecontent');
elements.rightButtonDiv = $('#linkright');
elements.rightNodes = elements.rightButtonDiv.childNodes;
elements.rightBlank = elements.rightNodes[elements.rightNodes.length-1];
elements.content = $('#content');
elements.contentmain = $('#contentmain');
elements.spliterDemo = document.createTextNode(' | ');
}
function collectInfos() {
const elements = pageResource.elements;
const infos = pageResource.infos;
infos.title = elements.title.innerText;
infos.isImagePage = elements.images.length > 0;
infos.content = infos.isImagePage ? null : elements.content.innerText;
}
function initDownload() {
const elements = pageResource.elements;
const download = pageResource.download;
download.running = false;
download.finished = 0;
download.all = elements.images.length;
download.error = 0;
}
}
// Prevent URL.revokeObjectURL in script 轻小说文库下载
function revokeObjectURLHOOK() {
const Ori_revokeObjectURL = URL.revokeObjectURL;
URL.revokeObjectURL = function(arg) {
if (typeof(arg) === 'string' && arg.substr(0, 5) === 'blob:') {return false;};
return Ori_revokeObjectURL(arg);
}
}
// Side-Panel buttons
function sideButtons() {
// Download
SPanel.add({
faicon: 'fa-solid fa-download',
tip: TEXT_GUI_DOWNLOAD_THISCHAPTER,
onclick: dlNovel
});
// Next page
SPanel.add({
faicon: 'fa-solid fa-angle-right',
tip: '下一页',
onclick: (e) => {$('#foottext>a:nth-child(4)').click();}
});
// Previous page
SPanel.add({
faicon: 'fa-solid fa-angle-left',
tip: '上一页',
onclick: (e) => {$('#foottext>a:nth-child(3)').click();}
});
}
// Provide download GUI
function downloadGUI() {
const elements = pageResource.elements;
const infos = pageResource.infos;
// Create donwload button
const dlBtn = elements.downloadBtn = $CrE('span');
dlBtn.classList.add(CLASSNAME_BUTTON);
dlBtn.addEventListener('click', dlNovel);
dlBtn.innerText = TEXT_GUI_DOWNLOAD_THISCHAPTER;
// Create spliter
const spliter = elements.spliterDemo.cloneNode();
// Append to rightButtonDiv
elements.rightButtonDiv.style.width = '550px';
elements.rightButtonDiv.insertBefore(spliter, elements.rightBlank);
elements.rightButtonDiv.insertBefore(dlBtn, elements.rightBlank);
}
// Page beautifier
function beautifier() {
CONFIG.BeautifierCfg.getConfig().novel.beautiful && beautiful();
function beautiful() {
const config = CONFIG.BeautifierCfg.getConfig();
const usedHeight = getRestHeight();
addStyle(CSS_NOVEL
.replaceAll('{BGI}', config.backgroundImage)
.replaceAll('{S}', config.textScale)
.replaceAll('{H}', usedHeight), 'beautifier'
);
unsafeWindow.stopScroll = beautiful_stopScroll;
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 += Number(reg[1]));
});
};
});
usedHeight = usedHeight.toString() + 'px';
return usedHeight;
}
// Mouse dblclick scroll with beautifier applied
function beautiful_scrolling() {
let contentmain = pageResource.elements.contentmain;
let currentpos = contentmain.scrollTop || 0;
contentmain.scrollTo(0, ++currentpos);
let nowpos = contentmain.scrollTop || 0;
pageResource.elements.content.style.userSelect = 'none';
currentpos != nowpos && beautiful_stopScroll();
}
function beautiful_stopScroll() {
pageResource.elements.content.style.userSelect = '';
unsafeWindow.clearInterval(timer);
}
}
}
// Provide font changer
function fontChanger() {
// Button
const bcolor = $('#bcolor');
const txtfont = $CrE('select');
txtfont.id = 'txtfont';
txtfont.addEventListener('change', applyFont);
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 = {
save: () => (setCookies('txtfont', txtfont[txtfont.selectedIndex].value)),
load: () => {
const tmpstr = ReadCookies("txtfont");
if (tmpstr != "") {
for (let i = 0; i < txtfont.length; i++) {
if (txtfont.options[i].value == tmpstr) {
txtfont.selectedIndex = i;
break;
}
}
}
applyFont();
}
};
// Load saved font
CSM.ConfigSets.txtfont.load();
function applyFont() {
$('#content').style['font-family'] = txtfont[txtfont.selectedIndex].value;
}
}
// Provide more font-sizes
function moreFontSizes() {
const select = $('#fonttype');
const savebtn = $('#saveset');
const sizes = [
{
name: '更小',
size: '10px'
},
{
name: '更大',
size: '28px'
},
{
name: '很大',
size: '32px'
},
{
name: '超大',
size: '36px'
},
{
name: '极大',
size: '40px'
},
{
name: '过大',
size: '44px'
},
];
for (const size of sizes) {
const option = $CrE('option');
option.innerHTML = size.name;
option.value = size.size;
// Insert with sorting
for (const opt of select.children) {
const sizeNum1 = getSizeNum(opt.value);
const sizeNum2 = getSizeNum(option.value);
if (isNaN(sizeNum1) || isNaN(sizeNum2)) {continue;} // Code shouldn't be here in normal cases
if (sizeNum1 > sizeNum2) {
select.insertBefore(option, opt);
break;
}
}
option.parentElement !== select && select.appendChild(option);
}
// Load saved fonttype
CSM.ConfigSets.fonttype.load();
function getSizeNum(size) {
return Number(size.match(/(\d+)px/)[1]);
}
}
// Provide content using AndroidAPI
function fillContent() {
// Check whether needs filling
if ($('#contentmain>span')) {
if ($('#contentmain>span').innerText.trim() !== 'null') {
return false;
}
} else {return false;}
// prepare
const content = pageResource.elements.content;
content.innerHTML = TEXT_GUI_NOVEL_FILLING;
const charset = (function() {
const match = document.cookie.match(/(; *)?jieqiUserCharset=(.+?)( *;|$)/);
return match && match[2] && match[2].toLowerCase() === 'big5' ? 1 : 0;
}) ();
// Get content xml
AndAPI.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 = htmlEncode(url.match(/<!--image-->([^<>]+?)<!--image-->/)[1]);
html += 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 += htmlEncode(text);
// Set content
pageResource.elements.content.innerHTML = html;
// Reset pageResource-image if need
pageResource.infos.isImagePage = imgUrls.length > 0;
pageResource.elements.images = $All('.imagecontent');
pageResource.download.all = pageResource.elements.images.length;
}
})
return true;
}
// Download button onclick
function dlNovel() {
pageResource.infos.isImagePage ? dlNovelImages() : dlNovelText();
}
// Download Images
function dlNovelImages() {
const elements = pageResource.elements;
const infos = pageResource.infos;
const download = pageResource.download;
if (download.running) {return false;};
download.running = true; download.finished = 0; download.error = 0;
updateDownloadStatus();
const lenNumber = String(elements.images.length).length;
for (let i = 0; i < elements.images.length; i++) {
const img = elements.images[i];
const name = infos.title + '_' + fillNumber(i+1, lenNumber) + '.jpg';
GM_xmlhttpRequest({
url: img.src,
responseType: 'blob',
onloadstart: function() {
DoLog(LogLevel.Info, '[' + String(i) + ']downloading novel image from ' + img.src);
},
onload: function(e) {
DoLog(LogLevel.Info, '[' + String(i) + ']image got: ' + img.src);
const image = new Image();
image.onload = function() {
const url = toImageFormatURL(image, 1);
DoLog(LogLevel.Info, '[' + String(i) + ']image transformed: ' + img.src);
const a = $CrE('a');
a.href = url;
a.download = name;
a.click();
download.finished++;
updateDownloadStatus();
// Code below seems can work, but actually it doesn't work well and somtimes some file cannot be saved
// The reason is still unknown, but from what I know I can tell that mistakes happend in GM_xmlhttpRequest
// Error stack: GM_xmlhttpRequest.onload ===> image.onload ===> downloadFile ===> GM_xmlhttpRequest =X=> .onload
// This Error will also stuck the GMXHRHook.ongoingList
/*downloadFile({
url: url,
name: name,
onload: function() {
download.finished++;
DoLog(LogLevel.Info, '[' + String(i) + ']file saved: ' + name);
alert('[' + String(i) + ']file saved: ' + name);
updateDownloadStatus();
},
onerror: function() {
alert('downloadfile error! url = ' + String(url) + ', i = ' + String(i));
}
})*/
}
image.onerror = function() {
throw new Error('image load error! image.src = ' + String(image.src) + ', i = ' + String(i));
}
image.src = URL.createObjectURL(e.response);
},
onerror: function(e) {
// Error dealing need...
DoLog(LogLevel.Error, '[' + String(i) + ']image fetch error: ' + img.src);
download.error++;
}
})
}
function updateDownloadStatus() {
elements.downloadBtn.innerText = TEXT_GUI_DOWNLOADING_ALL.replaceAll('C', String(download.finished)).replaceAll('A', String(download.all));
if (download.finished === download.all) {
DoLog(LogLevel.Success, 'All images got.');
elements.downloadBtn.innerText = TEXT_GUI_DOWNLOADED_ALL;
download.running = false;
}
}
}
// Download Text
function dlNovelText() {
const infos = pageResource.infos;
const name = infos.title + '.txt';
const text = infos.content.replaceAll(/[\r\n]+/g, '\r\n');
downloadText(text, name);
}
// Image format changing function
// image: <img> or Image(); format: 1 for jpeg, 2 for png, 3 for webp
function toImageFormatURL(image, format) {
if (typeof(format) === 'number') {format = ['image/jpeg', 'image/png', 'image/webp'][format-1]}
const cvs = $CrE('canvas');
cvs.width = image.width;
cvs.height = image.height;
const ctx = cvs.getContext('2d');
ctx.drawImage(image, 0, 0);
return cvs.toDataURL(format);
}
function ConfigSetManager() {
const CSM = this;
/*const setCookies = unsafeWindow.setCookies,
ReadCookies = unsafeWindow.ReadCookies,
bcolor = unsafeWindow.bcolor,
txtcolor = unsafeWindow.txtcolor,
fonttype = unsafeWindow.fonttype,
scrollspeed = unsafeWindow.scrollspeed,
setSpeed = unsafeWindow.setSpeed,
contentobj = unsafeWindow.contentobj;*/
CSM.ConfigSets = {
'bcolor': {
save: () => (setCookies("bcolor", bcolor.options[bcolor.selectedIndex].value)),
load: () => {
const tmpstr = ReadCookies("bcolor");
bcolor.selectedIndex = 0;
if (tmpstr != "") {
for (let i = 0; i < bcolor.length; i++) {
if (bcolor.options[i].value == tmpstr) {
bcolor.selectedIndex = i;
break;
}
}
}
document.bgColor = bcolor.options[bcolor.selectedIndex].value;
}
},
'txtcolor': {
save: () => (setCookies("txtcolor", txtcolor.options[txtcolor.selectedIndex].value)),
load: () => {
const tmpstr = ReadCookies("txtcolor");
txtcolor.selectedIndex = 0;
if (tmpstr != "") {
for (let i = 0; i < txtcolor.length; i++) {
if (txtcolor.options[i].value == tmpstr) {
txtcolor.selectedIndex = i;
break;
}
}
}
$('#content').style.color = txtcolor.options[txtcolor.selectedIndex].value;
}
},
'fonttype': {
save: () => (setCookies("fonttype", fonttype.options[fonttype.selectedIndex].value)),
load: () => {
const tmpstr = ReadCookies("fonttype");
fonttype.selectedIndex = 2;
if (tmpstr != "") {
for (let i = 0; i < fonttype.length; i++) {
if (fonttype.options[i].value == tmpstr) {
fonttype.selectedIndex = i;
break;
}
}
}
$('#content').style.fontSize = fonttype.options[fonttype.selectedIndex].value;
}
},
'scrollspeed': {
save: () => (setCookies("scrollspeed", scrollspeed.value)),
load: () => {
const tmpstr = ReadCookies("scrollspeed");
if (tmpstr == '') {tmpstr = 5;}
scrollspeed.value = tmpstr;
setSpeed();
}
}
};
CSM.saveSet = function() {
for (const [name, set] of Object.entries(CSM.ConfigSets)) {
set.save();
}
};
CSM.loadSet = function() {
for (const [name, set] of Object.entries(CSM.ConfigSets)) {
set.load();
}
};
CSM.install = function() {
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
});
};
}
}
// Search form add-on
function formSearch() {
const searchForm = $('form[name="articlesearch"]');
if (!searchForm) {return false;};
const typeSelect = $(searchForm, '#searchtype');
const searchText = $(searchForm, '#searchkey');
const searchSbmt = $(searchForm, 'input[class="button"][type="submit"]');
let optionTags;
provideTagOption();
onsubmitHOOK();
function provideTagOption() {
optionTags = $CrE('option');
optionTags.value = VALUE_STR_NULL;
optionTags.innerText = TEXT_GUI_SEARCH_OPTION_TAG;
typeSelect.appendChild(optionTags);
if (tipready) {
// tipshow and tiphide is coded inside wenku8 itself, its function is to show a text tip besides the mouse
typeSelect.addEventListener('mouseover', show);
searchSbmt.addEventListener('mouseover', show);
typeSelect.addEventListener('mouseout' , tiphide);
searchSbmt.addEventListener('mouseout' , tiphide);
} else {
typeSelect.title = TEXT_TIP_SEARCH_OPTION_TAG;
searchSbmt.title = TEXT_TIP_SEARCH_OPTION_TAG;
}
function show() {
optionTags.selected ? tipshow(TEXT_TIP_SEARCH_OPTION_TAG) : function() {};
}
}
function onsubmitHOOK() {
const onsbmt = searchForm.onsubmit;
searchForm.onsubmit = function() {
if (optionTags.selected) {
// DON'T USE window.open()!
// Wenku8 has no window.open used in its own scripts, so do not use it in userscript either.
// It might cause security problems.
//window.open('https://www.wenku8.net/modules/article/tags.php?t=' + $URL.encode(searchText.value));
if (typeof($URL) === 'undefined' ) {
$URLError();
return true;
} else {
GM_openInTab(URL_TAGSEARCH.replace('{TU}', $URL.encode(searchText.value)), {
active: true, insert: true, setParent: true, incognito: false
});
return false;
}
}
}
function $URLError() {
DoLog(LogLevel.Error, '$URL(from gbk.js) is not loaded.');
DoLog(LogLevel.Warning, 'Search as plain text instead.');
// Search as plain text instead
for (const node of typeSelect.childNodes) {
node.selected = (node.tagName === 'OPTION' && node.value === 'articlename') ? true : false;
}
}
}
}
// Tags page add-on
function pageTags() {
}
// Mylink page add-on
function pageMylink() {
// Get elements
const main = $('#content');
const tbllink = $('#content>table');
linkEnhance();
function fixEdit(link) {
const aedit = link.aedit;
aedit.setAttribute('onclick', "editlink({ULID},'{NAME}','{HREF}','{INFO}')".replace('{ULID}', deal(link.ulid)).replace('{NAME}', deal(link.name)).replace('{HREF}', deal(link.href)).replace('{INFO}', deal(link.info)));
function deal(str) {
return str.replaceAll("'", "\\'");
}
}
function linkEnhance() {
const links = getAllLinks();
for (const link of links) {
fixEdit(link);
}
}
function getAllLinks() {
const links = [];
const trs = $All(tbllink, 'tbody>tr+tr');
for (const tr of trs) {
const link = {};
// All <td>
link.tdlink = tr.children[0];
link.tdinfo = tr.children[1];
link.tdtime = tr.children[2];
link.tdoprt = tr.children[3];
// Inside <td>
link.alink = link.tdlink.children[0];
link.aedit = link.tdoprt.children[0];
link.apos = link.tdoprt.children[1];
link.adel = link.tdoprt.children[2];
// Infos
link.href = link.alink.href;
link.ulid = getUrlArgv({url: link.apos.href, name: 'ulid'});
link.name = link.alink.innerText;
link.info = link.tdinfo.innerText;
link.time = link.tdtime.innerText;
link.purl = link.apos.href;
links.push(link);
}
return links;
}
}
// User page add-on
function pageUser() {
const UID = Number(getUrlArgv('uid'));
// Provide review search option
reviewButton();
// Review search option
function reviewButton() {
// clone button and container div
const oriContainer = $All('.blockcontent .userinfo')[0].parentElement;
const container = oriContainer.cloneNode(true);
const button = $(container, 'a');
button.innerText = TEXT_GUI_USER_REVIEWSEARCH;
button.href = URL_REVIEWSEARCH.replaceAll('{K}', String(UID));
oriContainer.parentElement.appendChild(container);
}
}
// Detail page add-on
function pageDetail() {
// Get elements
const content = $('#content');
const tbody = $(content, 'table>tbody');
insertSettings();
// Insert Settings GUI
function insertSettings() {
let elements = GUI();
function GUI() {
const review = CONFIG.BkReviewPrefs.getConfig();
const settings = [
[{html: TEXT_GUI_DETAIL_TITLE_SETTINGS, colSpan: 3, class: 'foot'}],
[{html: TEXT_GUI_DETAIL_TITLE_BGI}, {colSpan: 2, key: 'bgimage', tiptitle: TEXT_TIP_IMAGE_FIT}],
[{html: TEXT_GUI_DETAIL_BGI_UPLOAD}, {colSpan: 2, key: 'bgupload'}],
[{html: TEXT_GUI_DETAIL_GUI_IMAGER}, {colSpan: 2, key: 'imager'}],
[{html: TEXT_GUI_DETAIL_GUI_SCALE}, {colSpan: 2, key: 'scalectnr'}],
[{html: TEXT_GUI_DETAIL_BTF_NOVEL}, {colSpan: 2, key: 'btfnvlctnr'}],
[{html: TEXT_GUI_DETAIL_BTF_REVIEW}, {colSpan: 2, key: 'btfrvwctnr'}],
[{html: TEXT_GUI_DETAIL_BTF_COMMON}, {colSpan: 2, key: 'btfcmnctnr'}],
[{html: TEXT_GUI_DETAIL_FVR_LASTPAGE}, {colSpan: 2, key: 'favoropen'}],
[{html: TEXT_GUI_DETAIL_VERSION_CURVER}, {colSpan: 2, key: 'curversion'}],
[{html: TEXT_GUI_DETAIL_VERSION_CHECKUPDATE}, {colSpan: 2, key: 'updatecheck'}],
[{html: TEXT_GUI_DETAIL_FEEDBACK_TITLE, colSpan: 1, key: 'feedbackttle'}, {html: TEXT_GUI_DETAIL_FEEDBACK, colSpan: 2, key: 'feedback'}],
[{html: TEXT_GUI_DETAIL_UPDATEINFO_TITLE, colSpan: 1, key: 'feedbackttle'}, {html: TEXT_GUI_DETAIL_UPDATEINFO, colSpan: 2, key: 'updateinfo'}],
[{html: TEXT_GUI_DETAIL_CONFIG_EXPORT}, {html: TEXT_GUI_DETAIL_EXPORT_CLICK, colSpan: 2, key: 'exportcfg'}],
[{html: TEXT_GUI_DETAIL_CONFIG_EXPORT_NOPASS}, {html: TEXT_GUI_DETAIL_EXPORT_CLICK, colSpan: 2, key: 'exportcfgnp'}],
[{html: TEXT_GUI_DETAIL_CONFIG_IMPORT, colSpan: 1, key: 'importcfgttle'}, {html: TEXT_GUI_DETAIL_IMPORT_CLICK, colSpan: 2, key: 'importcfg'}],
[{html: TEXT_GUI_DETAIL_CONFIG_MANAGE, colSpan: 1, key: 'managecfgttle'}, {html: TEXT_GUI_DETAIL_MANAGE_CLICK, colSpan: 2, key: 'managecfg'}],
//[{html: TEXT_GUI_DETAIL_XXXXXX_XXXXXX, colSpan: 1, key: 'xxxxxxxx'}, {html: TEXT_GUI_DETAIL_XXXXXX_XXXXXX, colSpan: 2, key: 'xxxxxxxx'}],
]
const elements = createTableGUI(settings);
const tdBgi = elements.bgimage;
const imageinput = elements.imageinput = $CrE('input');
const bgioprt = elements.bgioprt = $CrE('span');
const bgiupld = elements.bgupload;
const ckbgiup = elements.ckbgiup = $CrE('input');
ckbgiup.type = 'checkbox';
ckbgiup.checked = CONFIG.BeautifierCfg.getConfig().upload;
ckbgiup.addEventListener('change', uploadChange);
settip(ckbgiup, TEXT_GUI_DETAIL_BGI_LEGAL);
bgiupld.appendChild(ckbgiup);
imageinput.type = 'file';
imageinput.style.display = 'none';
imageinput.addEventListener('change', pictureGot);
bgioprt.innerHTML = TEXT_GUI_DETAIL_DEFAULT_BGI + '</br>' + TEXT_GUI_DETAIL_BGI.replace('{N}', CONFIG.BeautifierCfg.getConfig().bgiName);
bgioprt.style.color = 'grey';
settip(bgioprt, TEXT_TIP_IMAGE_FIT);
tdBgi.addEventListener("dragenter", destroyEvent);
tdBgi.addEventListener("dragover", destroyEvent);
tdBgi.addEventListener('drop', pictureGot);
tdBgi.style.textAlign = 'center';
tdBgi.addEventListener('click', ()=>{elements.imageinput.click();});
tdBgi.appendChild(imageinput);
tdBgi.appendChild(bgioprt);
// Imager
const curimager = CONFIG.UserGlobalCfg.getConfig().imager;
elements.imager.style.padding = '0px 0.5em';
for (const [key, imager] of Object.entries(DATA_IMAGERS)) {
if (typeof(imager) !== 'object' || !imager.isImager) {continue;}
const span = $CrE('span');
const radio = $CrE('input');
const text = $CrE('span');
radio.type = 'radio';
radio.value = '';
radio.id = 'imager_'+key;
radio.imagerkey = key;
radio.name = 'imagerselect';
radio.style.cursor = 'pointer';
radio.addEventListener('change', imagerChange);
radio.disabled = !imager.available;
text.innerText = imager.name + (imager.available ? '' : '(已失效)');
text.style.marginRight = '1em';
text.style.cursor = 'pointer';
text.addEventListener('click', function() {radio.click();});
span.style.display = 'inline-block';
span.appendChild(radio);
span.appendChild(text);
if (imager.tip) {
let tip = imager.tip;
DATA_IMAGERS.default === key && (tip += TEXT_TIP_IMAGER_DEFAULT);
!imager.available && (tip = '<del>{T}</del></br>已失效'.replace('{T}', tip));
settip(radio, tip);
settip(text, tip);
//settip(span, imager.tip);
}
elements.imager.appendChild(span);
}
$(elements.imager, '#imager_'+curimager).checked = true;
// Text scale
const textScale = CONFIG.BeautifierCfg.getConfig().textScale;
const scalectnr = elements.scalectnr;
const elmscale = elements.scale = $CrE('input');
elmscale.type = 'number';
elmscale.id = 'textScale';
elmscale.value = textScale;
elmscale.addEventListener('change', scaleChange);
elmscale.addEventListener('keydown', (e) => {e.keyCode === 13 && scaleChange();});
scalectnr.appendChild(elmscale);
scalectnr.appendChild(document.createTextNode('%'));
// Beautifier
const btfnvlctnr = elements.btfnvlctnr;
const btfrvwctnr = elements.btfrvwctnr;
const btfcmnctnr = elements.btfcmnctnr;
const ckbtfnvl = elements.ckbtfnvl = $CrE('input');
const ckbtfrvw = elements.ckbtfrvw = $CrE('input');
const ckbtfcmn = elements.ckbtfcmn = $CrE('input');
ckbtfnvl.type = ckbtfrvw.type = ckbtfcmn.type = 'checkbox';
ckbtfnvl.page = 'novel';
ckbtfrvw.page = 'reviewshow';
ckbtfcmn.page = 'common';
ckbtfnvl.checked = CONFIG.BeautifierCfg.getConfig().novel.beautiful;
ckbtfrvw.checked = CONFIG.BeautifierCfg.getConfig().reviewshow.beautiful;
ckbtfcmn.checked = CONFIG.BeautifierCfg.getConfig().common.beautiful;
ckbtfnvl.addEventListener('change', beautifulChange);
ckbtfrvw.addEventListener('change', beautifulChange);
ckbtfcmn.addEventListener('change', beautifulChange);
btfnvlctnr.appendChild(ckbtfnvl);
btfrvwctnr.appendChild(ckbtfrvw);
btfcmnctnr.appendChild(ckbtfcmn);
// Favorite open
const favoropen = elements.favoropen;
const favorlast = elements.favorlast = $CrE('input');
favorlast.type = 'checkbox';
favorlast.checked = CONFIG.BkReviewPrefs.getConfig().favorlast;
favorlast.addEventListener('change', favorlastChange);
favoropen.appendChild(favorlast);
// Version control
const curversion = elements.curversion;
const updatecheck = elements.updatecheck;
const versiondisplay = $CrE('span');
versiondisplay.innerText = 'v' + GM_info.script.version;
updatecheck.innerText = TEXT_GUI_DETAIL_VERSION_CHECK;
updatecheck.style.color = 'grey';
updatecheck.style.textAlign = 'center';
updatecheck.addEventListener('click', updateOnclick);
curversion.appendChild(versiondisplay);
// Feedback
const feedback = elements.feedback;
feedback.style.color = 'grey';
feedback.style.textAlign = 'center';
feedback.addEventListener('click', function() {
window.open('https://greasyfork.org/scripts/416310/feedback');
});
// Update info
const updateinfo = elements.updateinfo;
updateinfo.style.color = 'grey';
updateinfo.style.textAlign = 'center';
updateinfo.addEventListener('click', function() {
window.open('https://greasyfork.org/scripts/416310#updateinfo');
})
// Config export/import
const exportcfg = elements.exportcfg;
const exportcfgnp = elements.exportcfgnp;
const importcfg = elements.importcfg;
const configinput = elements.configinput = $CrE('input');
configinput.type = 'file';
configinput.style.display = 'none';
importcfg.style.color = exportcfgnp.style.color = exportcfg.style.color = 'grey';
importcfg.style.textAlign = exportcfgnp.style.textAlign = exportcfg.style.textAlign = 'center';
exportcfg.addEventListener('click', ()=>{exportConfig(false);});
exportcfgnp.addEventListener('click', ()=>{exportConfig(true);});
importcfg.addEventListener('click', () => {configinput.click()});
configinput.addEventListener('change', configfileGot);
importcfg.addEventListener("dragenter", destroyEvent);
importcfg.addEventListener("dragover", destroyEvent);
importcfg.addEventListener('drop', configfileGot);
//importcfg.appendChild(configinput);
// Config management
const managecfg = elements.managecfg;
managecfg.style.color = 'grey';
managecfg.style.textAlign = 'center';
managecfg.addEventListener('click', openManagePanel);
// Paste event
window.addEventListener('paste', filePasted);
return elements;
}
function filePasted(e) {
const input = e.dataTransfer || e.clipboardData || window.clipboardData || e.target;
if (!input.files || input.files.length === 0) {return false;};
for (const file of input.files) {
switch (file.type) {
case 'image/bmp':
case 'image/gif':
case 'image/vnd.microsoft.icon':
case 'image/jpeg':
case 'image/png':
case 'image/svg+xml':
case 'image/tiff':
case 'image/webp':
confirm(TEXT_GUI_DETAIL_CONFIG_IMPORT_CONFIRM_SELECT.replace('{N}', file.name)) && pictureGot(e);
break;
case '': {
const splited = file.name.split('.');
const ext = splited[splited.length-1];
switch (ext) {
case 'wkp':
confirm(TEXT_GUI_DETAIL_CONFIG_IMPORT_CONFIRM_PASTE.replace('{N}', file.name)) && configfileGot(e);
}
}
}
}
}
function pictureGot(e) {
e.preventDefault();
// Get file
const input = e.dataTransfer || e.clipboardData || window.clipboardData || e.target;
if (!input.files || input.files.length === 0) {return false;};
const fileObj = input.files[0];
const mimetype = fileObj.type;
const name = fileObj.name;
// Create a new file input
elements.bgimage.removeChild(elements.imageinput);
const imageinput = elements.imageinput = $CrE('input');
imageinput.type = 'file';
imageinput.style.display = 'none';
imageinput.addEventListener('change', pictureGot);
elements.bgimage.appendChild(imageinput);
if (!mimetype || mimetype.split('/')[0] !== 'image') {
alertify.error(TEXT_ALT_IMAGE_FORMATERROR);
return false;
}
elements.bgioprt.innerHTML = TEXT_GUI_DETAIL_BGI_WORKING;
// Get object url
const objurl = URL.createObjectURL(fileObj);
// Get image url(format base64)
getImageUrl(objurl, true, true, (url) => {
if (!url) {return false;};
// Save to config
const config = CONFIG.BeautifierCfg.getConfig();
config.backgroundImage = url;
config.bgiName = name;
CONFIG.BeautifierCfg.saveConfig(config);
elements.bgioprt.innerHTML = name;
URL.revokeObjectURL(objurl);
// Upload if need
if (config.upload) {
alertify.notify(TEXT_ALT_IMAGE_UPLOAD_WORKING);
elements.bgioprt.innerHTML = TEXT_GUI_DETAIL_BGI_UPLOADING.replace('{NAME}', name);
const file = dataURLtoFile(url, name);
uploadImage({
file: file,
onerror: (e) => {
alertify.error(TEXT_ALT_IMAGE_UPLOAD_ERROR);
DoLog(LogLevel.Error, ['Upload error at pictureGot:', e]);
elements.bgioprt.innerHTML = TEXT_GUI_DETAIL_BGI_UPLOADFAILED.replace('{NAME}', name);
const config = CONFIG.BeautifierCfg.getConfig();
config.upload = elements.ckbgiup.checked = /^https?:\/\//.test(config.reveiwshow.backgroundImage);
CONFIG.BeautifierCfg.saveConfig(config);
},
onload: (json) => {
const config = CONFIG.BeautifierCfg.getConfig();
config.backgroundImage = json.url;
CONFIG.BeautifierCfg.saveConfig(config);
elements.bgioprt.innerHTML = TEXT_GUI_DETAIL_DEFAULT_BGI + '</br>' + TEXT_GUI_DETAIL_BGI.replace('{N}', CONFIG.BeautifierCfg.getConfig().bgiName);
alertify.success(TEXT_ALT_IMAGE_UPLOAD_SUCCESS.replace('{NAME}', json.name).replace('{URL}', json.url));
}
})
}
});
}
function uploadChange(e) {
e.preventDefault();
const config = CONFIG.BeautifierCfg.getConfig();
config.upload = !config.upload;
CONFIG.BeautifierCfg.saveConfig(config);
const name = config.bgiName ? config.bgiName : 'image.jpeg';
if (config.upload) {
// Upload
const url = config.backgroundImage;
if (!/^https?:\/\//.test(url)) {
alertify.notify(TEXT_ALT_IMAGE_UPLOAD_WORKING);
elements.bgioprt.innerHTML = TEXT_GUI_DETAIL_BGI_UPLOADING.replace('{NAME}', name);
const file = dataURLtoFile(url, name);
uploadImage({
file: file,
onerror: (e) => {
alertify.error(TEXT_ALT_IMAGE_UPLOAD_ERROR);
DoLog(LogLevel.Error, ['Upload error at uploadChange:', e]);
elements.bgioprt.innerHTML = TEXT_GUI_DETAIL_BGI_UPLOADFAILED.replace('{NAME}', name);
const config = CONFIG.BeautifierCfg.getConfig();
config.upload = elements.ckbgiup.checked = /^https?:\/\//.test(config.backgroundImage);
CONFIG.BeautifierCfg.saveConfig(config);
},
onload: (json) => {
const config = CONFIG.BeautifierCfg.getConfig();
config.backgroundImage = json.url;
config.bgiName = elements.bgioprt.innerHTML = json.name;
CONFIG.BeautifierCfg.saveConfig(config);
alertify.success(TEXT_ALT_IMAGE_UPLOAD_SUCCESS.replace('{NAME}', json.name).replace('{URL}', json.url));
}
});
}
} else {
// Download
const url = config.backgroundImage;
if (/^https?:\/\//.test(url)) {
alertify.notify(TEXT_ALT_IMAGE_DOWNLOAD_WORKING);
elements.bgioprt.innerHTML = TEXT_GUI_DETAIL_BGI_DOWNLOADING.replace('{NAME}', name);
getImageUrl(url, true, true, (dataurl) => {
if (!dataurl) {
const config = CONFIG.BeautifierCfg.getConfig();
config.upload = elements.ckbgiup.checked = /^https?:\/\//.test(config.backgroundImage);
CONFIG.BeautifierCfg.saveConfig(config);
return false;
};
// Save to config
const config = CONFIG.BeautifierCfg.getConfig();
config.backgroundImage = dataurl;
CONFIG.BeautifierCfg.saveConfig(config);
alertify.success(TEXT_ALT_IMAGE_DOWNLOAD_SUCCESS.replace('{NAME}', name));
elements.bgioprt.innerHTML = name;
});
}
}
setTimeout(()=>{elements.ckbgiup.checked = config.upload;}, 0);
}
function imagerChange(e) {
e.stopPropagation();
const radio = e.target;
if (radio.checked) {
const imager = DATA_IMAGERS[radio.imagerkey];
const config = CONFIG.UserGlobalCfg.getConfig();
config.imager = radio.imagerkey;
CONFIG.UserGlobalCfg.saveConfig(config);
alertify.message('图床已切换到{NAME}'.replace('{NAME}', imager.name));
imager.warning && alertify.warning(imager.warning);
}
}
function scaleChange(e) {
e.stopPropagation();
const config = CONFIG.BeautifierCfg.getConfig();
config.textScale = e.target.value;
CONFIG.BeautifierCfg.saveConfig(config);
alertify.message(TEXT_ALT_TEXTSCALE_CHANGED.replaceAll('{S}', config.textScale));
}
function beautifulChange(e) {
e.stopPropagation();
const checkbox = e.target;
const config = CONFIG.BeautifierCfg.getConfig();
config[checkbox.page].beautiful = checkbox.checked;
CONFIG.BeautifierCfg.saveConfig(config);
alertify.message(checkbox.checked ? TEXT_ALT_BEAUTIFUL_ON : TEXT_ALT_BEAUTIFUL_OFF);
}
function favorlastChange(e) {
e.stopPropagation();
const checkbox = e.target;
const config = CONFIG.BkReviewPrefs.getConfig();
config.favorlast = checkbox.checked;
CONFIG.BkReviewPrefs.saveConfig(config);
alertify.message(checkbox.checked ? TEXT_ALT_FAVORITE_LAST_ON : TEXT_ALT_FAVORITE_LAST_OFF);
}
function updateOnclick(e) {
TASK.Script.update(true);
}
function configfileGot(e) {
e.preventDefault();
// Get file
const input = e.dataTransfer || e.clipboardData || window.clipboardData || e.target;
if (!input.files || input.files.length === 0) {return false;};
const fileObj = input.files[0];
const splitedname = fileObj.name.split('.');
const ext = splitedname[splitedname.length-1].toLowerCase();
if (ext !== 'wkp') {
alertify.error(TEXT_ALT_DETAIL_CONFIG_IMPORT_ERROR_SELECT);
DoLog(LogLevel.Warning, 'pageDetail.insertSettings.GUI.configfileGot: userinput error.')
return false;
}
// Read config from file
try {
const FR = new FileReader();
FR.onload = fileOnload;
FR.readAsText(fileObj);
} catch(e) {
fileError(e);
}
function fileOnload(e) {
try {
// Get json
const json = JSON.parse(e.target.result);
// Import
importConfig(json);
alertify.success(TEXT_ALT_DETAIL_IMPORTED);
} catch(err) {
fileError(err);
}
}
function fileError(e) {
DoLog(LogLevel.Error, ['pageDetail.insertSettings.GUI.configfileGot:', e]);
alertify.error(TEXT_ALT_DETAIL_CONFIG_IMPORT_ERROR_READ);
}
}
function openManagePanel(e) {
const settings = {
id: 'ConfigPanel'
};
const SetPanel = new SettingPanel(settings);
const tblAccount = SetPanel.tables[0];
account();
drafts();
review_favorites();
pending_tip();
SetPanel.usercss += '.settingpanel-block.sp-center {text-align: center;} .settingpanel-block{overflow-wrap: anywhere;}';
function account() {
const userConfig = CONFIG.GlobalConfig.getConfig();
const users = userConfig.users ? userConfig.users : {};
// Create table
const table = new SetPanel.PanelTable({
rows: [{
blocks: [{
className: 'sp-center',
innerHTML: '账号管理',
colSpan: 3
}]
},{
blocks: [{
className: 'sp-center',
innerHTML: '用户名'
},{
className: 'sp-center',
innerHTML: '密码'
},{
className: 'sp-center',
innerHTML: '操作'
}]
}]
});
SetPanel.appendTable(table);
for (const [name, user] of Object.entries(users)) {
// Get account
const username = user.username;
const password = user.password;
// Row
const row = new SetPanel.PanelRow();
table.appendRow(row);
// Block username
const block_username = new SetPanel.PanelBlock({
className: 'sp-center',
innerHTML: username
});
// Block password
const spanpswd = $CrE('span');;
spanpswd.innerHTML = '*'.repeat(password.length);
const block_password = new SetPanel.PanelBlock({
className: 'sp-center',
children: [spanpswd]
});
// Block operator
const btndel = _createBtn('删除', make_del_callback(row, username));
const elmshow = $CrE('span'); elmshow.innerHTML = '查看';
const btnshow = _createBtn(elmshow, make_show_callback(elmshow, spanpswd, password));
const block_operator = new SetPanel.PanelBlock({
className: 'sp-center',
children: [btnshow, btndel]
});
// Append row to SettingPanel
row.appendBlock(block_username).appendBlock(block_password).appendBlock(block_operator);
}
function make_del_callback(row, username) {
return function(e) {
const userConfig = CONFIG.GlobalConfig.getConfig();
delete userConfig.users[username];
CONFIG.GlobalConfig.saveConfig(userConfig);
row.remove();
}
}
function make_show_callback(btn, span, password) {
let show = false;
let timeout;
return function toggle(e) {
show = !show;
span.innerHTML = show ? password : '*'.repeat(password.length);
btn.innerHTML = show ? '隐藏' : '查看';
}
}
}
function drafts() {
// Get config
const allCData = CONFIG.commentDrafts.getConfig();
// Create table
const table = new SetPanel.PanelTable({
rows: [{
blocks: [{
className: 'sp-center',
innerHTML: '书评草稿管理',
colSpan: 3
}]
},{
blocks: [{
className: 'sp-center',
innerHTML: '标题'
},{
className: 'sp-center',
innerHTML: '内容'
},{
className: 'sp-center',
innerHTML: '操作'
}]
}]
});
SetPanel.appendTable(table);
// Append rows
for (const [propkey, commentData] of Object.entries(allCData)) {
if (propkey === KEY_DRAFT_VERSION) {continue;}
const title = commentData.title;
const content = commentData.content;
const key = commentData.key;
// Row
const row = new SetPanel.PanelRow();
table.appendRow(row);
// Block title
const span_title = $CrE('span');
span_title.innerHTML = _decorate(title);
const block_title = new SetPanel.PanelBlock({className: 'draft-title sp-center', children: [span_title]});
// Block content
const span_content = $CrE('span');
span_content.innerHTML = _decorate(content);
const block_content = new SetPanel.PanelBlock({className: 'draft-content', children: [span_content]});
// Block operator
const elmshow = $CrE('span'); elmshow.innerHTML = '展开';
const btnshow = _createBtn(elmshow, make_show_callback(elmshow, key, row, span_title, span_content));
//const btnedit = _createBtn('编辑', make_edit_callback(key, row));
const btnopen = _createBtn('打开', make_open_callback(key));
const btndel = _createBtn('删除', make_del_callback(key, row));
const block_operator = new SetPanel.PanelBlock({className: 'draft-operator sp-center', children: [btnshow, btnopen, btndel]});
// Append to row
row.appendBlock(block_title).appendBlock(block_content).appendBlock(block_operator);
}
// Append css
SetPanel.usercss += '.settingpanel-block.draft-title {width: 20%;} .settingpanel-block.draft-content {width: 50%;} .settingpanel-block.draft-operator {width: 30%}';
function make_show_callback(btn, key, row, span_title, span_content) {
let show = false;
return function() {
const allCData = CONFIG.commentDrafts.getConfig();
const data = allCData[key];
if (!data) {
alertify.warning(TEXT_ALT_DETAIL_MANAGE_NOTFOUND);
row.remove();
return false;
}
show = !show;
btn.innerHTML = show ? '收起' : '展开';
span_title.innerHTML = _decorate(show ? {text: data.title, length: -1} : data.title);
span_content.innerHTML = _decorate(show ? {text: data.content, length: -1} : data.content);
};
}
function make_edit_callback(key, row) {
return function() {
// Get data
const allCData = CONFIG.commentDrafts.getConfig();
const data = allCData[key];
if (!data) {
alertify.warning(TEXT_ALT_DETAIL_MANAGE_NOTFOUND);
row.remove();
return false;
}
// Create box gui
const box = alertify.alert();
const container = box.elements.content;
makeEditor(container, data.rid.toString());
const form = $(container, 'form');
const ptitle = $(container, '#ptitle');
const pcontent = $(container, '#pcontent');
ptitle.value = data.title;
pcontent.value = data.content;
box.setting({
maximizable: false,
resizable: true
});
box.resizeTo('80%', '60%');
box.show();
};
}
function make_open_callback(key, row) {
return function() {
const allCData = CONFIG.commentDrafts.getConfig();
const data = allCData[key];
if (!data) {
alertify.warning(TEXT_ALT_DETAIL_MANAGE_NOTFOUND);
row.remove();
return false;
}
const url = data.rid ? URL_REVIEWSHOW_1.replace('{R}', data.rid.toString()) : URL_NOVELINDEX.replace('{I}', data.bid.toString());
window.open(url);
}
}
function make_del_callback(key, row) {
return function() {
const allCData = CONFIG.commentDrafts.getConfig();
delete allCData[key];
CONFIG.commentDrafts.saveConfig(allCData);
row.remove();
};
}
}
function review_favorites() {
// Get config
const config = CONFIG.BkReviewPrefs.getConfig();
const favs = config.favorites;
// Create table
const table = new SetPanel.PanelTable({
rows: [{
blocks: [{
className: 'sp-center',
innerHTML: '书评收藏管理',
colSpan: 3
}]
},{
blocks: [{
className: 'sp-center',
innerHTML: '主题'
},{
className: 'sp-center',
innerHTML: '备注'
},{
className: 'sp-center',
innerHTML: '操作'
}]
}]
});
SetPanel.appendTable(table);
// Append rows
for (const [rid, fav] of Object.entries(favs)) {
// Row
const row = new SetPanel.PanelRow();
table.appendRow(row);
// Title block
const span_title = $CrE('span');
span_title.innerHTML = _decorate({text: fav.name, length: 0});
const block_title = new SetPanel.PanelBlock({className: 'fav-title sp-center', children: [span_title]});
// Note block
const span_note = $CrE('span');
span_note.innerHTML = _decorate({text: fav.tiptitle, length: 0});
const block_note = new SetPanel.PanelBlock({className: 'fav-note sp-center', children: [span_note]});
// Operator block
const btn_open = _makeBtn({
tagName: 'a',
innerHTML: TEXT_GUI_DETAIL_MANAGE_FAV_NOTE_BTN_OPEN,
props: {
href: fav.href + (config.favorlast ? '&page=last' : ''),
target: '_blank'
}
});
const btn_edit = _makeBtn({
innerHTML: TEXT_GUI_DETAIL_MANAGE_FAV_NOTE_BTN_NOTE,
onclick: edit.bind(null, fav, row)
});
const btn_delete = _makeBtn({
innerHTML: TEXT_GUI_DETAIL_MANAGE_FAV_NOTE_BTN_DELETE,
onclick: del.bind(null, rid, row)
});
const block_oprt = new SetPanel.PanelBlock({className: 'fav-operator sp-center', children: [btn_open, btn_edit, btn_delete]});
// Append to row
row.appendBlock(block_title).appendBlock(block_note).appendBlock(block_oprt);
}
// Append css
SetPanel.usercss += '.settingpanel-block.fav-title {width: 35%;} .settingpanel-block.fav-note {width: 35%;} .settingpanel-block.fav-operator {width: 30%}';
function edit(fav, row) {
alertify.prompt(TEXT_GUI_DETAIL_MANAGE_FAV_NOTE_TITLE, TEXT_GUI_DETAIL_MANAGE_FAV_NOTE_TIP.replace('{TITLE}', fav.name), fav.tiptitle || '', onok, function() {});
function onok(e, value) {
// Save empty value as null
value === value || null;
fav.tiptitle = value;
CONFIG.BkReviewPrefs.saveConfig(config);
row.blocks[1].element.firstChild.innerHTML = _decorate({text: value, length: 0});
alertify.success(TEXT_GUI_DETAIL_MANAGE_FAV_SAVED);
}
}
function del(rid, row) {
alertify.confirm(TEXT_GUI_DETAIL_MANAGE_FAV_DELETE_TITLE, TEXT_GUI_DETAIL_MANAGE_FAV_DELETE_TIP.replace('{TITLE}', favs[rid].name), onok, function() {});
function onok() {
delete favs[rid];
CONFIG.BkReviewPrefs.saveConfig(config);
row.remove();
alertify.success(TEXT_GUI_DETAIL_MANAGE_FAV_DELETED);
}
}
}
function pending_tip() {
const span = $CrE('span');
span.innerHTML = '*其他管理项尚在开发中,请耐心等待';
span.classList.add(CLASSNAME_TEXT);
SetPanel.element.appendChild(span);
}
function _createBtn(htmlorbtn, onclick) {
const innerHTML = typeof htmlorbtn === 'string' ? htmlorbtn : htmlorbtn.innerHTML;
const btn = htmlorbtn instanceof HTMLElement ? htmlorbtn : $CrE('span');
!btn.classList.contains(CLASSNAME_BUTTON) && btn.classList.add(CLASSNAME_BUTTON);
btn.innerHTML = innerHTML;
btn.style.margin = '0px 0.5em';
onclick && btn.addEventListener('click', onclick);
return btn;
}
function _makeBtn(details) {
// Create element
const elm = $CrE(details.tagName || 'span');
// Write innerHTML
copyProp(details, elm, 'innerHTML');
// Write other properties
details.props && copyProps(details.props, elm, Object.keys(details.props));
// Make onclick
const onclick = details.onclick || (details.onclickMaker ? details.onclickMaker.apply(null, details.onclickArgs || []) : null);
// Create button
const btn = _createBtn(elm, onclick);
// Custom classes
details.classes && details.classes.forEach(function(c) {!btn.classList.contains(c) && btn.classList.add(c);});
return btn;
}
// details: 'string' or {text: '', length: 16}
function _decorate(details) {
// Get Args
details = !details ? '' : details;
details = typeof details === 'string' ? {text: details} : details;
const text = details.text || '';
const length = typeof details.length === 'number' ? (details.length > 0 ? details.length : Infinity) : 16;
const len = length > 0 ? length : 9999999999999;
const overflow = (text.length - len) > length;
const cut = overflow ? text.substr(0, len) : text;
const encoded = htmlEncode(cut).replaceAll('\n', '</br>');
const filled = text.length === 0 ? TEXT_GUI_DETAIL_CONFIG_MANAGE_EMPTY : (overflow ? encoded + TEXT_GUI_DETAIL_CONFIG_MANAGE_MORE : encoded);
return filled;
}
}
}
function createTableGUI(lines) {
const elements = {};
for (const line of lines) {
const tr = $CrE('tr');
for (const item of line) {
const td = $CrE('td');
item.html && (td.innerHTML = item.html);
item.colSpan && (td.colSpan = item.colSpan);
item.class && (td.className = item.class);
item.id && (td.id = item.id);
item.tiptitle && settip(td, item.tiptitle);
item.key && (elements[item.key] = td);
td.style.padding = '3px';
tr.appendChild(td);
}
tbody.appendChild(tr);
}
return elements;
function ElementObject(element) {
const p = new Proxy(element, {
get: function(elm, id, receiver) {
return elm[id] || $(elm, '#'+id);
}
});
return p;
}
}
}
// Index page add-on
function pageIndex() {
insertStatus();
showFavorites();
showLaterReads();
// Insert usersript inserted tip
function insertStatus() {
const blockcontent = $('#centers>.block:nth-child(1)>.blockcontent');
blockcontent.appendChild($CrE('br'));
const textNode = $CrE('span');
textNode.innerText = TEXT_GUI_INDEX_STATUS;
textNode.classList.add(CLASSNAME_TEXT);
blockcontent.appendChild(textNode);
}
// Show favorite reviews
function showFavorites() {
const links = [];
const config = CONFIG.BkReviewPrefs.getConfig();
for (const [rid, favorite] of Object.entries(config.favorites)) {
const href = favorite.href + (config.favorlast ? '&page=last' : '');
const tiptitle = favorite.tiptitle ? favorite.tiptitle : href;
const innerHTML = favorite.name.substr(0, 12) // prevent overflow
links.push({
innerHTML: innerHTML,
tiptitle: tiptitle,
href: href
});
}
const block = createWenkuBlock({
type: 'toplist',
parent: '#left',
title: TEXT_GUI_INDEX_FAVORITES,
items: links
});
}
// Show top-6 read-later books
function showLaterReads() {
const config = CONFIG.bookcasePrefs.getConfig().laterbooks;
const books = sortLaterReads(config.books, config.sortby).filter((e,i,a)=>(i<6));
const items = books.map(function(book, i) {
return {
href: URL_NOVELINDEX.replace('{I}', book.aid),
src: book.cover,
tiptitle: book.name,
text: book.name
}
});
const block = createWenkuBlock({
type: 'imagelist',
parent: '#centers',
title: TEXT_GUI_INDEX_LATERBOOKS,
items: items
});
settip($(block, '.blocktitle'), TEXT_TIP_INDEX_LATERREADS);
}
}
// Download page add-on
function pageDownload() {
let i;
let dlCount = 0; // number of active download tasks
let dlAllRunning = false; // whether there is downloadAll running
// Get novel info
const novelInfo = {}; collectNovelInfo();
const myDlBtns = [];
// Donwload GUI
downloadGUI();
// Server GUI
serverGUI();
/* ******************* Code ******************* */
function collectNovelInfo() {
novelInfo.novelName = $('html body div.main div#centerm div#content table.grid caption a').innerText;
novelInfo.displays = getAllNameEles();
novelInfo.volumeNames = getAllNames();
novelInfo.type = getUrlArgv('type');
novelInfo.ext = novelInfo.type !== 'txtfull' ? novelInfo.type : 'txt';
}
// Donwload GUI
function downloadGUI() {
switch (novelInfo.type) {
case 'txt':
downloadGUI_txt();
break;
case 'txtfull':
downloadGUI_txtfull();
break;
case 'umd':
downloadGUI_umd();
break;
case 'jar':
downloadGUI_jar();
break;
default:
DoLog(LogLevel.Warning, 'pageDownload.downloadGUI: Unknown download type');
}
}
// Donwload GUI for txt
function downloadGUI_txt() {
// Only txt is really separated by volumes
if (novelInfo.type !== 'txt') {return false;};
// define vars
let i;
const tbody = $('table>tbody');
const header = $(tbody, 'th').parentElement;
const thead = $(header, 'th');
// Append new th
const newHead = thead.cloneNode(true);
newHead.innerText = TEXT_GUI_SDOWNLOAD;
thead.width = '40%';
header.appendChild(newHead);
// Append new td
const trs = $All(tbody, 'tr');
for (i = 1; i < trs.length; i++) { /* i = 1 to trs.length-1: skip header */
const index = i-1;
const tr = trs[i];
const newTd = $(tr, 'td.even').cloneNode(true);
const links = $All(newTd, 'a');
for (const a of links) {
a.classList.add(CLASSNAME_BUTTON);
a.info = {
description: 'volume download button',
name: novelInfo.volumeNames[index],
filename: TEXT_GUI_SDOWNLOAD_FILENAME
.replace('{NovelName}', novelInfo.novelName)
.replace('{VolumeName}', novelInfo.volumeNames[index])
.replace('{Extension}', novelInfo.ext),
index: index,
display: novelInfo.displays[index]
}
a.onclick = downloadOnclick;
myDlBtns.push(a);
}
tr.appendChild(newTd);
}
// Append new tr, provide batch download
const newTr = trs[trs.length-1].cloneNode(true);
const newTds = $All(newTr, 'td');
newTds[0].innerText = TEXT_GUI_DOWNLOADALL;
//clearChildnodes(newTds[1]); clearChildnodes(newTds[2]);
newTds[1].innerHTML = newTds[2].innerHTML = TEXT_GUI_NOTHINGHERE;
tbody.insertBefore(newTr, tbody.children[1]);
const allBtns = $All(newTds[3], 'a');
for (i = 0; i < allBtns.length; i++) {
const a = allBtns[i];
a.href = 'javascript:void(0);';
a.info = {
description: 'download all button',
index: i
}
a.onclick = downloadAllOnclick;
}
// Download button onclick
function downloadOnclick() {
const a = this;
a.info.display.innerText = a.info.name + TEXT_GUI_WAITING;
downloadFile({
url: a.href,
name: a.info.filename,
onloadstart: function(e) {
a.info.display.innerText = a.info.name + TEXT_GUI_DOWNLOADING;
},
onload: function(e) {
a.info.display.innerText = a.info.name + TEXT_GUI_DOWNLOADED;
}
});
return false;
}
// DownloadAll button onclick
function downloadAllOnclick() {
const a = this;
const index = (a.info.index+1)%3;
for (let i = 0; i < myDlBtns.length; i++) {
if ((i+1)%3 !== index) {continue;};
const btn = myDlBtns[i];
btn.click();
}
return false;
}
}
// Donwload GUI for txtfull
function downloadGUI_txtfull() {
const container = $('#content>table tr>td:nth-child(3)');
const links = arrfilter(container.children, (e,i)=>([1,3,5].includes(i)));
const TEXTS = ['简体(G)', '简体(U)', '繁体(U)'];
const elms = [];
elms.push($CrE('br'));
elms.push(document.createTextNode('程序重命名('));
for (let i = 0; i < links.length; i++) {
const a = links[i];
const btn = $CrE('a');
btn.href = a.previousElementSibling.href;
btn.download = novelInfo.novelName + '.txt';
btn.innerHTML = TEXTS[i];
btn.classList.add(CLASSNAME_BUTTON);
btn.addEventListener('click', downloadFromA);
elms.push(btn);
i+1 < links.length && elms.push(a.previousSibling.cloneNode());
}
elms.push(document.createTextNode(')'));
for (const elm of elms) {
container.appendChild(elm);
}
}
// Donwload GUI for umd
function downloadGUI_umd() {
const container = $('#content>table tr>td:nth-child(5)');
const a = container.firstChild;
const btn = $CrE('a');
btn.href = a.href;
btn.download = novelInfo.novelName + '.umd';
btn.innerHTML = '重命名下载';
btn.classList.add(CLASSNAME_BUTTON);
btn.addEventListener('click', downloadFromA);
a.insertAdjacentElement('afterend', btn);
a.insertAdjacentElement('afterend', $CrE('br'));
}
// Donwload GUI for jar
function downloadGUI_jar() {
const container = $('#content>table tr>td:nth-child(5)');
const links = arrfilter(container.children, ()=>(true));
const TEXTS = ['重命名JAR', '重命名JAD'];
const EXTS = ['.jar', '.jad'];
const elms = [];
elms.push($CrE('br'));
for (let i = 0; i < links.length; i++) {
const a = links[i];
const btn = $CrE('a');
btn.href = a.href;
btn.download = novelInfo.novelName + EXTS[i];
btn.innerHTML = TEXTS[i];
btn.classList.add(CLASSNAME_BUTTON);
btn.addEventListener('click', downloadFromA);
elms.push(btn);
i+1 < links.length && elms.push(a.nextSibling.cloneNode());
}
for (const elm of elms) {
container.appendChild(elm);
}
$('#content>table tr>th:nth-child(4)').setAttribute('width', '47%');
$('#content>table tr>th:nth-child(5)').setAttribute('width', '20%');
}
function downloadFromA(e) {
e.preventDefault();
const btn = e.target;
const url = btn.href;
downloadFile({
url: url,
name: btn.download
});
}
// Get all name display elements
function getAllNameEles() {
return $All('.grid tbody tr .odd');
}
// Get all names
function getAllNames() {
const all = getAllNameEles()
const names = [];
for (let i = 0; i < all.length; i++) {
names[i] = all[i].innerText;
}
return names;
}
// Server GUI
function serverGUI() {
let servers = $All('#content>b');
let serverEles = [];
for (i = 0; i < servers.length; i++) {
if (servers[i].innerText.includes('wenku8.com')) {
serverEles.push(servers[i]);
}
}
for (i = 0; i < serverEles.length; i++) {
serverEles[i].classList.add(CLASSNAME_BUTTON);
serverEles[i].addEventListener('click', function () {
changeAllServers(this.innerText);
});
settip(serverEles[i], TEXT_TIP_SERVERCHANGE);
}
}
// Change all server elements
function changeAllServers(server) {
let i;
const allA = $All('.even a');
for (i = 0; i < allA.length; i++) {
changeServer(server, allA[i]);
}
}
// Change server for an element
function changeServer(server, element) {
if (!element.href) {return false;};
element.href = element.href.replace(/\/\/dl\d?\.wenku8\.com\//g, '//' + server + '/');
}
// Array.prototype.filter
function arrfilter(arr, callback) {
return Array.prototype.filter.call(arr, callback);
}
}
// Login page add-on
function pageLogin() {
const form = $('form[name="frmlogin"]');
if (!form) {return false;}
const eleUsername = $(form, 'input.text[name="username"]');
const elePassword = $(form, 'input.text[name="password"]')
catchAccount();
// Save account info
function catchAccount() {
form.addEventListener('submit', () => {
const config = CONFIG.GlobalConfig.getConfig();
const username = eleUsername.value;
const password = elePassword.value;
config.users = config.users ? config.users : {};
config.users[username] = {
username: username,
password: password
}
CONFIG.GlobalConfig.saveConfig(config);
});
}
}
// Account fast switching
function multiAccount() {
if (!$('.fl')) {return false;};
GUI();
function GUI() {
// Add switch select
const eleTopLeft = $('.fl');
const eletext = $CrE('span');
const sltSwitch = $CrE('select');
eletext.innerText = TEXT_GUI_ACCOUNT_SWITCH;
eletext.classList.add(CLASSNAME_TEXT);
eletext.style.marginLeft = '0.5em';
eleTopLeft.appendChild(eletext);
eleTopLeft.appendChild(sltSwitch);
// Not logged in, create and select an empty option
// Select current user's option
if (!getUserName()) {
appendOption(TEXT_GUI_ACCOUNT_NOTLOGGEDIN, '').selected = true;
};
// Add select options
const userConfig = CONFIG.GlobalConfig.getConfig();
const users = userConfig.users ? userConfig.users : {};
const names = Object.keys(users);
if (names.length === 0) {
appendOption(TEXT_GUI_ACCOUNT_NOACCOUNT, '');
settip(sltSwitch, TEXT_TIP_ACCOUNT_NOACCOUNT);
}
for (const username of names) {
appendOption(username, username)
}
// Select current user's option
if (getUserName()) {selectCurUser();};
// onchange: switch account
sltSwitch.addEventListener('change', (e) => {
const select = e.target;
if (!select.value || !confirm(TEXT_GUI_ACCOUNT_CONFIRM.replace('{N}', select.value))) {
selectCurUser();
destroyEvent(e);
return;
}
switchAccount(select.value);
});
function appendOption(text, value) {
const option = $CrE('option');
option.innerText = text;
option.value = value;
sltSwitch.appendChild(option);
return option;
}
function selectCurUser() {
for (const option of $All(sltSwitch, 'option')) {
option.selected = getUserName().toLowerCase() === option.value.toLowerCase();
}
}
}
function switchAccount(username) {
// Logout
alertify.notify(TEXT_ALT_ACCOUNT_WORKING_LOGOFF);
GM_xmlhttpRequest({
method: 'GET',
url: URL_USRLOGOFF,
onload: function(response) {
// Login
alertify.notify(TEXT_ALT_ACCOUNT_WORKING_LOGIN);
const account = CONFIG.GlobalConfig.getConfig().users[username];
const data = DATA_XHR_LOGIN
.replace('{U}', $URL.encode(account.username))
.replace('{P}', $URL.encode(account.password))
.replace('{C}', $URL.encode('315360000')) // Expire time: 1 year
GM_xmlhttpRequest({
method: 'POST',
url: URL_USRLOGIN,
data: data,
headers: {
"Content-Type": "application/x-www-form-urlencoded"
},
onload: function() {
let box = alertify.success(TEXT_ALT_ACCOUNT_SWITCHED.replace('{N}', username));
redirectGMStorage(getUserID());
DoLog(LogLevel.Info, 'GM_storage redirected to ' + String(getUserID()));
const timeout = setTimeout(()=>{location.href=location.href;}, 3000);
box.callback = (isClicked) => {
isClicked && clearTimeout(timeout);
};
}
})
}
})
}
}
// API page and its sub pages add-on
function pageAPI(API) {
addStyle(CSS_PAGE_API, 'plus_api_css');
//logAPI();
let result;
switch(API) {
case 'modules/article/addbookcase.php':
result = pageAddbookcase();
break;
case 'modules/article/packshow.php':
result = pagePackshow();
break;
default:
result = logAPI();
}
return result;
function logAPI() {
DoLog('This is wenku API page.');
DoLog('API is: [' + API + ']');
DoLog('There is nothing to do. Quiting...');
}
function pageAddbookcase() {
// Append link to bookcase page
addBottomButton({
href: `https://${location.host}/modules/article/bookcase.php`,
innerHTML: TEXT_GUI_API_ADDBOOKCASE_TOBOOKCASE
});
// Append link to remove from bookcase (not finished)
/*addBottomButton({
href: `https://${location.host}/modules/article/bookcase.php?delid=` + getUrlArgv('bid'),
innerHTML: TEXT_GUI_API_ADDBOOKCASE_REMOVE,
onclick: function() {
confirm('确实要将本书移出书架么?')
}
});*/
}
function pagePackshow() {
// Load packshow page
loadPage();
// Packshow page loader
function loadPage() {
// Data
const language = getLang();
const aid = getUrlArgv('id');
const type = getUrlArgv('type');
if (!['txt', 'txtfull', 'umd', 'jar'].includes(type)) {
return false;
}
// Hide api box
const apiBox = $('body>div:nth-child(1)');
apiBox.style.display = 'none';
// Disable api css
addStyle('', 'plus_api_css');
// AsyncManager
const resource = {xmlIndex: null, xmlInfo: null, oDoc: null};
const AM = new AsyncManager();
AM.onfinish = fetchFinish;
// Show soft alert
alertify.message(TEXT_TIP_API_PACKSHOW_LOADING);
// Set Title
document.title = TEXT_GUI_API_PACKSHOW_TITLE_LOADING;
// Load model page
const bgImage = $('body>.plus_cbty_image');
AM.add();
getDocument(URL_PACKSHOW.replace('{A}', "1").replace('{T}', type), function(oDoc) {
resource.oDoc = oDoc;
// Insert body elements
const nodes = Array.prototype.map.call(oDoc.body.childNodes, (elm) => (elm));
for (const node of nodes) {
document.body.insertBefore(node, bgImage);
}
// Insert css link and scripts
const links = Array.prototype.map.call($All(oDoc, 'link[rel="stylesheet"][href]'), (elm) => (elm));
const olinks = Array.prototype.map.call($All('link[rel="stylesheet"][href]'), (elm) => (elm));
for (const link of links) {
if (!link.href.startsWith('http')) {continue;}
for (const olink of Array.prototype.filter.call(olinks, (l) => (l.href === link.href))) {olink.parentElement.removeChild(olink);}
document.head.appendChild(link);
}
const scripts = Array.prototype.map.call($All(oDoc, 'script[src]'), (elm) => (elm));
for (const script of scripts) {
if (!script.src.startsWith('http')) {continue;}
if (Array.prototype.filter.call($All('script[src]'), (s) => (s.src === script.src)).length > 0) {continue;}
document.head.appendChild(script);
}
// Fix all <a>.href
Array.from($All('a')).forEach((a) => {
if (/https?:\/\/www\.wenku8\.(net|cc)\/modules\/article\/packshow.php\??/.test(a.href)) {
a.href = a.href.replace(/([\?&])id=\d+/, '$1id='+aid)
}
});
AM.finish();
});
// Load novel index
AM.add();
AndAPI.getNovelIndex({
aid: aid,
lang: language,
callback: function(xml) {
resource.xmlIndex = xml;
AM.finish();
}
});
AM.add();
AndAPI.getNovelShortInfo({
aid: aid,
lang: language,
callback: function(xml) {
resource.xmlInfo = xml;
AM.finish();
}
});
AM.finishEvent = true;
function fetchFinish() {
// Resources
const xmlIndex = resource.xmlIndex;
const xmlInfo = resource.xmlInfo;
const oDoc = resource.oDoc;
// Elements
const content = $('#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)'];
// Set Title
document.title = TEXT_GUI_API_PACKSHOW_TITLE.replace('{N}', name);
// Set book
aBook.innerText = name;
aBook.href = URL_BOOKINTRO.replace('{A}', aid);
// Load book index
loadIndex();
// Soft alert
alertify.success(TEXT_TIP_API_PACKSHOW_LOADED);
// Enter common download page enhance
pageDownload();
// 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 = TEXT_GUI_UNKNOWN;
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 = toArray(tr.children);
tds[0].innerText = tds[1].innerText = TEXT_GUI_UNKNOWN;
tds[2].innerText = lastupdate;
tds[3].innerText = $(xmlIndex, 'volume:first-child').childNodes[0].nodeValue + '—' + $(xmlIndex, 'volume:last-child').childNodes[0].nodeValue;
const as = [].concat(toArray($All(tds[4], 'a'))).concat(toArray($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();
}
function toArray(_arr) {
return Array.prototype.map.call(_arr, (elm) => (elm));
}
}
}
}
// 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]);});
}
}
// Check if current page is an wenku API page ('处理成功', '出现错误!')
function isAPIPage() {
// API page has just one .block div and one close-page button
const block = $All('.block');
const close = $All('a[href="javascript:window.close()"]');
return block.length === 1 && close.length === 1;
}
// Basic functions
// querySelector
function $() {
switch(arguments.length) {
case 2:
return arguments[0].querySelector(arguments[1]);
break;
default:
return document.querySelector(arguments[0]);
}
}
// querySelectorAll
function $All() {
switch(arguments.length) {
case 2:
return arguments[0].querySelectorAll(arguments[1]);
break;
default:
return document.querySelectorAll(arguments[0]);
}
}
// createElement
function $CrE() {
switch(arguments.length) {
case 2:
return arguments[0].createElement(arguments[1]);
break;
default:
return document.createElement(arguments[0]);
}
}
// Object1[prop] ==> Object2[prop]
function copyProp(obj1, obj2, prop) {obj1[prop] !== undefined && (obj2[prop] = obj1[prop]);}
function copyProps(obj1, obj2, props) {props.forEach((prop) => (copyProp(obj1, obj2, prop)));}
// Display an alertify prompt for editing user-remark
function editUserRemark(uid, name, callback) {
const config = CONFIG.RemarksConfig.getConfig();
const user = config.user[uid] || {uid: uid, name: name, remark: ''};
// Update name
user.name = name;
CONFIG.RemarksConfig.saveConfig(config);
// Display dialog
alertify.prompt(TEXT_GUI_USER_USERREMARKEDIT_TITLE, TEXT_GUI_USER_USERREMARKEDIT_MSG.replace('{N}', user.name), user.remark, onChange, onCancel);
function onChange(evt, value) {
const config = CONFIG.RemarksConfig.getConfig();
if (value) {
const user = config.user[uid] || {uid: uid, name: name, remark: ''};
user.remark = value;
config.user[uid] = user;
} else {
delete config.user[uid]
}
CONFIG.RemarksConfig.saveConfig(config);
callback(value);
}
function onCancel() {}
}
// Send reply for bookreview
// Arg: {rid, title, content, onload:(oDoc)=>{}, onerror:()=>{}}
function sendReviewReply(detail) {
if (typeof($URL) !== 'object') {
DoLog(LogLevel.Error, 'sendReviewReply: $URL not found.');
return false;
}
const data = '&ptitle=' + $URL.encode(detail.title) + '&pcontent=' + $URL.encode(detail.content);
const url = `https://${location.host}/modules/article/reviewshow.php?rid=` + detail.rid.toString();
GM_xmlhttpRequest({
method: 'POST',
url: url,
headers: {
'Content-Type': 'application/x-www-form-urlencoded'
},
data: data,
responseType: 'blob',
onload: function (response) {
if (!detail.onload) {return false;}
parseDocument(response.response, detail.onload);
},
onerror: function (e) {
detail.onerror && detail.onerror(e);
}
});
}
// 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) {
DoLog(LogLevel.Error, 'AndroidAPI.request Error while requesting "' + url + '"');
DoLog(LogLevel.Error, e);
}
});
};
// 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'
});
};
}
// Create reply-area with enhanced UBBEditor
function makeEditor(parent, rid, aid) {
parent.innerHTML = `<form name="frmreview" method="post" action="https://${location.host}/modules/article/reviewshow.php?rid={RID}"><table class="grid" width="100%" align="center"><tbody><tr><td class="odd" width="25%">标题</td><td class="even"><input type="text" class="text" name="ptitle" id="ptitle" size="60" maxlength="60" value="" /></td></tr></tbody><caption>回复书评:</caption><tbody><tr><td class="odd" width="25%">内容(每帖+1积分)</td><td class="even"><textarea class="textarea" name="pcontent" id="pcontent" cols="60" rows="12"></textarea></td></tr><tr><td class="odd" width="25%"> </td><td class="even"><input type="submit" name="Submit" class="button" value="发表书评(Ctrl+Enter)" style="padding: 0.3em 0.4em; height: auto;" /><span></span></td></tr></tbody></table></form>`.replace('{RID}', rid).replace('{AID}', aid);
const script = $CrE('script');
script.innerHTML = `loadJs("https://${location.host}/scripts/ubbeditor_gbk.js", function(){UBBEditor.Create("pcontent");});`;
$(parent, '#pcontent').parentElement.appendChild(script);
areaReply();
}
// getMyUserDetail with soft alerts
function refreshMyUserDetail(callback, args=[]) {
alertify.notify(TEXT_ALT_USRDTL_REFRESH);
getMyUserDetail(function() {
const alertBox = alertify.success(TEXT_ALT_USRDTL_REFRESHED);
// rewrite onclick function from copying to showing details
alertBox.callback = function(isClicked) {
isClicked && alertify.message(altMyUserDetail()/*JSON.stringify(getMyUserDetail())*/);
}
// callback if exist
callback ? callback.apply(args) : function() {};
})
}
// Get my user info detail
// if no argument provided, this function will just read userdetail from gm_storage
// otherwise, the function will make a http request to get the latest userdetail
// if no argument provided and no gm_storage record, then will just return false
// if not logged in, return false
// if callback is not a function, then will just request&store but not callback
function getMyUserDetail(callback, args=[]) {
if (getUserID() === null) {
return false;
}
if (callback) {
requestWeb();
return true;
} else {
const storage = CONFIG.userDtlePrefs.getConfig();
if (!storage.userDetail && !storage.userFriends) {
DoLog(LogLevel.Warning, 'Attempt to read userDetail from gm_storage but no record found');
return false;
};
const userDetail = storage;
return userDetail;
}
function requestWeb() {
const lastStorage = CONFIG ? CONFIG.userDtlePrefs.getConfig() : undefined;
let restXHR = 2;
let storage = {};
// Request userDetail
getDocument(URL_USRDETAIL, detailLoaded)
// Request userFriends
getDocument(URL_USRFRIEND, friendLoaded)
function detailLoaded(oDoc) {
const content = $(oDoc, '#content');
storage.userDetail = {
userID: Number($(content, 'tr:nth-child(1)>.even').innerText), // '用户ID'
userLink: $(content, 'tr:nth-child(2)>.even').innerText, // '推广链接'
userName: $(content, 'tr:nth-child(3)>.even').innerText, // '用户名'
displayName: $(content, 'tr:nth-child(4)>.even').innerText, // '用户昵称'
userType: $(content, 'tr:nth-child(5)>.even').innerText, // '等级'
userGrade: $(content, 'tr:nth-child(6)>.even').innerText, // '头衔'
gender: $(content, 'tr:nth-child(7)>.even').innerText, // '性别'
email: $(content, 'tr:nth-child(8)>.even').innerText, // 'Email'
qq: $(content, 'tr:nth-child(9)>.even').innerText, // 'QQ'
msn: $(content, 'tr:nth-child(10)>.even').innerText, // 'MSN'
site: $(content, 'tr:nth-child(11)>.even').innerText, // '网站'
signupDate: $(content, 'tr:nth-child(13)>.even').innerText, // '注册日期'
contibute: $(content, 'tr:nth-child(14)>.even').innerText, // '贡献值'
exp: $(content, 'tr:nth-child(15)>.even').innerText, // '经验值'
credit: $(content, 'tr:nth-child(16)>.even').innerText, // '现有积分'
friends: $(content, 'tr:nth-child(17)>.even').innerText, // '最多好友数'
mailbox: $(content, 'tr:nth-child(18)>.even').innerText, // '信箱最多消息数'
bookcase: $(content, 'tr:nth-child(19)>.even').innerText, // '书架最大收藏量'
vote: $(content, 'tr:nth-child(20)>.even').innerText, // '每天允许推荐次数'
sign: $(content, 'tr:nth-child(22)>.even').innerText, // '用户签名'
intoduction: $(content, 'tr:nth-child(23)>.even').innerText, // '个人简介'
userImage: $(content, 'tr>td>img').src // '头像'
}
loaded();
}
function friendLoaded(oDoc) {
const content = $(oDoc, '#content');
const trs = $All(content, 'tr');
const friends = [];
const lastFriends = lastStorage ? lastStorage.userFriends : undefined;
for (let i = 1; i < trs.length; i++) {
getFriends(trs[i]);
}
storage.userFriends = friends;
loaded();
function getFriends(tr) {
// Check if userID exist
if (isNaN(Number($(tr.children[2], 'a').href.match(/\?uid=(\d+)/)[1]))) {return false;};
// Collect information
let friend = {
userID: Number($(tr.children[2], 'a').href.match(/\?uid=(\d+)/)[1]),
userName: tr.children[0].innerText,
signupDate: tr.children[1].innerText
}
friend = fillLocalInfo(friend)
friends.push(friend);
}
function fillLocalInfo(friend) {
if (!lastFriends) {return friend;};
for (const f of lastFriends) {
if (f.userID === friend.userID) {
for (const [key, value] of Object.entries(f)) {
if (friend.hasOwnProperty(key)) {continue;};
friend[key] = value;
}
break;
}
}
return friend;
}
}
function loaded() {
restXHR--;
if (restXHR === 0) {
// Save to gm_storage
if (CONFIG) {
storage.lasttime = getTime('-', false);
CONFIG.userDtlePrefs.saveConfig(storage);
}
// Callback
typeof(callback) === 'function' ? callback.apply(null, [storage].concat(args)) : function() {};
}
}
}
}
// Show userdetail in an alertify alertbox
function altMyUserDetail() {
const json = getMyUserDetail();
alertify.message(JSON.stringify(getMyUserDetail()));
}
function exportConfig(noPass=false) {
// Get config
const config = {};
const getValue = window.getValue ? window.getValue : GM_getValue;
const listValues = window.listValues ? window.listValues : GM_listValues;
for (const key of listValues()) {
config[key] = getValue(key);
}
// Remove username and password if required
noPass && (config[KEY_CM].users = {});
// Download
const text = JSON.stringify(config);
const name = '轻小说文库+_配置文件({P})_v{V}_{T}.wkp'.replace('{P}', noPass ? '无账号密码' : '含账号密码').replace('{V}', GM_info.script.version).replace('{T}', getTime());
downloadText(text, name);
alertify.success(TEXT_ALT_CONFIG_EXPORTED.replace('{N}', name));
}
function importConfig(json) {
// Redirect
redirectGMStorage();
// Preserve users
const users = GM_getValue('Config-Manager').users;
// Delete json
for (const [key, value] of GM_listValues()) {
GM_deleteValue(key, value);
}
// Set json
for (const [key, value] of Object.entries(json)) {
GM_setValue(key, value);
}
// Preserve users
const config = GM_getValue('Config-Manager', {});
if (!config.users) {config.users = {}}
for (const [name, user] of Object.entries(users)) {
config.users[name] = user;
}
GM_setValue('Config-Manager', config);
// Reload
location.reload();
}
function sortLaterReads(books, sortby) {
const sorter = FUNC_LATERBOOK_SORTERS[sortby].sorter;
return Object.values(books).sort(sorter);
}
function getUserID() {
const match = $URL.decode(document.cookie).match(/jieqiUserId=(\d+)/);
const id = match && match[1] ? Number(match[1]) : null;
return isNaN(id) ? null : id;
}
function getUserName() {
const match = $URL.decode(document.cookie).match(/jieqiUserName=([^, ;]+)/);
const name = match ? match[1] : null;
return name;
}
// Reload page without re-sending form data, and keeps reviewshow-page
function reloadPage() {
const url = /^https?:\/\/www\.wenku8\.(net|cc)\/modules\/article\/reviewshow\.php/.test(location.href) ? URL_REVIEWSHOW_2.replace('{R}', getUrlArgv('rid')).replace('{P}', $('#pagelink>strong').innerText) : location.href;
location.href = url;
}
// Check if tipobj is ready, if not, then make it
function tipcheck() {
DoLog(LogLevel.Info, 'checking tipobj...');
if (typeof(tipobj) === 'object' && tipobj !== null) {
DoLog(LogLevel.Info, 'tipobj ready...');
return true;
} else {
DoLog(LogLevel.Warning, 'tipobj not ready');
if (typeof(tipinit) === 'function') {
DoLog(LogLevel.Success, 'tipinit executed');
tipinit();
return true;
} else {
DoLog(LogLevel.Error, 'tipinit not found');
return false;
}
}
}
// New tipobj movement method. Makes sure the tipobj stay close with the mouse.
function tipscroll() {
if (!tipready) {return false;}
DoLog('tipscroll executed. ')
tipobj.style.position = 'fixed';
window.addEventListener('mousemove', tipmoveplus)
return true;
function tipmoveplus(e) {
tipobj.style.left = e.clientX + tipx + 'px';
tipobj.style.top = e.clientY + tipy + 'px';
}
}
// show & hide tip when mouse in & out. accepts tip as a string or a function that returns the tip string
function settip(elm, tip) {
typeof(tip) === 'string' && (elm.tiptitle = tip);
typeof(tip) === 'function' && (elm.tipgetter = tip);
elm.removeEventListener('mouseover', showtip);
elm.removeEventListener('mouseout', hidetip);
elm.addEventListener('mouseover', showtip);
elm.addEventListener('mouseout', hidetip);
}
function showtip(e) {
if (e && e.currentTarget && (e.currentTarget.tiptitle || e.currentTarget.tipgetter)) {
const tip = e.currentTarget.tiptitle || e.currentTarget.tipgetter();
if (tipready) {
tipshow(tip);
e.currentTarget.title && e.currentTarget.removeAttribute('title');
} else {
e.currentTarget.title = e.currentTarget.tiptitle;
}
} else if (typeof(e) === 'string') {
tipready && tipshow(e);
}
}
function hidetip() {
tipready && tiphide();
}
// Side-located control panel
// Requirements: FontAwesome, tooltip.css(from https://github.com/felipefialho/css-components/blob/main/build/tooltip/tooltip.css)
// 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'
},
_css: '#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;}',
_directions: ['left', 'right', 'top', 'bottom']
};
Object.defineProperty(SP, 'css', {
configurable: false,
enumerable: true,
get: () => (_SP.css),
set: (css) => {
_SP.css = css;
spAddStyle(css, _SP._id.css);
}
});
Object.defineProperty(SP, 'usercss', {
configurable: false,
enumerable: true,
get: () => (_SP.usercss),
set: (usercss) => {
_SP.usercss = usercss;
spAddStyle(usercss, _SP._id.usercss);
}
});
SP.css = _SP._css;
SP.create = function() {
// Create panel
const panel = elms.panel = document.createElement('div');
panel.id = _SP._id.panel;
SP.setPosition('bottom-right');
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 = document.createElement('div');
text && (button.innerHTML = text);
id && (button.id = id);
tip && setTooltip(button, tip); //settip(button, tip);
className && (button.className = className);
onclick && (button.onclick = onclick);
if (faicon) {
const i = document.createElement('i');
i.className = faicon;
button.faicon = i;
button.appendChild(i);
}
for (const listener of listeners) {
button.addEventListener.apply(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.parentElement.removeChild(elm);
}
// 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(function(button) {
if (button.getAttribute('role') === 'tooltip') {
setTooltipDirection(button)
}
});
}
// Gets the current display position
SP.getPosition = function() {
return _SP.direction.join('-');
}
// 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.tooltip = tippy(elm, {
content: text,
arrow: true,
hideOnClick: false
});
// Old version, uses tooltip.css
/*
elm.setAttribute('role', 'tooltip');
elm.setAttribute('aria-label', text);
*/
setTooltipDirection(elm, direction);
}
function setTooltipDirection(elm, direction='auto') {
direction === 'auto' && (direction = _SP.direction.includes('left') ? 'right' : 'left');
if (!_SP._directions.includes(direction)) {throw new Error('setTooltip: invalid direction');}
// Tippy direction
if (!elm.tooltip) {
DoLog(LogLevel.Error, 'SidePanel.setTooltipDirection: Given elm has no tippy instance(elm.tooltip)');
throw new Error('SidePanel.setTooltipDirection: Given elm has no tippy instance(elm.tooltip)');
}
elm.tooltip.setProps({
placement: direction
});
// Old version, uses tooltip.css
/*
for (const dirct of _SP._directions) {
elm.classList.remove('tooltip-'+dirct);
}
elm.classList.add('tooltip-'+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;
}
}
// 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(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;
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(CLASSNAME_LIST_BUTTON);
li.classList.add(CLASSNAME_LIST_ITEM);
item.value && (btn.value = item.value);
item.onclick && btn.addEventListener('click', item.onclick);
item.tip && settip(li, item.tip);
item.tip && 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';
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;
}
}
// A table-based setting panel using alertify-js
// Requires: alertify-js
// Use 'new' keyword
// Usage:
/*
var panel = new SettingPanel({
className: '',
id: '',
name: '',
tables: [
{
className: '',
id: '',
name: '',
rows: [
{
className: '',
id: '',
name: '',
blocks: [
{
innerHTML / innerText: ''
colSpan: 1,
rowSpan: 1,
className: '',
id: '',
name: '',
children: [HTMLElement, ...]
},
...
]
},
...
]
},
...
]
});
*/
function SettingPanel(details={}) {
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 = $C('div');
copyProps(details, elm, ['id', 'name', 'className']);
elm.classList.add('settingpanel-container');
// Configure object
let css='', usercss='';
SP.element = elm;
SP.elements = {};
SP.children = {};
SP.tables = [];
SP.length = 0;
details.id !== undefined && (SP.elements[details.id] = elm);
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; text-align: center; vertical-align: middle; padding: 3px; text-align: left;}'
// Create tables
if (details.tables) {
for (const table of details.tables) {
if (table instanceof PanelTable) {
appendTable(table);
} else {
appendTable(new PanelTable(table));
}
}
}
// Make alerity box
const box = SP.alertifyBox = alertify.alert();
clearChildNodes(box.elements.content);
box.elements.content.appendChild(elm);
box.elements.content.style.overflow = 'auto';
box.setHeader(TEXT_GUI_DETAIL_MANAGE_HEADER);
box.setting({
maximizable: true,
overflow: true
});
box.show();
// 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;
// Inherit elements
for (const [id, subelm] of Object.entries(table.elements)) {
SP.elements[id] = subelm;
}
// Inherit children
for (const [id, child] of Object.entries(table.children)) {
SP.children[id] = child;
}
return 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 = $C('table');
copyProps(details, elm, ['id', 'name', 'className']);
elm.classList.add('settingpanel-table');
// Configure
PT.element = elm;
PT.elements = {};
PT.children = {};
PT.rows = [];
PT.length = 0;
details.id !== undefined && (PT.elements[details.id] = elm);
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);
row.id !== undefined && (PT.children[row.id] = row);
PT.length++;
// Set parent
row.parent = PT;
// Inherit elements
for (const [id, subelm] of Object.entries(row.elements)) {
PT.elements[id] = subelm;
}
// Inherit children
for (const [id, child] of Object.entries(row.children)) {
PT.children[id] = child;
}
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 = $C('tr');
copyProps(details, elm, ['id', 'name', 'className']);
elm.classList.add('settingpanel-row');
// Configure object
PR.element = elm;
PR.elements = {};
PR.children = {};
PR.blocks = [];
PR.length = 0;
details.id !== undefined && (PR.elements[details.id] = elm);
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);
block.id !== undefined && (PR.children[block.id] = block);
PR.length++;
// Set parent
block.parent = PR;
// Inherit elements
for (const [id, subelm] of Object.entries(block.elements)) {
PR.elements[id] = subelm;
}
// Inherit children
for (const [id, child] of Object.entries(block.children)) {
PR.children[id] = child;
}
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 = $C('td');
copyProps(details, elm, ['innerText', 'innerHTML', 'colSpan', 'rowSpan', 'id', 'name', 'className']);
elm.classList.add('settingpanel-block');
// Configure object
PB.element = elm;
PB.elements = {};
PB.children = {};
details.id !== undefined && (PB.elements[details.id] = elm);
copyProps(details, PB, ['id', 'name']);
// Append to parent if need
details.parent instanceof PanelRow && (PB.parent = details.parent.appendBlock(PB));
// 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 $(e) {return document.querySelector(e);}
function $C(e) {return document.createElement(e);}
function $R(e) {return $(e) && $(e).parentElement.removeChild($(e));}
function clearChildNodes(elm) {for (const el of elm.childNodes) {elm.removeChild(el);}}
function copyProp(obj1, obj2, prop) {obj1[prop] !== undefined && (obj2[prop] = obj1[prop]);}
function copyProps(obj1, obj2, props) {props.forEach((prop) => (copyProp(obj1, obj2, prop)));}
function insertItem(arr, item, index) {
for (let i = arr.length; i > index ; i--) {
arr[i] = arr[i-1];
}
arr[index] = item;
return arr;
}
function removeItem(arr, index) {
for (let i = index; i < arr.length-1; i++) {
arr[i] = arr[i+1];
}
delete arr[arr.length-1];
return arr;
}
function addStyle(css, id) {
$R('#'+id);
const style = $C('style');
style.innerHTML = css;
style.id = id;
document.head.appendChild(style);
return style
}
}
// Create a left .block operatingArea
// options = {type: '', ...opts}
// Supported type: 'mypage', 'toplist'
function createWenkuBlock(details) {
// Args
//title=TEXT_GUI_BLOCK_TITLE_DEFULT, append=false, options
const title = details.title || TEXT_GUI_BLOCK_TITLE_DEFULT;
const parent = ({'string': $(details.parent), 'object': details.parent})[typeof details.parent];
const type = details.type ? details.type.toLowerCase() : null;
const items = details.items;
const options = details.options;
// Standard block
const stdBlock = makeStandardBlock();
const block = stdBlock.block;
const blocktitle = stdBlock.blocktitle;
const blockcontent = stdBlock.blockcontent;
blocktitle.innerHTML = title;
makeContent();
parent && parent.appendChild(block);
return block;
// Create a standard block structure
function makeStandardBlock() {
const block = $CrE('div'); block.classList.add('block');
const blocktitle = $CrE('div'); blocktitle.classList.add('blocktitle');
const blockcontent = $CrE('div'); blockcontent.classList.add('blockcontent');
block.appendChild(blocktitle); block.appendChild(blockcontent);
return {block: block, blocktitle: blocktitle, blockcontent: blockcontent};
}
function makeContent() {
switch (type) {
case 'mypage': typeMypage(); break;
case 'toplist': typeToplist(); break;
case 'imagelist': typeImglist(); break;
case 'element': typeElement(); break;
default: DoLog(LogLevel.Error, 'createWenkuBlock: Invalid block type');
}
}
// Links such as https://www.wenku8.net/userdetail.php
function typeMypage() {
const ul = $CrE('ul');
ul.classList.add('ulitem');
for (const link of details.items) {
const li = $CrE('li');
const a = $CrE('a');
a.href = link.href ? link.href : 'javascript: void(0);';
link.href && (a.target = '_blank');
link.tiptitle && settip(a, link.tiptitle);
a.innerHTML = link.innerHTML;
a.id = link.id ? link.id : '';
li.appendChild(a);
ul.appendChild(li);
}
blockcontent.appendChild(ul);
}
// Links such as top-books-list inside #right in index page
// links = [...{href: '', innerHTML: '', tiptitle: '', id: ''}]
function typeToplist() {
const ul = $CrE('ul');
ul.classList.add('ultop');
for (const link of details.items) {
const li = $CrE('li');
const a = $CrE('a');
a.href = link.href ? link.href : 'javascript: void(0);';
link.href && (a.target = '_blank');
link.tiptitle && settip(a, link.tiptitle);
a.innerHTML = link.innerHTML;
a.id = link.id ? link.id : '';
li.appendChild(a);
ul.appendChild(li);
}
blockcontent.appendChild(ul);
}
// Links with images like center blocks in index page
function typeImglist() {
const container = $CrE('div');
container.style.height = '155px';
for (const item of items) {
const div = $CrE('div');
div.setAttribute('style', 'float: left;text-align:center;width: 95px; height:155px;overflow:hidden;');
const a = $CrE('a');
a.href = item.href;
a.target = '_blank';
item.tiptitle && settip(a, item.tiptitle);
const img = $CrE('img');
img.src = item.src;
setAttributes(img, {
'border': '0',
'width': '90',
'height': '127'
});
a.appendChild(img);
const br = $CrE('br');
const a2 = $CrE('a');
a2.href = item.href;
a2.target = '_blank';
a2.innerHTML = item.text;
div.appendChild(a);
div.appendChild(br);
div.appendChild(a2);
container.appendChild(div);
}
blockcontent.appendChild(container);
}
// Just append given elements into block content
function typeElement() {
const elms = Array.isArray(items) ? items : [items];
for (const elm of elms) {
blockcontent.appendChild(elm);
}
}
// Set attributes to an element
function setAttributes(elm, attributes) {
for (const [name, attr] of Object.entries(attributes)) {
elm.setAttribute(name, attr);
}
}
}
// Get a review's last page url
function getLatestReviewPageUrl(rid, callback, args=[]) {
const reviewUrl = `https://${location.host}/modules/article/reviewshow.php?rid=` + String(rid);
getDocument(reviewUrl, firstPage, args);
function firstPage(oDoc, ...args) {
const url = $(oDoc, '#pagelink>a.last').href;
args = [url].concat(args);
callback.apply(null, args);
};
};
// Upload image to KIENG images
// details: {file: File, onload: Function, onerror: Function, type: 'sm.ms/jd/sg/tt/...'}
function uploadImage(details) {
const file = details.file;
const onload = details.onload ? details.onload : function() {};
const onerror = details.onerror ? details.onerror : uploadError;
const type = details.type ? details.type : CONFIG.UserGlobalCfg.getConfig().imager;
if (!DATA_IMAGERS.hasOwnProperty(type) || !DATA_IMAGERS[type].available) {
onerror();
return false;
}
const imager = DATA_IMAGERS[type];
const upload = imager.upload;
const request = upload.request;
const response = upload.response;
// Construct request url
let url = request.url;
if (request.urlargs) {
const args = request.urlargs;
const makearg = (key, value) => ('{K}={V}'.replace('{K}', key).replace('{V}', value));
const replacers = {
'$filename$': () => (encodeURIComponent(file.name)),
'$random$': () => (Math.random().toString()),
'$time$': () => ((new Date()).getTime().toString())
};
for (let [key, value] of Object.entries(args)) {
url += url.includes('?') ? '&' : '?';
for (const [str, replacer] of Object.entries(replacers)) {
while (value !== null && value.includes(str)) {
const val = replacer(key);
value = (val !== null) ? value.replace(str, val) : null;
}
}
(value !== null) && (url += makearg(key, value));
}
}
// Construst request body
let data;
if (request.data) {
data = new FormData();
const replacers = {
'$file$': (key) => ((data.append(key, file), null)),
'$random$': () => (Math.random().toString()),
'$time$': () => ((new Date()).getTime().toString())
};
for (let [key, value] of Object.entries(request.data)) {
for (const [str, replacer] of Object.entries(replacers)) {
while (value !== null && value.includes(str)) {
const val = replacer(key);
value = (val !== null) ? value.replace(str, val) : null;
}
}
(value !== null) && data.append(key, value);
}
} else {
data = file;
}
// headers
const headers = request.headers || {};
GM_xmlhttpRequest({
method: 'POST',
url: url,
timeout: 15 * 1000,
data: data,
headers: headers,
responseType: request.responseType ? request.responseType : 'json',
onerror: onerror,
ontimeout: onerror,
onabort: onerror,
onload: (e) => {
const json = e.response;
const success = e.status === 200 && response.checksuccess(json);
if (success) {
const url = response.geturl(json);
const name = response.getname ? (response.getname(json) ? response.getname(json) : TEXT_ALT_IMAGE_RESPONSE_NONAME) : TEXT_ALT_IMAGE_RESPONSE_NONAME
onload({
url: url,
name: name,
});
} else {
onerror(json);
return;
}
}
})
/* Common xhr version. Cannot bypass CORS.
const re = new XMLHttpRequest();
re.open('POST', request.url, true);
re.timeout = 15 * 1000;
re.onerror = re.ontimeout = re.onabort = uploadError;
re.responseType = request.responseType ? request.responseType : 'json';
re.onload = (e) => {
const json = re.response;
const success = response.checksuccess(json)
if (success) {
onload({
url: response.geturl(json),
name: response.getname ? response.getname(json) : TEXT_ALT_IMAGE_RESPONSE_NONAME,
});
} else {
uploadError(json);
return;
}
}
re.send(data);*/
function uploadError(json) {
alertify.error(TEXT_ALT_IMAGE_UPLOAD_ERROR);
DoLog(LogLevel.Error, [TEXT_ALT_IMAGE_UPLOAD_ERROR, json]);
}
}
// Wait until a variable loaded, and call callback
function waitUntilLoaded(varnames, callback, args=[]) {
if (!varnames) {callback.apply(null, args)}
if (!Array.isArray(varnames)) {varnames = [varnames];}
const AM = new AsyncManager();
AM.onfinish = function() {
callback.apply(null, args);
};
for (const varname of varnames) {
AM.add();
makeWaitFunc(varname, AM)();
}
AM.finishEvent = true;
function makeWaitFunc(varname, AM) {
return function wait() {
if (typeof(getvar(varname)) === 'undefined') {
setTimeout(wait, NUMBER_ELEMENT_LOADING_WAIT_INTERVAL);
return false;
}
AM.finish();
};
}
}
// Remove all childnodes from an element
function clearChildnodes(element) {
const cns = []
for (const cn of element.childNodes) {
cns.push(cn);
}
for (const cn of cns) {
element.removeChild(cn);
}
}
// Change location.href without reloading using history.pushState/replaceState
function setPageUrl(url, push=false) {
return history[push ? 'pushState' : 'replaceState']({modified: true, ...history.state}, '', url);
}
// Just stopPropagation and preventDefault
function destroyEvent(e) {
if (!e) {return false;};
if (!e instanceof Event) {return false;};
e.stopPropagation();
e.preventDefault();
}
// eval() function with security check that only allows to get variable values, but don't allow executing js.
function getvar(varname) {
const unsafe_chars = ['(', ')', '+', '-', '*', '/', '&', '|', '[', ']', '=', '^', '%', '!', '.', '<', '>', '\\', '"', '\''];
for (const char of unsafe_chars) {
if (varname.includes(char)) {throw new Error('Function getvar(varname) called with insecure string "{V}"'.replaceAll('V', varname.replaceAll('"', '\\"')))}
}
return eval(varname);
}
// GM_XHR HOOK: The number of running GM_XHRs in a time must under maxXHR
// Returns the abort function to stop the request anyway(no matter it's still waiting, or requesting)
// (If the request is invalid, such as url === '', will return false and will NOT make this request)
// If the abort function called on a request that is not running(still waiting or finished), there will be NO onabort event
// Requires: function delItem(){...} & function uniqueIDMaker(){...}
function GMXHRHook(maxXHR=5) {
const GM_XHR = GM_xmlhttpRequest;
const getID = uniqueIDMaker();
let todoList = [], ongoingList = [];
GM_xmlhttpRequest = safeGMxhr;
function safeGMxhr() {
// Get an id for this request, arrange a request object for it.
const id = getID();
const request = {id: id, args: arguments, aborter: null};
// Deal onload function first
dealEndingEvents(request);
/* DO NOT DO THIS! KEEP ITS ORIGINAL PROPERTIES!
// Stop invalid requests
if (!validCheck(request)) {
return false;
}
*/
// Judge if we could start the request now or later?
todoList.push(request);
checkXHR();
return makeAbortFunc(id);
// Decrease activeXHRCount while GM_XHR onload;
function dealEndingEvents(request) {
const e = request.args[0];
// onload event
const oriOnload = e.onload;
e.onload = function() {
reqFinish(request.id);
checkXHR();
oriOnload ? oriOnload.apply(null, arguments) : function() {};
}
// onerror event
const oriOnerror = e.onerror;
e.onerror = function() {
reqFinish(request.id);
checkXHR();
oriOnerror ? oriOnerror.apply(null, arguments) : function() {};
}
// ontimeout event
const oriOntimeout = e.ontimeout;
e.ontimeout = function() {
reqFinish(request.id);
checkXHR();
oriOntimeout ? oriOntimeout.apply(null, arguments) : function() {};
}
// onabort event
const oriOnabort = e.onabort;
e.onabort = function() {
reqFinish(request.id);
checkXHR();
oriOnabort ? oriOnabort.apply(null, arguments) : function() {};
}
}
// Check if the request is invalid
function validCheck(request) {
const e = request.args[0];
if (!e.url) {
return false;
}
return true;
}
// Call a XHR from todoList and push the request object to ongoingList if called
function checkXHR() {
if (ongoingList.length >= maxXHR) {return false;};
if (todoList.length === 0) {return false;};
const req = todoList.shift();
const reqArgs = req.args;
const aborter = GM_XHR.apply(null, reqArgs);
req.aborter = aborter;
ongoingList.push(req);
return req;
}
// Make a function that aborts a certain request
function makeAbortFunc(id) {
return function() {
let i;
// Check if the request haven't been called
for (i = 0; i < todoList.length; i++) {
const req = todoList[i];
if (req.id === id) {
// found this request: haven't been called
delItem(todoList, i);
return true;
}
}
// Check if the request is running now
for (i = 0; i < ongoingList.length; i++) {
const req = todoList[i];
if (req.id === id) {
// found this request: running now
req.aborter();
reqFinish(id);
checkXHR();
}
}
// Oh no, this request is already finished...
return false;
}
}
// Remove a certain request from ongoingList
function reqFinish(id) {
let i;
for (i = 0; i < ongoingList.length; i++) {
const req = ongoingList[i];
if (req.id === id) {
ongoingList = delItem(ongoingList, i);
return true;
}
}
return false;
}
}
}
// Redirect GM_storage API
// Each key points to a different storage area
// Original GM_functions will be backuped in window object
// PS: No worry for GM_functions leaking, because Tempermonkey's Sandboxing
function redirectGMStorage(key) {
// Recover if redirected before
GM_setValue = typeof(window.setValue) === 'function' ? window.setValue : GM_setValue;
GM_getValue = typeof(window.getValue) === 'function' ? window.getValue : GM_getValue;
GM_listValues = typeof(window.listValues) === 'function' ? window.listValues : GM_listValues;
GM_deleteValue = typeof(window.deleteValue) === 'function' ? window.deleteValue : GM_deleteValue;
// Stop if no key
if (!key) {return;};
// Save original GM_functions
window.setValue = typeof(GM_setValue) === 'function' ? GM_setValue : function() {};
window.getValue = typeof(GM_getValue) === 'function' ? GM_getValue : function() {};
window.listValues = typeof(GM_listValues) === 'function' ? GM_listValues : function() {};
window.deleteValue = typeof(GM_deleteValue) === 'function' ? GM_deleteValue : function() {};
// Redirect GM_functions
typeof(GM_setValue) === 'function' ? GM_setValue = RD_GM_setValue : function() {};
typeof(GM_getValue) === 'function' ? GM_getValue = RD_GM_getValue : function() {};
typeof(GM_listValues) === 'function' ? GM_listValues = RD_GM_listValues : function() {};
typeof(GM_deleteValue) === 'function' ? GM_deleteValue = RD_GM_deleteValue : function() {};
// Get global storage
//const storage = getStorage();
function getStorage() {
return window.getValue(key, {});
}
function saveStorage(storage) {
return window.setValue(key, storage);
}
function RD_GM_setValue(key, value) {
const storage = getStorage();
storage[key] = value;
saveStorage(storage);
}
function RD_GM_getValue(key, defaultValue) {
const storage = getStorage();
return storage[key] || defaultValue;
}
function RD_GM_listValues() {
const storage = getStorage();
return Object.keys(storage);
}
function RD_GM_deleteValue(key) {
const storage = getStorage();
delete storage[key];
saveStorage(storage);
}
}
// Aim to separate big data from config, to boost up the speed of config reading.
// FAILED. NEVER USE THESE CODES. NEVER DO THESE THINGS AGAIN. FUCK MYSELF ME STUPID.
// NOOOOOOOO!!!!!!! WHY ARE YOU DICKHEAD STILL THINGKING ABOUT THIS SHIT??????? NEVER EVER THINK ABOUT THIS FUCKING UNACHIEVABLE FUNCTION AGAIN!!!!!!
// See https://www.wenku8.net/modules/article/reviewshow.php?rid=244568&aid=1973&page=202#yid930393 if you still want to try, you'll pay for that.
function GMBigData(maxsize=1024) {
const BD = this;
BD.maxsize = maxsize;
BD.keyPrefix = 'GM_BIGDATA:' + btoa(encodeURIComponent(GM_info.script.name + (GM_info.script.namespace || '')));
BD.hook = function() {
hookget();
hookset();
}
BD.unhook = function() {
if (!BD.GM_getValue || !BD.GM_setValue) {
throw TypeError('GMBigData: BD.GM_getValue or BD.GM_setValue missing');
}
GM_getValue = BD.GM_getValue;
GM_setValue = BD.GM_setValue;
}
function hookget() {
const oGet = BD.GM_getValue = GM_getValue;
GM_getValue = function(name, defaultValue) {
return decodeValue(oGet(name, defaultValue));
}
function decodeValue(value) {
return (({
'string': decodeString,
'object': value !== null ? decodeObject : null
})[typeof value] || ((v) => (v)))(value);
function decodeString(str) {
return (isDatakey(str) && keyExists(str)) ? localStorage.getItem(str) : str;
}
function decodeObject(obj) {
return new Proxy(obj, {
get: function(target, property, receiver) {
return decodeValue(target[property]);
}
});
}
}
}
function hookset() {
const oSet = BD.GM_setValue = GM_setValue;
GM_setValue = function(name, value) {
const encoded = encodeValue(value);
clearUnusedBigData(encoded);
return oSet(name, encoded);
}
function encodeValue(value) {
return (({
'string': encodeString,
'object': value !== null ? encodeObject : value
})[typeof value] || ((v) => (v)))(value);
function encodeString(str) {
if (getDataSize(str) <= BD.maxsize) {
return str;
} else {
const key = generateKey();
localStorage.setItem(key, str);
return key;
}
}
function encodeObject(obj) {
return new Proxy(obj, {
get: function(target, property, receiver) {
return encodeValue(target[property]);
}
});
}
}
function clearUnusedBigData(data) {
const usingKeys = getAllUsingKeys(data);
for (const key of Object.keys(localStorage)) {
if (isDatakey(key) && !usingKeys.includes(key)) {
localStorage.removeItem(key);
}
}
function getAllUsingKeys(data) {
const usingKeys = [];
(({
'string': checkString,
'object': data !== null ? getAllUsingKeys : null
})[typeof data] || function() {})();
return usingKeys;
function checkString(str) {
isDatakey(str) && keyExists(str) && usingKeys.push(str);
}
}
}
}
// Datakey generator
function generateKey(length=16) {
let datakey = newKey();
while (keyExists(datakey)) {
datakey = newKey();
}
return datakey;
function newKey() {
return BD.keyPrefix + ',' + randstr(length);
}
}
// Check whether a datakey already exists
function keyExists(datakey) {
return Object.keys(localStorage).includes(datakey);
}
// Check whether the value is a datakey
function isDatakey(value) {
return typeof value === 'string' && value.startsWith(BD.keyPrefix);
}
// Get the size of data
function getDataSize(data) {
return (new Blob([data])).size;
}
}
// Download and parse a url page into a html document(dom).
// when xhr onload: callback.apply([dom, args])
function getDocument(url, callback, args=[]) {
GM_xmlhttpRequest({
method : 'GET',
url : url,
responseType : 'blob',
timeout : 15 * 1000,
onloadstart : function() {
DoLog(LogLevel.Info, 'getting document, url=\'' + url + '\'');
},
onload : function(response) {
const htmlblob = response.response;
parseDocument(htmlblob, callback, args);
},
onerror : reqerror,
ontimeout : reqerror
});
function reqerror(e) {
DoLog(LogLevel.Error, 'getDocument: Request Error');
DoLog(LogLevel.Error, e);
throw new Error('getDocument: Request 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);
}
// Get a base64-formatted url of an image
// When image load error occurs, callback will be called without any argument
function getImageUrl(src, fitx, fity, callback, args=[]) {
const image = new Image();
image.setAttribute("crossOrigin",'anonymous');
image.onload = convert;
image.onerror = image.onabort = callback;
image.src = src;
function convert() {
const cvs = $CrE('canvas');
const ctx = cvs.getContext('2d');
let width, height;
if (fitx && fity) {
width = window.innerWidth;
height = window.innerHeight;
} else if (fitx) {
width = window.innerWidth;
height = (width / image.width) * image.height;
} else if (fity) {
height = window.innerHeight;
width = (height / image.height) * image.width;
} else {
width = image.width;
height = image.height;
}
cvs.width = width;
cvs.height = height;
ctx.drawImage(image, 0, 0, width, height);
try {
callback.apply(null, [cvs.toDataURL()].concat(args));
} catch (e) {
DoLog(LogLevel.Error, ['Error at getImageUrl.convert()', e]);
callback();
}
}
}
// Convert a '....' to a Blob object
function b64toBlob(dataURI) {
const mime = dataURI.match(/data:(.+?);/)[1];
const byteString = atob(dataURI.split(',')[1]);
const ab = new ArrayBuffer(byteString.length);
const ia = new Uint8Array(ab);
for (let i = 0; i < byteString.length; i++) {
ia[i] = byteString.charCodeAt(i);
}
return new Blob([ab], {type: mime});
}
//将base64转换为文件
function dataURLtoFile(dataurl, filename) {
var arr = dataurl.split(','),
mime = arr[0].match(/:(.*?);/)[1],
bstr = atob(arr[1]),
n = bstr.length,
u8arr = new Uint8Array(n);
while (n--) {
u8arr[n] = bstr.charCodeAt(n);
}
return new File([u8arr], filename, {
type: mime
});
}
// Save dataURL to file
function saveFile(dataURL, filename) {
const a = $CrE('a');
a.href = dataURL;
a.download = filename;
a.click();
}
// File download function
// details looks like the detail of GM_xmlhttpRequest
// onload function will be called after file saved to disk
function downloadFile(details) {
if (!details.url || !details.name) {return false;};
// Configure request object
const requestObj = {
url: details.url,
responseType: 'blob',
onload: function(e) {
// Save file
const url = URL.createObjectURL(e.response);
saveFile(URL.createObjectURL(e.response), details.name);
URL.revokeObjectURL(url);
// onload callback
details.onload ? details.onload(e) : function() {};
}
}
if (details.onloadstart ) {requestObj.onloadstart = details.onloadstart;};
if (details.onprogress ) {requestObj.onprogress = details.onprogress;};
if (details.onerror ) {requestObj.onerror = details.onerror;};
if (details.onabort ) {requestObj.onabort = details.onabort;};
if (details.onreadystatechange) {requestObj.onreadystatechange = details.onreadystatechange;};
if (details.ontimeout ) {requestObj.ontimeout = details.ontimeout;};
// Send request
GM_xmlhttpRequest(requestObj);
}
// 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();
}
function requestText(url, callback, args=[]) {
GM_xmlhttpRequest({
method: 'GET',
url: url,
responseType: 'text',
onload: function(response) {
const text = response.responseText;
const argvs = [text].concat(args);
callback.apply(null, argvs);
}
})
}
// Get a url argument from lacation.href
// also recieve a function to deal the matched string
// returns defaultValue if name not found
// Args: {url=location.href, name, dealFunc=((a)=>{return a;}), defaultValue=null} or 'name'
function getUrlArgv(details) {
typeof(details) === 'string' && (details = {name: details});
typeof(details) === 'undefined' && (details = {});
if (!details.name) {return null;};
const url = details.url ? details.url : location.href;
const name = details.name ? details.name : '';
const dealFunc = details.dealFunc ? details.dealFunc : ((a)=>{return a;});
const defaultValue = details.defaultValue ? details.defaultValue : null;
const matcher = new RegExp('[\\?&]' + name + '=([^&#]+)');
const result = url.match(matcher);
const argv = result ? dealFunc(result[1]) : defaultValue;
return argv;
}
// 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;
}
// Get a time text like 1970-01-01 00:00:00
// if dateSpliter provided false, there will be no date part. The same for timeSpliter.
function getTime(dateSpliter='-', timeSpliter=':') {
const d = new Date();
let fulltime = ''
fulltime += dateSpliter ? fillNumber(d.getFullYear(), 4) + dateSpliter + fillNumber((d.getMonth() + 1), 2) + dateSpliter + fillNumber(d.getDate(), 2) : '';
fulltime += dateSpliter && timeSpliter ? ' ' : '';
fulltime += timeSpliter ? fillNumber(d.getHours(), 2) + timeSpliter + fillNumber(d.getMinutes(), 2) + timeSpliter + fillNumber(d.getSeconds(), 2) : '';
return fulltime;
}
// Get key-value object from text like 'key: value'/'key:value'/' key : value '
// returns: {key: value, KEY: key, VALUE: value}
function getKeyValue(text, delimiters=[':', ':', ',', '︰']) {
// Modify from https://developer.mozilla.org/zh-CN/docs/Web/JavaScript/Reference/Global_Objects/Error#examples
// Create a new object, that prototypally inherits from the Error constructor.
function SplitError(message) {
this.name = 'SplitError';
this.message = message || 'SplitError Message';
this.stack = (new Error()).stack;
}
SplitError.prototype = Object.create(Error.prototype);
SplitError.prototype.constructor = SplitError;
if (!text) {return [];};
const result = {};
let key, value;
for (let i = 0; i < text.length; i++) {
const char = text.charAt(i);
for (const delimiter of delimiters) {
if (delimiter === char) {
if (!key && !value) {
key = text.substr(0, i).trim();
value = text.substr(i+1).trim();
result[key] = value;
result.KEY = key;
result.VALUE = value;
} else {
throw new SplitError('Mutiple Delimiter in Text');
}
}
}
}
return result;
}
function htmlEncode(text) {
const span = $CrE('div');
span.innerText = text;
return span.innerHTML;
}
// Convert rgb color(e.g. 51,51,153) to hex color(e.g. '333399')
function rgbToHex(r, g, b) {return fillNumber(((r << 16) | (g << 8) | b).toString(16), 6);}
// Fill number text to certain length with '0'
function fillNumber(number, length) {
let str = String(number);
for (let i = str.length; i < length; i++) {
str = '0' + str;
}
return str;
}
// Judge whether the str is a number
function isNumeric(str, disableFloat=false) {
const result = Number(str);
return !isNaN(result) && str !== '' && (!disableFloat || result===Math.floor(result));
}
// Del a item from an array using its index. Returns the array but can NOT modify the original array directly!!
function delItem(arr, delIndex) {
arr = arr.slice(0, delIndex).concat(arr.slice(delIndex+1));
return arr;
}
// Clone(deep) an object variable
// Returns the new object
function deepclone(obj) {
if (obj === null) return null;
if (typeof(obj) !== 'object') return obj;
if (obj.constructor === Date) return new Date(obj);
if (obj.constructor === RegExp) return new RegExp(obj);
var newObj = new obj.constructor(); //保持继承的原型
for (let key in obj) {
if (obj.hasOwnProperty(key)) {
const val = obj[key];
newObj[key] = typeof val === 'object' ? deepclone(val) : val;
}
}
return newObj;
}
// Makes a function that returns a unique ID number each time
function uniqueIDMaker() {
let id = 0;
return makeID;
function makeID() {
id++;
return id;
}
}
// 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;
}
function AsyncManager() {
const AM = this;
// Ongoing xhr count
this.taskCount = 0;
// Whether generate finish events
let finishEvent = false;
Object.defineProperty(this, 'finishEvent', {
configurable: true,
enumerable: true,
get: () => (finishEvent),
set: (b) => {
finishEvent = b;
b && AM.taskCount === 0 && AM.onfinish && AM.onfinish();
}
});
// Add one task
this.add = () => (++AM.taskCount);
// Finish one task
this.finish = () => ((--AM.taskCount === 0 && AM.finishEvent && AM.onfinish && AM.onfinish(), AM.taskCount));
}
function loadinResourceCSS() {
for (const res of NMonkey_Info.resources) {
if (res.isCss) {
const css = GM_getResourceText(res.name);
css && addStyle(css);
}
}
}
function loadinFontAwesome() {
// https://cdn.jsdelivr.net/npm/@fortawesome/fontawesome-free@6.1.1/css/all.min.css
const url = 'https://cdnjs.cloudflare.com/ajax/libs/font-awesome/6.1.1/css/all.min.css';
const alts = [
'https://cdn.jsdelivr.net/npm/@fortawesome/fontawesome-free@6.1.1/css/all.min.css',
'https://bowercdn.net/c/fontAwesome-6.1.1/css/all.min.css',
];
let i = -1;
const link = $CrE('link');
link.href = url;
link.rel = 'stylesheet';
link.onerror = function() {
i++;
if (i < alts.length) {
link.href = alts[i];
} else {
alertify.error(TEXT_ALT_SCRIPT_ERROR_AJAX_FA);
}
}
document.head.appendChild(link);
}
// NMonkey By PY-DNG, 2021.07.18 - 2022.02.18, License GPL-3
// NMonkey: Provides GM_Polyfills and make your userscript compatible with non-script-manager environment
// Description:
/*
Simulates a script-manager environment("NMonkey Environment") for non-script-manager browser, load @require & @resource, provides some GM_functions(listed below), and also compatible with script-manager environment.
Provides GM_setValue, GM_getValue, GM_deleteValue, GM_listValues, GM_xmlhttpRequest, GM_openInTab, GM_setClipboard, GM_getResourceText, GM_getResourceURL, GM_addStyle, GM_addElement, GM_log, unsafeWindow(object), GM_info(object)
Also provides an object called GM_POLYFILLED which has the following properties that shows you which GM_functions are actually polyfilled.
Returns true if polyfilled is environment is ready, false for not. Don't worry, just follow the usage below.
*/
// Note: DO NOT DEFINE GM-FUNCTION-NAMES IN YOUR CODE. DO NOT DEFINE GM_POLYFILLED AS WELL.
// Note: NMonkey is an advanced version of GM_PolyFill (and BypassXB), it includes more functions than GM_PolyFill, and provides better stability and compatibility. Do NOT use NMonkey and GM_PolyFill (and BypassXB) together in one script.
// Usage:
/*
// ==UserScript==
// @name xxx
// @namespace xxx
// @version 1.0
// ...
// @require https://.../xxx.js
// @require ...
// ...
// @resource https://.../xxx
// @resource ...
// ...
// ==/UserScript==
// Use a closure to wrap your code. Make sure you have it a name.
(function YOUR_MAIN_FUNCTION() {
'use strict';
// Strict mode is optional. You can use strict mode or not as you want.
// Polyfill first. Do NOT do anything before Polyfill.
var NMonkey_Ready = NMonkey({
mainFunc: YOUR_MAIN_FUNCTION,
name: "script-storage-key, aims to separate different scripts' storage area. Use your script's @namespace value if you don't how to fill this field.",
requires: [
{
name: "", // Optional, used to display loading error messages if anything went wrong while loading this item
src: "https://.../xxx.js",
loaded: function() {return boolean_value_shows_whether_this_js_has_already_loaded;}
execmode: "'eval' for eval code in current scope or 'function' for Function(code)() in global scope or 'script' for inserting a <script> element to document.head"
},
...
],
resources: [
{
src: "https://.../xxx"
name: "@resource name. Will try to get it from @resource using this name before fetch it from src",
},
...
],
GM_info: {
// You can get GM_info object, if you provide this argument(and there is no GM_info provided by the script-manager).
// You can provide any object here, what you provide will be what you get.
// Additionally, two property of NMonkey itself will be attached to GM_info if polyfilled:
// {
// scriptHandler: "NMonkey"
// version: "NMonkey's version, it should look like '0.1'"
// }
// The following is just an example.
script: {
name: 'my first userscript for non-scriptmanager browsers!',
description: 'this script works well both in my PC and my mobile!',
version: '1.0',
released: true,
version_num: 1,
authors: ['Johnson', 'Leecy', 'War Mars']
update_history: {
'0.9': 'First beta version',
'1.0': 'Finally released!'
}
}
surprise: 'if you check GM_info.surprise and you will read this!'
// And property "scriptHandler" & "version" will be attached here
}
});
if (!NMonkey_Ready) {
// Stop executing of polyfilled environment not ready.
// Don't worry, during polyfill progress YOUR_MAIN_FUNCTION will be called twice, and on the second call the polyfilled environment will be ready.
return;
}
// Your code here...
// Make sure your code is written after NMonkey be called
if
// ...
// Just place NMonkey function code here
function NMonkey(details) {
...
}
}) ();
// Oh you want to write something here? Fine. But code you write here cannot get into the simulated script-manager-environment.
*/
function NMonkey(details) {
// Constances
const CONST = {
Text: {
Require_Load_Failed: '动态加载依赖js库失败(自动重试也都失败了),请刷新页面后再试:(\n一共尝试了{I}个备用加载源\n加载项目:{N}',
Resource_Load_Failed: '动态加载依赖resource资源失败(自动重试也都失败了),请刷新页面后再试:(\n一共尝试了{I}个备用加载源\n加载项目:{N}',
UnkownItem: '未知项目',
}
};
// Init DoLog
DoLog();
// Get argument
const mainFunc = details.mainFunc;
const name = details.name || 'default';
const requires = details.requires || [];
const resources = details.resources || [];
details.GM_info = details.GM_info || {};
details.GM_info.scriptHandler = 'NMonkey';
details.GM_info.version = '1.0';
// Run in variable-name-polifilled environment
if (InNPEnvironment()) {
// Already in polifilled environment === polyfill has alredy done, just return
return true;
}
// Polyfill functions and data
const GM_POLYFILL_KEY_STORAGE = 'GM_STORAGE_POLYFILL';
let GM_POLYFILL_storage;
const Supports = {
GetStorage: function() {
let gstorage = localStorage.getItem(GM_POLYFILL_KEY_STORAGE);
gstorage = gstorage ? JSON.parse(gstorage) : {};
let storage = gstorage[name] ? gstorage[name] : {};
return storage;
},
SaveStorage: function() {
let gstorage = localStorage.getItem(GM_POLYFILL_KEY_STORAGE);
gstorage = gstorage ? JSON.parse(gstorage) : {};
gstorage[name] = GM_POLYFILL_storage;
localStorage.setItem(GM_POLYFILL_KEY_STORAGE, JSON.stringify(gstorage));
},
};
const Provides = {
// GM_setValue
GM_setValue: function(name, value) {
GM_POLYFILL_storage = Supports.GetStorage();
name = String(name);
GM_POLYFILL_storage[name] = value;
Supports.SaveStorage();
},
// GM_getValue
GM_getValue: function(name, defaultValue) {
GM_POLYFILL_storage = Supports.GetStorage();
name = String(name);
if (GM_POLYFILL_storage.hasOwnProperty(name)) {
return GM_POLYFILL_storage[name];
} else {
return defaultValue;
}
},
// GM_deleteValue
GM_deleteValue: function(name) {
GM_POLYFILL_storage = Supports.GetStorage();
name = String(name);
if (GM_POLYFILL_storage.hasOwnProperty(name)) {
delete GM_POLYFILL_storage[name];
Supports.SaveStorage();
}
},
// GM_listValues
GM_listValues: function() {
GM_POLYFILL_storage = Supports.GetStorage();
return Object.keys(GM_POLYFILL_storage);
},
// unsafeWindow
unsafeWindow: window,
// GM_xmlhttpRequest
// not supported properties of details: synchronous binary nocache revalidate context fetch
// not supported properties of response(onload arguments[0]): finalUrl
// ---!IMPORTANT!--- DOES NOT SUPPORT CROSS-ORIGIN REQUESTS!!!!! ---!IMPORTANT!---
// details.synchronous is not supported as Tampermonkey
GM_xmlhttpRequest: function(details) {
const xhr = new XMLHttpRequest();
// open request
const openArgs = [details.method, details.url, true];
if (details.user && details.password) {
openArgs.push(details.user);
openArgs.push(details.password);
}
xhr.open.apply(xhr, openArgs);
// set headers
if (details.headers) {
for (const key of Object.keys(details.headers)) {
xhr.setRequestHeader(key, details.headers[key]);
}
}
details.cookie ? xhr.setRequestHeader('cookie', details.cookie) : function () {};
details.anonymous ? xhr.setRequestHeader('cookie', '') : function () {};
// properties
xhr.timeout = details.timeout;
xhr.responseType = details.responseType;
details.overrideMimeType ? xhr.overrideMimeType(details.overrideMimeType) : function () {};
// events
xhr.onabort = details.onabort;
xhr.onerror = details.onerror;
xhr.onloadstart = details.onloadstart;
xhr.onprogress = details.onprogress;
xhr.onreadystatechange = details.onreadystatechange;
xhr.ontimeout = details.ontimeout;
xhr.onload = function (e) {
const response = {
readyState: xhr.readyState,
status: xhr.status,
statusText: xhr.statusText,
responseHeaders: xhr.getAllResponseHeaders(),
response: xhr.response
};
(details.responseType === '' || details.responseType === 'text') ? (response.responseText = xhr.responseText) : function () {};
(details.responseType === '' || details.responseType === 'document') ? (response.responseXML = xhr.responseXML) : function () {};
details.onload(response);
}
// send request
details.data ? xhr.send(details.data) : xhr.send();
return {
abort: xhr.abort
};
},
// NOTE: options(arg2) is NOT SUPPORTED! if provided, then will just be skipped.
GM_openInTab: function(url) {
window.open(url);
},
// NOTE: needs to be called in an event handler function, and info(arg2) is NOT SUPPORTED!
GM_setClipboard: function(text) {
// Create a new textarea for copying
const newInput = document.createElement('textarea');
document.body.appendChild(newInput);
newInput.value = text;
newInput.select();
document.execCommand('copy');
document.body.removeChild(newInput);
},
GM_getResourceText: function(name) {
const _get = typeof(GM_getResourceText) === 'function' ? GM_getResourceText : () => (null);
let text = _get(name);
if (text) {return text;}
for (const resource of resources) {
if (resource.name === name) {
return resource.content ? resource.content : null;
}
}
return null;
},
GM_getResourceURL: function(name) {
const _get = typeof(GM_getResourceURL) === 'function' ? GM_getResourceURL : () => (null);
let url = _get(name);
if (url) {return url;}
for (const resource of resources) {
if (resource.name === name) {
return resource.src ? btoa(resource.src) : null;
}
}
return null;
},
GM_addStyle: function(css) {
const style = document.createElement('style');
style.innerHTML = css;
document.head.appendChild(style);
},
GM_addElement: function() {
let parent_node, tag_name, attributes;
const head_elements = ['title', 'base', 'link', 'style', 'meta', 'script', 'noscript'/*, 'template'*/];
if (arguments.length === 2) {
tag_name = arguments[0];
attributes = arguments[1];
parent_node = head_elements.includes(tag_name.toLowerCase()) ? document.head : document.body;
} else if (arguments.length === 3) {
parent_node = arguments[0];
tag_name = arguments[1];
attributes = arguments[2];
}
const element = document.createElement(tag_name);
for (const [prop, value] of Object.entries(attributes)) {
element[prop] = value;
}
parent_node.appendChild(element);
},
GM_log: function() {
const args = [];
for (let i = 0; i < arguments.length; i++) {
args[i] = arguments[i];
}
console.log.apply(null, args);
},
GM_info: details.GM_info,
GM: {info: details.GM_info}
};
const _GM_POLYFILLED = Provides.GM_POLYFILLED = {};
for (const pname of Object.keys(Provides)) {
_GM_POLYFILLED[pname] = true;
}
// Not in polifilled environment, then polyfill functions and create & move into the environment
// Bypass xbrowser's useless GM_functions
bypassXB();
// Create & move into polifilled environment
ExecInNPEnv();
return false;
// Bypass xbrowser's useless GM_functions
function bypassXB() {
if (typeof(mbrowser) === 'object' || (typeof(GM_info) === 'object' && GM_info.scriptHandler === 'XMonkey')) {
// Useless functions in XMonkey 1.0
const GM_funcs = [
'unsafeWindow',
'GM_getValue',
'GM_setValue',
'GM_listValues',
'GM_deleteValue',
//'GM_xmlhttpRequest',
];
for (const GM_func of GM_funcs) {
window[GM_func] = undefined;
eval('typeof({F}) === "function" && ({F} = Provides.{F});'.replaceAll('{F}', GM_func));
}
// Delete dirty data saved by these stupid functions before
for (let i = 0; i < localStorage.length; i++) {
const key = localStorage.key(i);
const value = localStorage.getItem(key);
value === '[object Object]' && localStorage.removeItem(key);
}
}
}
// Check if already in name-predefined environment
// I think there won't be anyone else wants to use this fxxking variable name...
function InNPEnvironment() {
return (typeof(GM_POLYFILLED) === 'object' && GM_POLYFILLED !== null && GM_POLYFILLED !== window.GM_POLYFILLED) ? true : false;
}
function ExecInNPEnv() {
const NG = new NameGenerator();
// Init names
const tnames = ['context', 'fapply', 'CDATA', 'uneval', 'define', 'module', 'exports', 'window', 'globalThis', 'console', 'cloneInto', 'exportFunction', 'createObjectIn', 'GM', 'GM_info'];
const pnames = Object.keys(Provides);
const fnames = tnames.slice();
const argvlist = [];
const argvs = [];
// Add provides
for (const pname of pnames) {
!fnames.includes(pname) && fnames.push(pname);
}
// Add grants
if (typeof(GM_info) === 'object' && GM_info.script && GM_info.script.grant) {
for (const gname of GM_info.script.grant) {
!fnames.includes(gname) && fnames.push(gname);
}
}
// Make name code
for (let i = 0; i < fnames.length; i++) {
const fname = fnames[i];
const exist = eval('typeof ' + fname + ' !== "undefined"') && fname !== 'GM_POLYFILLED';
argvlist[i] = exist ? fname : (Provides.hasOwnProperty(fname) ? 'Provides.'+fname : '');
argvs[i] = exist ? eval(fname) : (Provides.hasOwnProperty(fname) ? Provides[name] : undefined);
pnames.includes(fname) && (_GM_POLYFILLED[fname] = !exist);
}
// Load all @require and @resource
loadRequires(requires, resources, function(requires, resources) {
// Join requirecode
let requirecode = '';
for (const require of requires) {
const mode = require.execmode ? require.execmode : 'eval';
const content = require.content;
if (!content) {continue;}
switch(mode) {
case 'eval':
requirecode += content + '\n';
break;
case 'function': {
const func = Function.apply(null, fnames.concat(content));
func.apply(null, argvs);
break;
}
case 'script': {
const s = document.createElement('script');
s.innerHTML = content;
document.head.appendChild(s);
break;
}
}
}
// Make final code & eval
const varnames = ['NG', 'tnames', 'pnames', 'fnames', 'argvist', 'argvs', 'code', 'finalcode', 'wrapper', 'ExecInNPEnv', 'GM_POLYFILL_KEY_STORAGE', 'GM_POLYFILL_storage', 'InNPEnvironment', 'NameGenerator', 'LocalCDN', 'loadRequires', 'requestText', 'Provides', 'Supports', 'bypassXB', 'details', 'mainFunc', 'name', 'requires', 'resources', '_GM_POLYFILLED', 'CONST', 'NMonkey', 'polyfill_status'];
const code = requirecode + 'let ' + varnames.join(', ') + ';\n(' + mainFunc.toString() + ') ();';
const wrapper = Function.apply(null, fnames.concat(code));
const finalcode = '(' + wrapper.toString() + ').apply(this, [' + argvlist.join(', ') + ']);';
eval(finalcode);
});
function NameGenerator() {
const NG = this;
const letters = 'abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ';
let index = [0];
NG.generate = function() {
const chars = [];
indexIncrease();
for (let i = 0; i < index.length; i++) {
chars[i] = letters.charAt(index[i]);
}
return chars.join('');
}
NG.randtext = function(len=32) {
const chars = [];
for (let i = 0; i < len; i++) {
chars[i] = letters[randint(0, letter.length-1)];
}
return chars.join('');
}
function indexIncrease(i=0) {
index[i] === undefined && (index[i] = -1);
++index[i] >= letters.length && (index[i] = 0, indexIncrease(i+1));
}
function randint(min, max) {
return Math.floor(Math.random() * (max - min + 1)) + min;
}
}
}
// Load all @require and @resource for non-GM/TM environments (such as Alook javascript extension)
// Requirements: function AsyncManager(){...}, function LocalCDN(){...}
function loadRequires(requires, resoures, callback, args=[]) {
// LocalCDN
const LCDN = new LocalCDN();
// AsyncManager
const AM = new AsyncManager();
AM.onfinish = function() {
callback.apply(null, [requires, resoures].concat(args));
}
// Load js
for (const js of requires) {
!js.loaded() && loadinJs(js);
}
// Load resource
for (const resource of resoures) {
loadinResource(resource);
}
AM.finishEvent = true;
function loadinJs(js) {
AM.add();
const srclist = js.srcset ? LCDN.sort(js.srcset).srclist : [];
let i = -1;
LCDN.get(js.src, onload, [], onfail);
function onload(content) {
js.content = content;
AM.finish();
}
function onfail() {
i++;
if (i < srclist.length) {
LCDN.get(srclist[i], onload, [], onfail);
} else {
alert(CONST.Text.Require_Load_Failed.replace('{I}', i.toString()).replace('{N}', js.name ? js.name : CONST.Text.UnkownItem));
}
}
}
function loadinResource(resource) {
let content;
if (typeof GM_getResourceText === 'function' && (content = GM_getResourceText(resource.name))) {
resource.content = content;
} else {
AM.add();
let i = -1;
LCDN.get(resource.src, onload, [], onfail);
function onload(content) {
resource.content = content;
AM.finish();
}
function onfail(content) {
i++;
if (resource.srcset && i < resource.srcset.length) {
LCDN.get(resource.srcset[i], onload, [], onfail);
} else {
debugger;
alert(CONST.Text.Resource_Load_Failed.replace('{I}', i.toString()).replace('{N}', js.name ? js.name : CONST.Text.UnkownItem));
}
}
}
}
}
// Loads web resources and saves them to GM-storage
// Tries to load web resources from GM-storage in subsequent calls
// Updates resources every $(this.expire) hours, or use $(this.refresh) function to update all resources instantly
// Dependencies: GM_getValue(), GM_setValue(), requestText(), AsyncManager(), KEY_LOCALCDN
function LocalCDN() {
const LC = this;
const _GM_getValue = typeof(GM_getValue) === 'function' ? GM_getValue : Provides.GM_getValue;
const _GM_setValue = typeof(GM_setValue) === 'function' ? GM_setValue : Provides.GM_setValue;
const KEY_LOCALCDN = 'LOCAL-CDN';
const KEY_LOCALCDN_VERSION = 'version';
const VALUE_LOCALCDN_VERSION = '0.3';
// Default expire time (by hour)
LC.expire = 72;
// Try to get resource content from loaclCDN first, if failed/timeout, request from web && save to LocalCDN
// Accepts callback only: onload & onfail(optional)
// Returns true if got from LocalCDN, false if got from web
LC.get = function(url, onload, args=[], onfail=function(){}) {
const CDN = _GM_getValue(KEY_LOCALCDN, {});
const resource = CDN[url];
const time = (new Date()).getTime();
if (resource && resource.content !== null && !expired(time, resource.time)) {
onload.apply(null, [resource.content].concat(args));
return true;
} else {
LC.request(url, _onload, [], onfail);
return false;
}
function _onload(content) {
onload.apply(null, [content].concat(args));
}
}
// Generate resource obj and set to CDN[url]
// Returns resource obj
// Provide content means load success, provide null as content means load failed
LC.set = function(url, content) {
const CDN = _GM_getValue(KEY_LOCALCDN, {});
const time = (new Date()).getTime();
const resource = {
url: url,
time: time,
content: content,
success: content !== null ? (CDN[url] ? CDN[url].success + 1 : 1) : (CDN[url] ? CDN[url].success : 0),
fail: content === null ? (CDN[url] ? CDN[url].fail + 1 : 1) : (CDN[url] ? CDN[url].fail : 0),
};
CDN[url] = resource;
_GM_setValue(KEY_LOCALCDN, CDN);
return resource;
}
// Delete one resource from LocalCDN
LC.delete = function(url) {
const CDN = _GM_getValue(KEY_LOCALCDN, {});
if (!CDN[url]) {
return false;
} else {
delete CDN[url];
_GM_setValue(KEY_LOCALCDN, CDN);
return true;
}
}
// Delete all resources in LocalCDN
LC.clear = function() {
_GM_setValue(KEY_LOCALCDN, {});
upgradeConfig();
}
// List all resource saved in LocalCDN
LC.list = function() {
const CDN = _GM_getValue(KEY_LOCALCDN, {});
const urls = LC.listurls();
return LC.listurls().map((url) => (CDN[url]));
}
// List all resource's url saved in LocalCDN
LC.listurls = function() {
return Object.keys(_GM_getValue(KEY_LOCALCDN, {})).filter((url) => (url !== KEY_LOCALCDN_VERSION));
}
// Request content from web and save it to CDN[url]
// Accepts callbacks only: onload & onfail(optional)
LC.request = function(url, onload, args=[], onfail=function(){}) {
const CDN = _GM_getValue(KEY_LOCALCDN, {});
requestText(url, _onload, [], _onfail);
function _onload(content) {
LC.set(url, content);
onload.apply(null, [content].concat(args));
}
function _onfail() {
LC.set(url, null);
onfail();
}
}
// Re-request all resources in CDN instantly, ignoring LC.expire
LC.refresh = function(callback, args=[]) {
const urls = LC.listurls();
const AM = new AsyncManager();
AM.onfinish = function() {
callback.apply(null, [].concat(args))
};
for (const url of urls) {
AM.add();
LC.request(url, function() {
AM.finish();
});
}
AM.finishEvent = true;
}
// Sort src && srcset, to get a best request sorting
LC.sort = function(srcset) {
const CDN = _GM_getValue(KEY_LOCALCDN, {});
const result = {srclist: [], lists: []};
const lists = result.lists;
const srclist = result.srclist;
const suc_rec = lists[0] = []; // Recent successes take second (not expired yet)
const suc_old = lists[1] = []; // Old successes take third
const fails = lists[2] = []; // Fails & unused take the last place
const time = (new Date()).getTime();
// Make lists
for (const s of srcset) {
const resource = CDN[s];
if (resource && resource.content !== null) {
if (!expired(resource.time, time)) {
suc_rec.push(s);
} else {
suc_old.push(s);
}
} else {
fails.push(s);
}
}
// Sort lists
// Recently successed: Choose most recent ones
suc_rec.sort((res1, res2) => (res2.time - res1.time));
// Successed long ago or failed: Sort by success rate & tried time
[suc_old, fails].forEach((arr) => (arr.sort(sorting)));
// Push all resources into seclist
[suc_rec, suc_old, fails].forEach((arr) => (arr.forEach((res) => (srclist.push(res)))));
DoLog(['LocalCDN: sorted', result]);
return result;
function sorting(res1, res2) {
const sucRate1 = (res1.success+1) / (res1.fail+1);
const sucRate2 = (res2.success+1) / (res2.fail+1);
if (sucRate1 !== sucRate2) {
// Success rate: high to low
return sucRate2 - sucRate1;
} else {
// Tried time: less to more
// Less tried time means newer added source
return (res1.success+res1.fail) - (res2.success+res2.fail);
}
}
}
function upgradeConfig() {
const CDN = _GM_getValue(KEY_LOCALCDN, {});
switch(CDN[KEY_LOCALCDN_VERSION]) {
case undefined:
init();
break;
case '0.1':
v01_To_v02();
logUpgrade();
break;
case '0.2':
v01_To_v02();
v02_To_v03();
logUpgrade();
break;
case VALUE_LOCALCDN_VERSION:
DoLog('LocalCDN is in latest version.');
break;
default:
DoLog(LogLevel.Error, 'LocalCDN.upgradeConfig: Invalid config version({V}) for LocalCDN. '.replace('{V}', CDN[KEY_LOCALCDN_VERSION]));
}
CDN[KEY_LOCALCDN_VERSION] = VALUE_LOCALCDN_VERSION;
_GM_setValue(KEY_LOCALCDN, CDN);
function logUpgrade() {
DoLog(LogLevel.Success, 'LocalCDN successfully upgraded From v{V1} to v{V2}. '.replaceAll('{V1}', CDN[KEY_LOCALCDN_VERSION]).replaceAll('{V2}', VALUE_LOCALCDN_VERSION));
}
function init() {
// Nothing to do here
}
function v01_To_v02() {
const urls = LC.listurls();
for (const url of urls) {
if (url === KEY_LOCALCDN_VERSION) {continue;}
CDN[url] = {
url: url,
time: 0,
content: CDN[url]
};
}
}
function v02_To_v03() {
const urls = LC.listurls();
for (const url of urls) {
CDN[url].success = CDN[url].fail = 0;
}
}
}
function clearExpired() {
const resources = LC.list();
const time = (new Date()).getTime();
for (const resource of resources) {
expired(resource.time, time) && LC.delete(resource.url);
}
}
function expired(t1, t2) {
return (t1 - t2) > (LC.expire * 60 * 60 * 1000);
}
upgradeConfig();
clearExpired();
}
function requestText(url, callback, args=[], onfail=function(){}) {
const req = typeof(GM_xmlhttpRequest) === 'function' ? GM_xmlhttpRequest : Provides.GM_xmlhttpRequest;
req({
method: 'GET',
url: url,
responseType: 'text',
timeout: 45*1000,
onload: function(response) {
const text = response.responseText;
const argvs = [text].concat(args);
callback.apply(null, argvs);
},
onerror: onfail,
ontimeout: onfail,
onabort: onfail,
})
}
function AsyncManager() {
const AM = this;
// Ongoing xhr count
this.taskCount = 0;
// Whether generate finish events
let finishEvent = false;
Object.defineProperty(this, 'finishEvent', {
configurable: true,
enumerable: true,
get: () => (finishEvent),
set: (b) => {
finishEvent = b;
b && AM.taskCount === 0 && AM.onfinish && AM.onfinish();
}
});
// Add one task
this.add = () => (++AM.taskCount);
// Finish one task
this.finish = () => ((--AM.taskCount === 0 && AM.finishEvent && AM.onfinish && AM.onfinish(), AM.taskCount));
}
// Arguments: level=LogLevel.Info, logContent, asObject=false
// Needs one call "DoLog();" to get it initialized before using it!
function DoLog() {
const win = typeof(unsafeWindow) !== 'undefined' ? unsafeWindow : window;
// Global log levels set
win.LogLevel = {
None: 0,
Error: 1,
Success: 2,
Warning: 3,
Info: 4,
}
win.LogLevelMap = {};
win.LogLevelMap[LogLevel.None] = {prefix: '' , color: 'color:#ffffff'}
win.LogLevelMap[LogLevel.Error] = {prefix: '[Error]' , color: 'color:#ff0000'}
win.LogLevelMap[LogLevel.Success] = {prefix: '[Success]' , color: 'color:#00aa00'}
win.LogLevelMap[LogLevel.Warning] = {prefix: '[Warning]' , color: 'color:#ffa500'}
win.LogLevelMap[LogLevel.Info] = {prefix: '[Info]' , color: 'color:#888888'}
win.LogLevelMap[LogLevel.Elements] = {prefix: '[Elements]', color: 'color:#000000'}
// Current log level
DoLog.logLevel = win.isPY_DNG ? LogLevel.Info : LogLevel.Warning; // Info Warning Success Error
// Log counter
DoLog.logCount === undefined && (DoLog.logCount = 0);
if (++DoLog.logCount > 512) {
console.clear();
DoLog.logCount = 0;
}
// Get args
let level, logContent, asObject;
switch (arguments.length) {
case 1:
level = LogLevel.Info;
logContent = arguments[0];
asObject = false;
break;
case 2:
level = arguments[0];
logContent = arguments[1];
asObject = false;
break;
case 3:
level = arguments[0];
logContent = arguments[1];
asObject = arguments[2];
break;
default:
level = LogLevel.Info;
logContent = 'DoLog initialized.';
asObject = false;
break;
}
// Log when log level permits
if (level <= DoLog.logLevel) {
let msg = '%c' + LogLevelMap[level].prefix;
let subst = LogLevelMap[level].color;
if (asObject) {
msg += ' %o';
} else {
switch(typeof(logContent)) {
case 'string': msg += ' %s'; break;
case 'number': msg += ' %d'; break;
case 'object': msg += ' %o'; break;
}
}
console.log(msg, subst, logContent);
}
}
}
// Polyfill alert
function polyfillAlert() {
if (typeof(GM_POLYFILLED) !== 'object') {return false;}
if (GM_POLYFILLED.GM_setValue) {
alertify.notify(TEXT_ALT_POLYFILL);
}
}
// Polyfill String.prototype.replaceAll
// replaceValue does NOT support regexp match groups($1, $2, etc.)
function polyfill_replaceAll() {
String.prototype.replaceAll = String.prototype.replaceAll ? String.prototype.replaceAll : PF_replaceAll;
function PF_replaceAll(searchValue, replaceValue) {
const str = String(this);
if (searchValue instanceof RegExp) {
const global = RegExp(searchValue, 'g');
if (/\$/.test(replaceValue)) {console.error('Error: Polyfilled String.protopype.replaceAll does support regexp groups');};
return str.replace(global, replaceValue);
} else {
return str.split(searchValue).join(replaceValue);
}
}
}
// Append a style text to document(<head>) with a <style> element
function addStyle(css, id) {
const style = $CrE("style");
id && (style.id = id);
style.textContent = css;
for (const elm of $All('#'+id)) {
elm.parentElement && elm.parentElement.removeChild(elm);
}
document.head.appendChild(style);
}
// 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);
}
})();