// ==UserScript==
// @name 网页滚动条美化V2
// @namespace http://tampermonkey.net/
// @version 0.2.2
// @description 网页美化
// @author Eliauk
// @match https://*/*
// @match http://*/*
// @icon 
// @license GPL-3.0
// @grant GM_addStyle
// @grant GM_getValue
// @grant GM_setValue
// @grant GM_getResourceText
// @grant GM_notification
// @resource ArcoDesignStyle https://gitee.com/lzyws739307453/scrollbar-beautify/raw/master/arcodesign.min.css
// @grant GM_registerMenuCommand
// @grant GM_unregisterMenuCommand
// @run-at document-start
// ==/UserScript==
const isdebug = false; // 调试日志用
const isLocalDebug = false; // 加载本地资源用
const debug = isdebug ? console.log.bind(console) : () => {};
let nodes, scrollbarX, scrollbarY, ruleTitles, ruleConfig;
let data = {};
const DefaultConfig = {
scrollbar: {
height: 10,
width: 10,
radius: 10,
background: "linear-gradient(45deg, #000000 0%, #ff0000 100%) content-box;",
color: {
default: "#f2f3f5",
solid: "#f2f3f5",
gradient: {
direction: 45,
length: 1,
steps: {
"0": "#000000",
"1": "#ff0000"
}
}
}
},
rule: {
default: ["^((ht|f)tps?:\\/\\/)?[\\w-]+(\\.[\\w-]+)+"],
titles: ["^https?:\\/\\/(.+\\.)?bilibili\\.[a-z]{2,6}/?"],
config: {
"^((ht|f)tps?:\\/\\/)?[\\w-]+(\\.[\\w-]+)+": {
enable: true,
style: "::-webkit-scrollbar{width:10px;height:10px;}::-webkit-scrollbar-thumb{border-radius:10px;border:2px solid transparent;background:linear-gradient(45deg, #000000 0%, #ff0000 100%) content-box;}::-webkit-scrollbar-track{box-shadow:none;}html{overflow-y:overlay;}"
},
"^https?:\\/\\/(.+\\.)?bilibili\\.[a-z]{2,6}/?": {
enable: true,
style: "html{overflow-y:inherit;}body{overflow-y:overlay;}"
}
}
}
}
// 增加了一个css的缓冲层,保证多次CSS操作不导致页面卡顿,减少重排的次数
class FlushDomFragment {
constructor() {
this.init()
}
init() {
this.length = 0
this.fragmentHead = document.createDocumentFragment();
this.fragmentBody = document.createDocumentFragment();
this.fragmentDOM = document.createDocumentFragment();
this.reloadList = [] // 所有需要动态刷新的都在这里面
}
_removeReload() {
this.reloadList.map(selector => {
safeRemove(selector)
})
}
_singleInsert(node, toDom, checkDom) {
if (!node) return;
const {
dataset: {
xclass: selector = ''
} = {},
tagName = ''
} = node
if (tagName.toLowerCase() === 'style') {
if (selector) {
// 已经存在,不用添加
if (checkDom.querySelector(selector)) return
} else {
console.error('出现没有样式的节点', node)
}
}
// 添加
toDom.appendChild(node)
this.length += 1
debug('增加节点', node)
debug('长度变化', this.length)
}
// 如果CSS已经存在了,那么就不再添加了
_dropMultiCSS(fragment) {
const newFrag = document.createDocumentFragment();
[...fragment.childNodes].map(node => {
this._singleInsert(node, newFrag, document)
})
return newFrag
}
flush() {
// 有数据, 才进行flush, 否则没有必要
if (this.length <= 0) return
this._removeReload()
// MARK 保证线程安全, 将现有数据暂存, 然后生成新的节点, 避免其他js插入后丢失
const curBodyFrag = this.fragmentBody
const curHeadFrag = this.fragmentHead
const curDomFrag = this.fragmentDOM
this.init()
// 数据清除
const newBodyFrag = this._dropMultiCSS(curBodyFrag)
const newHeadFrag = this._dropMultiCSS(curHeadFrag)
const newDomFrag = this._dropMultiCSS(curDomFrag)
debug('数据flush1', newBodyFrag.children.length)
debug('数据flush2', newHeadFrag.children.length)
debug('数据flush3', newDomFrag.children.length)
safeWaitFunc("body", () => {
document.body.appendChild(newBodyFrag)
})
safeWaitFunc("head", () => {
document.head.appendChild(newHeadFrag)
})
document.insertBefore(newDomFrag, document.documentElement)
}
appendChild(node, to = 'head', config = {
isReload: false
}) {
return this.insert(node, to, config)
}
/**
*
* @param { 子节点 } node
* @param { 父节点 } to
* @param { 配置信息, isReload: 是否加入定时器刷新重新加载 } config
* @returns
*/
insert(node, to = 'head', config = {
isReload: false
}) {
if ('body' === to) {
this._singleInsert(node, this.fragmentBody, this.fragmentBody)
} else if ('head' === to) {
this._singleInsert(node, this.fragmentHead, this.fragmentHead)
} else if ('DOM' === to) {
this._singleInsert(node, this.fragmentDOM, this.fragmentDOM)
} else {
console.error('不支持的节点操作')
return
}
const {
isReload
} = config
if (isReload) {
const {
classList = []
} = node
if (classList.length) {
this.reloadList.push('.' + [...(classList || [])].join('.'))
} else {
console.error('异常的reload参数, 没有classList', node)
}
}
return this
}
}
function $one(selector, element = document) {
return element.querySelector(selector)
}
function $all(selector, element = document) {
return element.querySelectorAll(selector)
}
function parseBoolean(val) {
if (["yes", "y", "true", "1", "on"].indexOf(String(val).toLowerCase()) !== -1) return true;
if (["no", "n", "false", "0", "off"].indexOf(String(val).toLowerCase()) !== -1) return false;
return Boolean(val);
}
ruleConfigSwitch = function (e) {
const enable = parseBoolean(e.getAttribute("aria-checked"));
const data_v = e.dataset.v;
const titleComp = $one(`.arco-collapse-item-header-title[data-v="${data_v}"]`, ruleConfig);
const title = titleComp.innerText;
if (!data.rule.config[title]) {
data.rule.config[title] = {};
}
debug('ruleConfigSwitch', data.rule.config[title]);
debug('ruleConfigSwitch', data);
data.rule.config[title].enable = enable;
}
// 渐变长度更改事件
gradientLenghtChange = function (e) {
// debug("gradientLenghtChange");
// debug(e.value);
nodes.color.gradient.step
nodes.color.gradient.step.comp.setAttribute("max", `${e.value}`);
// 重置
resetGradientStep("0");
data.scrollbar.color.gradient.steps = {};
data.scrollbar.color.gradient.length = e.value;
resetGradientColor(data.scrollbar.color.default);
resetPreviewBackground(data.scrollbar.color.default);
redrawSliderStyle(nodes.color.gradient.step.comp);
}
// 纯色、渐变选择按钮事件
colorSelect = function (e) {
// debug(e);
const select = $one("#eliauk-container .arco-radio-checked .arco-radio-target", e).value;
if ('GradientColor' === select) {
$one("#solid-color-page").style.display = 'none';
$one("#gradient-color-page").style.display = 'block';
reloadGradientPage();
} else {
$one("#gradient-color-page").style.display = 'none';
$one("#solid-color-page").style.display = 'block';
reloadSolidPage();
}
};
// 滑动输入条绘制
function redrawSliderStyle(ele) {
let total = ele.max - ele.min;
let effset = Math.round((ele.value * 100) / total);
ele.style.background =
`linear-gradient(to right, rgb(var(--primary-6)) ${effset}%, var(--color-fill-3) ${effset}%)`;
};
function resetGradientStep(value) {
const step = nodes.color.gradient.step.comp;
step.value = value;
let total = step.max - step.min;
let effset = Math.round((value * 100) / total);
nodes.color.gradient.step.text.value = `${effset}%`;
redrawSliderStyle(nodes.color.gradient.step.comp);
reloadGradientPreviewBackground();
}
function resetGradientColor(value) {
nodes.color.gradient.color.comp.value = value;
nodes.color.gradient.color.text.value = value;
}
function resetPreviewBackground(background) {
data.scrollbar.background = background;
$all("#eliauk-container .context div").forEach(ele => {
ele.style.background = background;
})
}
function resetPreviewBasicSettings(width, height, radius) {
$all("#eliauk-container .context div").forEach(ele => {
ele.style.borderRadius = radius;
})
scrollbarY.style.width = `${width}px`;
scrollbarX.style.height = `${height}px`;
}
function reloadBasicSettingsPage() {
const {
width, height, radius
} = data.scrollbar;
nodes.scrollbar.width.comp.value = width;
nodes.scrollbar.width.text.value = width;
nodes.scrollbar.height.comp.value = height;
nodes.scrollbar.height.text.value = height;
nodes.scrollbar.radius.comp.value = radius;
nodes.scrollbar.radius.text.value = radius;
resetPreviewBasicSettings(width, height, radius);
}
function reloadSolidPage() {
nodes.color.solid.color.comp.value = data.scrollbar.color.solid;
nodes.color.solid.color.text.value = data.scrollbar.color.solid;
resetPreviewBackground(data.scrollbar.color.solid);
}
function reloadGradientDirection() {
nodes.color.gradient.direction.comp.value = data.scrollbar.color.gradient.direction;
nodes.color.gradient.direction.text.value = `${data.scrollbar.color.gradient.direction}deg`;
}
// 渐变背景修改预览
function reloadGradientPreviewBackground() {
const gradientDirection = nodes.color.gradient.direction.text.value;
const keys = Object.keys(data.scrollbar.color.gradient.steps);
// .sort((a, b) => a.match(/^(\d+)%$/)[1] - b.match(/^(\d+)%$/)[1])
const style = `linear-gradient(${gradientDirection}, ${keys.map((key) => {
let total = data.scrollbar.color.gradient.length;
let effset = Math.round((key * 100) / total);
return `${data.scrollbar.color.gradient.steps[key]} ${effset}%`
}).join(", ")})`
// debug(style);
resetPreviewBackground(style);
}
function reloadRules() {
ruleTitles.value = data.rule.titles.join(";\n");
}
function reloadGradientPage() {
nodes.color.gradient.length.text.value = data.scrollbar.color.gradient.length;
nodes.color.gradient.step.comp.max = data.scrollbar.color.gradient.length;
nodes.color.gradient.direction.comp.value = data.scrollbar.color.gradient.direction;
nodes.color.gradient.direction.text.value = `${data.scrollbar.color.gradient.direction}deg`;
redrawSliderStyle(nodes.color.gradient.direction.comp);
const steps = data.scrollbar.color.gradient.steps;
const keys = Object.keys(steps);
if (keys && keys.length > 0) {
resetGradientStep(keys[0]);
resetGradientColor(steps[keys[0]]);
} else {
resetGradientStep("0");
resetGradientColor(data.scrollbar.color.default);
}
};
/**
*
* @param { css选择器 } selector
* @param { 是否绑定消失动画 } withAni
*/
function safeRemove(selector, withAni = false) {
safeFunction(() => {
let removeNodes = document.querySelectorAll(selector);
for (let i = 0; i < removeNodes.length; i++) {
aniRemove(removeNodes[i], withAni)
}
})
}
/**
*
* @param { 节点 } node
* @param { 是否绑定消失动画 } withAni
*/
function aniRemove(node, withAni) {
if (withAni) {
node.classList.add('aniDelete')
setTimeout(() => {
node.remove();
}, 200)
} else {
node.remove();
}
}
function safeFunction(func, failCb) {
try {
func();
} catch (e) {
failCb && failCb(e)
}
}
function retryInterval(callback, period = 50, now = false, count = -1) {
if (now && count-- != 0) {
if (callback()) return;
}
const inter = setInterval(() => {
if (count-- === 0) {
return clearInterval(inter);
}
callback() && clearInterval(inter);
}, period);
}
function safeWaitFunc(selector, callbackFunc, time = 60, notClear) {
notClear = notClear || false;
let doClear = !notClear;
retryInterval(() => {
if ((typeof (selector) === "string" && document.querySelector(selector) != null)) {
callbackFunc(document.querySelector(selector));
if (doClear) return true;
} else if (typeof (selector) === "function" && (selector() != null || (selector() || []).length > 0)) {
callbackFunc(selector()[0]);
if (doClear) return true;
}
}, time, true);
}
/**
*
* @param {
* 回调函数, 需要返回是否, True: 结束、False: 相当于定时器
* callback return:
* true = 倒计时
* false = 计时器
* } callback
* @param { 周期,如: 200ms } period
* @param { 立即执行 } now
* @param { 次数, -1: Infinity } count
*/
function createStyleElement(css, className = '', type = "text/css") {
const element = document.createElement("style");
if (className) {
element.className = className
const xclass = '.' + className.split(' ').join('.')
element.dataset.xclass = xclass
}
element.setAttribute("type", type);
element.appendChild(document.createTextNode(css))
return element
};
(function () {
"use strict";
const hostname = window.location.href;
const CURRENT_LIST = "CurrentList";
const ELCONFIG = "ElConfig";
let ElConfig = {};
reloadAllConfig();
const globalStyle = `
#eliauk-container .arco-input-wrapper .arco-input.arco-input-size-mini {
height: 20px;
}
#eliauk-container a, #eliauk-container abbr, #eliauk-container address, #eliauk-container blockquote, #eliauk-container caption, #eliauk-container cite, #eliauk-container code, #eliauk-container dd, #eliauk-container del, #eliauk-container dfn, #eliauk-container dl, #eliauk-container dt, #eliauk-container em, #eliauk-container fieldset, #eliauk-container form, #eliauk-container h1, #eliauk-container h2, #eliauk-container h3, #eliauk-container h4, #eliauk-container h5, #eliauk-container h6, #eliauk-container iframe, #eliauk-container img, #eliauk-container ins, #eliauk-container label, #eliauk-container legend, #eliauk-container li, #eliauk-container object, #eliauk-container ol, #eliauk-container p, #eliauk-container pre, #eliauk-container q, #eliauk-container small, #eliauk-container strong, #eliauk-container sub, #eliauk-container sup, #eliauk-container table, #eliauk-container tbody, #eliauk-container td, #eliauk-container tfoot, #eliauk-container th, #eliauk-container thead, #eliauk-container tr, #eliauk-container ul {
border: 0;
margin: 0;
padding: 0;
}
#eliauk-container {
font-family: HarmonyOS Sans SC, Inter, -apple-system, BlinkMacSystemFont, PingFang SC, Hiragino Sans GB, noto sans, Microsoft YaHei, Helvetica Neue, Helvetica, Arial, sans-serif;
display: none;
position: fixed;
top: 3.9vw;
right: 12.5vw;
z-index: 999999;
box-shadow: -2px 2px 5px rgb(0 0 0 / 30%);
border-radius: var(--border-radius-medium);
}
/* 对话框 */
#eliauk-container .arco-modal {
position: static;
display: block;
width: 360px;
}
#eliauk-container .arco-modal-header {
border-bottom: none;
}
#eliauk-container .arco-modal-body {
padding-top: 0px;
overflow: visible;
}
#eliauk-container .arco-modal-body .arco-divider-horizontal.arco-divider-with-text:first-child {
margin-top: 0;
}
#eliauk-container .arco-form-item {
margin-bottom: 10px;
}
/* 折叠页相关 */
#eliauk-container .page-collapse {
display: block;
box-sizing: border-box;
width: 100%;
height: 100%;
}
#eliauk-container .page-one,
#eliauk-container .page-two {
padding: 0 2px 0 12px;
overflow-y: scroll;
}
#eliauk-container .page {
overflow: hidden;
}
#eliauk-container .page,
#eliauk-container .page-one,
#eliauk-container .page-two {
height: 435px;
opacity: 1;
transition: all .3s;
}
#eliauk-container .page-collapse-btn svg {
transition: all .3s;
}
#eliauk-container .page-collapse-btn-collapse svg {
-webkit-transform: rotate(180deg);
-moz-transform: rotate(180deg);
-o-transform: rotate(180deg);
-ms-transform: rotate(180deg);
transform: rotate(180deg);
}
#eliauk-container .page-collapse+div,
#eliauk-container .page-collapse.page-collapse-collapse {
height: 0;
opacity: 0;
}
#eliauk-container .page-collapse.page-collapse-collapse+div {
height: 435px;
opacity: 1;
}
#eliauk-container .page-collapse-btn {
position: relative;
left: 0;
transition: left .2s;
}
/* 滑动输入框相关 */
#eliauk-container .eliauk-slider-context {
min-height: 24px;
display: inline-flex;
justify-content: center;
align-items: center;
}
#eliauk-container .eliauk-slider {
-webkit-appearance: none;
appearance: none;
outline: none;
/* 避免点选会有蓝线或虚线 */
width: 100%;
height: 2px;
border-radius: 2px;
background: linear-gradient(to right,
rgb(var(--primary-6)) 100%,
var(--color-fill-3) 100%);
cursor: pointer;
}
#eliauk-container .eliauk-slider::-webkit-slider-thumb {
-webkit-appearance: none;
appearance: none;
width: 12px;
height: 12px;
background: #fff;
border: 2px solid rgb(var(--primary-6));
border-radius: 50%;
transition: all 0.3s cubic-bezier(0.3, 1.3, 0.3, 1);
}
#eliauk-container .eliauk-slider::-webkit-slider-thumb:hover {
box-shadow: 0 2px 5px #0000001a;
transform: scale(1.16666667);
}
/* 输入框 */
#eliauk-container .arco-input-wrapper {
padding: 0px;
}
#eliauk-container .arco-input-wrapper .arco-input {
text-align: center;
}
/* 颜色输入框 */
#eliauk-container .color {
border: none;
outline: none;
width: 100%;
border-radius: 0;
border-top-right-radius: var(--border-radius-small);
border-bottom-right-radius: var(--border-radius-small);
background-color: transparent;
padding: 0;
margin: 0;
overflow: hidden;
}
#eliauk-container .color::-webkit-color-swatch-wrapper {
padding: 0;
margin: 0;
}
#eliauk-container .color::-webkit-color-swatch {
border: none;
}
/* 预览区 */
#eliauk-container .context {
text-align: left;
}
#eliauk-container .context div {
background: ${data.scrollbar.background};
border-radius: ${data.scrollbar.radius}px;
display: inline-block;
}
#eliauk-container .context .context-y {
height: 100px;
width: ${data.scrollbar.width}px;
}
#eliauk-container .context .context-x {
height: ${data.scrollbar.height}px;
width: 100px;
}
/* 折叠面板 */
#eliauk-container .arco-collapse-item .arco-collapse-item-content {
/* display: none; */
height: 0;
padding: 0;
transition: all .2s;
}
#eliauk-container .arco-collapse-item-content-box {
padding: 5px;
height: 100%;
box-sizing: border-box;
}
#eliauk-container .arco-collapse-item.arco-collapse-item-active .arco-collapse-item-content {
display: block;
height: 100px;
}
#eliauk-container .arco-collapse-item .arco-collapse-item-header .arco-collapse-item-header-title {
font-weight: 500;
}
#rule-list-title-collapse {
margin-bottom: 5px;
}
#rule-list-title-collapse .arco-collapse-item.arco-collapse-item-active .arco-collapse-item-content {
height: 380px;
}
#rule-list-title-collapse:has(.arco-collapse-item.arco-collapse-item-active)+.arco-collapse {
height: 0;
opacity: 0;
}
/* 文本域 */
#eliauk-container .arco-textarea-wrapper {
display: inline-flex;
height: 100%;
}
#eliauk-container .arco-textarea::-webkit-scrollbar,
#eliauk-container div::-webkit-scrollbar {
width: 10px;
background: transparent;
}
#eliauk-container .arco-textarea::-webkit-scrollbar-thumb,
#eliauk-container div::-webkit-scrollbar-thumb {
background: var(--color-fill-3) content-box;
border: 2px solid transparent;
border-radius: 5px;
}
`;
// 添加 自定义的动画
const aniStyle = `
@keyframes ani_leftToright {
0% {
transform: translateX(-32px);
opacity: 0.2;
}
20% {
opacity: 0.5;
}
30% {
opacity: 0.8;
}
100% {
opacity: 1;
}
}
/* ani */
@keyframes ani_bottomTotop {
0% {
transform: translateY(32px);
opacity: 0.2;
}
20% {
opacity: 0.5;
}
30% {
opacity: 0.8;
}
100% {
opacity: 1;
}
}
@-webkit-keyframes ani_topTobuttom {
0% {
transform: translateY(-32px);
opacity: 0.2;
}
20% {
opacity: 0.5;
}
30% {
opacity: 0.8;
}
100% {
opacity: 1;
}
}
@-webkit-keyframes ani_hideToShow {
0% {
display:none;
opacity: 0.2;
}
20% {
opacity: 0.5;
}
30% {
opacity: 0.8;
}
100% {
opacity: 1;
}
}
@-webkit-keyframes ani_showToHide {
0% {
display:none;
opacity: 1;
}
20% {
opacity: 0.8;
}
30% {
opacity: 0.5;
}
100% {
opacity: 0.3;
}
}
.aniDelete {
transition: all 0.15s cubic-bezier(0.4, 0, 1, 1);
opacity: 0.1;
}
`;
const context = {
flushFragment: new FlushDomFragment()
}
main();
function main() {
reloadScrollbar();
// 注册菜单项
registerMenuCommand();
// 等待body加载完成
safeWaitFunc("body", () => {
retryInterval(() => {
if (!mountSettingModal()) return false;
reloadNodeParam();
reloadRules();
reloadBasicSettingsPage();
reloadSolidPage();
addEventListenerAll();
return true;
}, 200, true);
// retryInterval(() => {
// context.flushFragment.flush();
// debug("reload");
// return true;
// }, 200, true);
// debug("retry");
preloadGMStyle();
context.flushFragment.flush();
})
}
function reloadAllConfig() {
const res = GM_getValue(ELCONFIG);
debug(res);
if (res && (res !== 'undefined' && res !== 'null')) {
try {
data = JSON.parse(res);
} catch (e) {
data = res;
}
} else {
data = DefaultConfig;
debug("default", data);
}
}
function reloadScrollbar() {
debug(data.rule.config);
const globalConfig = data.rule.config[data.rule.default];
const isCodeSetting = globalConfig && parseBoolean(globalConfig.enable);
addCustomStyle(isCodeSetting);
if (!isCodeSetting) {
const { width, height, radius, background } = data.scrollbar;
const style = `
::-webkit-scrollbar{width:${width}px;height:${height}px;}::-webkit-scrollbar-thumb{border: 2px solid transparent;border-radius:${radius}px;background:${background} content-box;}::-webkit-scrollbar-track{box-shadow:none;}
`
GM_addStyle(style);
// context.flushFragment.insert(createStyleElement(style, "eliauk-custom-style"), 'head', {
// isReload: true
// });
}
}
function reloadNodeParam() {
nodes = {
scrollbar: {
width: {
comp: $one("#scrollbar-width"),
text: $one("#scrollbar-width-text")
},
height: {
comp: $one("#scrollbar-height"),
text: $one("#scrollbar-height-text")
},
radius: {
comp: $one("#scrollbar-radius"),
text: $one("#scrollbar-radius-text")
}
},
color: {
solid: {
color: {
comp: $one("#solid-color"),
text: $one("#solid-color-text")
}
},
gradient: {
color: {
comp: $one("#gradient-color"),
text: $one("#gradient-color-text")
},
length: {
text: $one("#gradient-lenght")
},
step: {
comp: $one("#gradient-step"),
text: $one("#gradient-step-text")
},
direction: {
comp: $one("#gradient-direction"),
text: $one("#gradient-direction-text")
}
}
}
}
scrollbarY = $one("#eliauk-container .context .context-y");
scrollbarX = $one("#eliauk-container .context .context-x");
ruleTitles = $one("#rule-list-title-collapse .arco-textarea");
ruleConfig = $one("#rule-list-config-collapse");
}
// 添加所有监听器
function addEventListenerAll() {
// 渐变方向滑动输入事件
nodes.color.gradient.direction.comp.addEventListener("input", function () {
// debug("input", this.value);
nodes.color.gradient.direction.text.value = `${this.value}deg`;
data.scrollbar.color.gradient.direction = this.value;
reloadGradientPreviewBackground();
})
// 渐变位置滑动输入事件
nodes.color.gradient.step.comp.addEventListener("input", function () {
// debug("input", this.value);
let total = this.max - this.min;
let effset = Math.round((this.value * 100) / total);
nodes.color.gradient.step.text.value = `${effset}%`;
const color = data.scrollbar.color.gradient.steps[this.value] ?
data.scrollbar.color.gradient.steps[this.value] : data.scrollbar.color.default;
nodes.color.gradient.color.comp.value = color;
nodes.color.gradient.color.text.value = color;
})
// 纯色颜色输入框事件
nodes.color.solid.color.comp.addEventListener('input', function () {
// debug(this.value);
nodes.color.solid.color.text.value = this.value;
data.scrollbar.color.solid = this.value;
resetPreviewBackground(this.value);
})
// 渐变颜色输入框事件
nodes.color.gradient.color.comp.addEventListener('input', function () {
// debug(this.value);
nodes.color.gradient.color.text.value = this.value;
const linear = nodes.color.gradient.step.comp.value;
data.scrollbar.color.gradient.steps[linear] = this.value;
reloadGradientPreviewBackground();
});
$all("#eliauk-container .close-button").forEach((element) => {
element.addEventListener("click", closePanel);
});
$one("#eliauk-container .confirm-button").addEventListener("click", function () {
debug(data);
GM_setValue(ELCONFIG, JSON.stringify(data));
setTimeout(function () {
window.location.reload();
}, 200);
});
// 按钮数组输入框
$all("#eliauk-container .arco-input-number-mode-button").forEach(ele => {
const wrapper_input = $one(".arco-input-wrapper > .arco-input", ele);
const prepend_button = $one(".arco-input-prepend > .arco-btn", ele);
const append_button = $one(".arco-input-append > .arco-btn", ele);
const max = Number(wrapper_input.max) || Infinity;
const min = Number(wrapper_input.min) || -Infinity;
const step = Number(wrapper_input.step) || 1;
prepend_button.addEventListener("click", () => {
if (Number(wrapper_input.value) > min) {
wrapper_input.value = Number(wrapper_input.value) - step;
wrapper_input.dispatchEvent(new Event('change'))
}
})
append_button.addEventListener("click", () => {
if (Number(wrapper_input.value) < max) {
wrapper_input.value = Number(wrapper_input.value) + step;
wrapper_input.dispatchEvent(new Event('change'))
}
})
})
// 单选组
$all("#eliauk-container .arco-radio-group-button").forEach(group => {
$all(".arco-radio-button", group).forEach(ele => {
ele.addEventListener('click', function (e) {
e.preventDefault();
$one(".arco-radio-checked", group).classList.remove("arco-radio-checked");
this.classList.add("arco-radio-checked");
group.dispatchEvent(new Event('select'));
})
})
})
// 滑动输入条
$all("#eliauk-container .eliauk-slider").forEach(ele => {
redrawSliderStyle(ele);
ele.addEventListener("input", () => redrawSliderStyle(ele));
ele.addEventListener("mousewheel", function (e) {
e.preventDefault();
const symbol = e.wheelDelta < 0 ? -1 : 1;
const step = Number(ele.step) || 1;
this.value = Number(this.value) + (symbol * step);
this.dispatchEvent(new Event('input'));
})
});
// 滚动条y方向宽度
nodes.scrollbar.width.comp.addEventListener("input", function () {
scrollbarY.style.width = `${this.value}px`;
data.scrollbar.width = this.value;
nodes.scrollbar.width.text.value = this.value
})
// 滚动条x方向高度
nodes.scrollbar.height.comp.addEventListener("input", function () {
scrollbarX.style.height = `${this.value}px`;
data.scrollbar.height = this.value;
nodes.scrollbar.height.text.value = this.value;
})
// 滚动条圆角值
nodes.scrollbar.radius.comp.addEventListener("input", function () {
// debug("input", this.value);
scrollbarX.style.borderRadius = `${this.value}px`;
scrollbarY.style.borderRadius = `${this.value}px`;
data.scrollbar.radius = this.value;
nodes.scrollbar.radius.text.value = this.value
})
// 基础、高级切换按钮
$one("#eliauk-container .page-collapse-btn").addEventListener("click", function () {
// debug(this);
// debug(this.classList.value);
const aside = $one("#eliauk-container .page-collapse");
if (this.classList.contains("page-collapse-btn-collapse")) {
this.classList.remove("page-collapse-btn-collapse");
aside.classList.remove("page-collapse-collapse");
$one("#page-title").innerText = "高级设置";
} else {
this.classList.add("page-collapse-btn-collapse")
aside.classList.add("page-collapse-collapse");
$one("#page-title").innerText = "基础设置";
}
})
$all("#eliauk-container .arco-collapse").forEach(coll => {
// 事件委派
coll.addEventListener("input", ev => {
debug(ev);
const $ev = ev || window.event;
const target = $ev.target || $ev.srcElement;
if (target.classList.contains("arco-textarea")) {
debug(target);
const titleComp = $one(`#rule-list-config-collapse .arco-collapse-item-header-title[data-v="${target.dataset.v}"]`, coll);
if (!titleComp) return;
const title = titleComp.innerText;
debug(title, target.value);
if (!data.rule.config[title]) {
data.rule.config[title] = {};
}
data.rule.config[title].style = target.value;
}
})
coll.addEventListener("click", ev => {
const $ev = ev || window.event;
const target = $ev.target || $ev.srcElement;
// debug(target);
if (isClickSwitch(target)) {
arcoSwitchChange(target, coll);
return;
}
if (isClickCollapseItemHeader(target)) {
arcoCollapseItemHeaderClick(target, coll);
return;
}
})
});
ruleTitles.addEventListener("change", () => {
// debug(JSON.stringify(ruleTitles.value));
data.rule.titles = ruleTitles.value
.split(";")
.map(val => val.trim())
.filter(val => !!val);
// debug(new RegExp(data.rule.titles[0]).test("https://.tbilibili.com"));
})
$one(`#rule-list-title-collapse .arco-collapse-item.arco-collapse-item-active
.arco-collapse-item-header`).addEventListener("click", function () {
clearChilrenNodes(ruleConfig);
// debug(title);
const titles = [...data.rule.default, ...data.rule.titles];
titles.forEach(item => {
const data_v = generateMixed(6);
const text_value = data.rule.config[item]?.style || "";
ruleConfig.insertAdjacentHTML('beforeend', `
<div class="arco-collapse-item" data-v="${data_v}">
<div role="button" aria-disabled="false" aria-expanded="false" tabindex="0"
class="arco-collapse-item-header arco-collapse-item-header-left" data-v="${data_v}">
<span class="arco-icon-hover arco-collapse-item-icon-hover">
<svg viewBox="0 0 48 48" fill="none" xmlns="http://www.w3.org/2000/svg"
stroke="currentColor"
class="arco-icon arco-icon-right arco-collapse-item-expand-icon"
stroke-width="4" stroke-linecap="butt" stroke-linejoin="miter" data-v="${data_v}">
<path d="m16 39.513 15.556-15.557L16 8.4"></path>
</svg>
</span>
<div class="arco-collapse-item-header-title" data-v="${data_v}">${item.trim()}</div>
<div class="arco-collapse-item-header-extra">
<button type="button" role="switch" onchange="ruleConfigSwitch(this)" aria-checked="${!!data.rule.config[item]?.enable}" class="arco-switch arco-switch-type-circle arco-switch-small${!!data.rule.config[item]?.enable ? " arco-switch-checked" : " "}" data-v="${data_v}">
<span class="arco-switch-handle" data-v="${data_v}">
<span class="arco-switch-handle-icon" data-v="${data_v}"></span>
</span>
</button>
</div>
</div>
<div role="region" class="arco-collapse-item-content">
<div class="arco-collapse-item-content-box">
<div class="arco-textarea-wrapper arco-textarea-scroll">
<textarea class="arco-textarea" style="resize: none; overflow: auto;" data-v="${data_v}">${text_value}</textarea>
</div>
</div>
</div>
</div>
`);
});
// debug(list);
// $one("#rule-list-config-collapse");
})
}
function mountSettingModal() {
if (document.body === null) return false;
if ($one("#eliauk-container") !== null) return true;
const Container = document.createElement("div");
Container.id = "eliauk-container";
Container.innerHTML = `
<div class="arco-modal">
<div class="arco-modal-header">
<div class="arco-modal-title arco-modal-title-align-center">
<div class="aside-top">
<button type="button"
class="page-collapse-btn-collapse arco-btn arco-btn-secondary arco-btn-shape-circle arco-btn-size-large arco-btn-status-normal page-collapse-btn">
<svg viewBox="0 0 48 48" fill="none" xmlns="http://www.w3.org/2000/svg"
stroke="currentColor" class="arco-icon arco-icon-up" stroke-width="4"
stroke-linecap="butt" stroke-linejoin="miter">
<path d="M39.6 30.557 24.043 15 8.487 30.557"></path>
</svg>
</button>
</div>
</div>
<div tabindex="-1" role="button" aria-label="Close" class="close-button arco-modal-close-btn">
<span class="arco-icon-hover">
<svg viewBox="0 0 48 48" fill="none" xmlns="http://www.w3.org/2000/svg" stroke="currentColor"
class="arco-icon arco-icon-close" stroke-width="4" stroke-linecap="butt"
stroke-linejoin="miter">
<path d="M9.857 9.858 24 24m0 0 14.142 14.142M24 24 38.142 9.858M24 24 9.857 38.142"></path>
</svg>
</span>
</div>
</div>
<div class="arco-modal-body">
<div class="arco-divider arco-divider-horizontal arco-divider-with-text">
<span id="page-title" class="arco-divider-text arco-divider-text-left">基础设置</span>
</div>
<div class="page">
<div class="page-one page-collapse page-collapse-collapse">
<div id="rule-list-title-collapse" class="arco-collapse">
<div class="arco-collapse-item arco-collapse-item-active" data-v="111">
<div role="button" aria-disabled="false" aria-expanded="true" tabindex="0"
class="arco-collapse-item-header arco-collapse-item-header-left" data-v="111">
<span class="arco-icon-hover arco-collapse-item-icon-hover">
<svg viewBox="0 0 48 48" fill="none" xmlns="http://www.w3.org/2000/svg"
stroke="currentColor"
class="arco-icon arco-icon-right arco-collapse-item-expand-icon"
stroke-width="4" stroke-linecap="butt" stroke-linejoin="miter" data-v="111">
<path d="m16 39.513 15.556-15.557L16 8.4"></path>
</svg>
</span>
<div class="arco-collapse-item-header-title" data-v="111">
域名列表
</div>
</div>
<div role="region" class="arco-collapse-item-content">
<div class="arco-collapse-item-content-box">
<div class="arco-textarea-wrapper arco-textarea-scroll">
<textarea class="arco-textarea"
style="resize: none; overflow: auto;"></textarea>
</div>
</div>
</div>
</div>
</div>
<div id="rule-list-config-collapse" class="arco-collapse"></div>
</div>
<div class="page-two">
<form class="arco-form arco-form-layout-horizontal arco-form-size-mini">
<div class="arco-row arco-row-align-start arco-row-justify-start">
<div class="arco-col arco-col-24">
<div
class="arco-row arco-row-nowrap arco-row-align-start arco-row-justify-start arco-form-item arco-form-item-layout-horizontal">
<div class="arco-col arco-form-item-label-col" style="flex: 0 0 110px">
<label class="arco-form-item-label">滚动条宽度(px)</label>
</div>
<div class="arco-col arco-form-item-wrapper-col">
<div class="arco-form-item-content-wrapper">
<div class="arco-form-item-content arco-form-item-content-flex">
<span class="arco-input-wrapper eliauk-input-number"
style="flex: 0 0 60px;">
<input id="scrollbar-width-text" type="text" value="6"
class="arco-input arco-input-size-mini" disabled />
</span>
<div class="eliauk-slider-context"
style="flex: 1; margin-left: 15px;">
<input id="scrollbar-width" type="range" value="6" min="0"
max="30" step="1" class="eliauk-slider" />
</div>
</div>
</div>
</div>
</div>
</div>
<div class="arco-col arco-col-24">
<div
class="arco-row arco-row-nowrap arco-row-align-start arco-row-justify-start arco-form-item arco-form-item-layout-horizontal">
<div class="arco-col arco-form-item-label-col" style="flex: 0 0 110px">
<label class="arco-form-item-label">滚动条高度(px)</label>
</div>
<div class="arco-col arco-form-item-wrapper-col">
<div class="arco-form-item-content-wrapper">
<div class="arco-form-item-content arco-form-item-content-flex">
<span class="arco-input-wrapper eliauk-input-number"
style="flex: 0 0 60px;">
<input id="scrollbar-height-text" value="6" type="text"
class="arco-input arco-input-size-mini" disabled />
</span>
<div class="eliauk-slider-context"
style="flex: 1; margin-left: 15px;">
<input id="scrollbar-height" type="range" value="6" min="0"
max="30" step="1" class="eliauk-slider" />
</div>
</div>
</div>
</div>
</div>
</div>
<div class="arco-col arco-col-24">
<div
class="arco-row arco-row-nowrap arco-row-align-start arco-row-justify-start arco-form-item arco-form-item-layout-horizontal">
<div class="arco-col arco-form-item-label-col" style="flex: 0 0 110px">
<label class="arco-form-item-label">滚动条弧度(px)</label>
</div>
<div class="arco-col arco-form-item-wrapper-col">
<div class="arco-form-item-content-wrapper">
<div class="arco-form-item-content arco-form-item-content-flex">
<span class="arco-input-wrapper eliauk-input-number"
style="flex: 0 0 60px;">
<input id="scrollbar-radius-text" value="10" type="text"
class="arco-input arco-input-size-mini" disabled />
</span>
<div class="eliauk-slider-context"
style="flex: 1; margin-left: 15px;">
<input id="scrollbar-radius" type="range" value="10" min="0"
max="15" step="0.1" class="eliauk-slider" />
</div>
</div>
</div>
</div>
</div>
</div>
</div>
<div class="arco-divider arco-divider-horizontal arco-divider-with-text">
<span class="arco-divider-text arco-divider-text-left">颜色</span>
</div>
<div style="margin-bottom: 10px;">
<span onselect="colorSelect(this)"
class="arco-radio-group-button arco-radio-group-size-mini arco-radio-group-direction-horizontal">
<label class="arco-radio-button arco-radio-checked">
<input type="radio" class="arco-radio-target" value="SolidColor">
<span class="arco-radio-button-content">纯色</span>
</label>
<label class="arco-radio-button">
<input type="radio" class="arco-radio-target" value="GradientColor">
<span class="arco-radio-button-content">渐变色</span>
</label>
</span>
</div>
<div id="solid-color-page" class="arco-row arco-row-align-start arco-row-justify-start">
<div class="arco-col arco-col-24">
<div
class="arco-row arco-row-nowrap arco-row-align-start arco-row-justify-start arco-form-item arco-form-item-layout-horizontal">
<div class="arco-col arco-form-item-label-col" style="flex: 0 0 110px">
<label class="arco-form-item-label">滚动条颜色(hex)</label>
</div>
<div class="arco-col arco-form-item-wrapper-col">
<div class="arco-form-item-content-wrapper">
<div class="arco-form-item-content arco-form-item-content-flex">
<span
class="arco-input-outer arco-input-outer-size-mini arco-input-search">
<span class="arco-input-wrapper" style="flex: 0 0 90px;">
<input class="arco-input arco-input-size-mini" type="text"
id="solid-color-text" value="#f2f3f5" disabled />
</span>
<span class="arco-input-append" style="width:30px">
<input value="#f2f3f5" type="color" id="solid-color"
class="color arco-btn-size-mini" />
</span>
</span>
</div>
</div>
</div>
</div>
</div>
</div>
<div id="gradient-color-page" class="arco-row arco-row-align-start arco-row-justify-start"
style="display: none;">
<div class="arco-col arco-col-24">
<div
class="arco-row arco-row-nowrap arco-row-align-start arco-row-justify-start arco-form-item arco-form-item-layout-horizontal">
<div class="arco-col arco-form-item-label-col" style="flex: 0 0 110px">
<label class="arco-form-item-label">滚动条渐变长度</label>
</div>
<div class="arco-col arco-form-item-wrapper-col">
<div class="arco-form-item-content-wrapper">
<div class="arco-form-item-content arco-form-item-content-flex">
<span style="flex: 0 0 90px"
class="arco-input-outer arco-input-outer-size-mini arco-input-number arco-input-number-mode-button arco-input-number-size-mini">
<span class="arco-input-prepend">
<button type="button"
class="arco-btn arco-btn-secondary arco-btn-shape-square arco-btn-size-mini arco-btn-status-normal arco-btn-only-icon arco-input-number-step-button">
<span class="arco-btn-icon">
<svg viewBox="0 0 48 48" fill="none"
xmlns="http://www.w3.org/2000/svg"
stroke="currentColor"
class="arco-icon arco-icon-minus"
stroke-width="4" stroke-linecap="butt"
stroke-linejoin="miter">
<path d="M5 24h38"></path>
</svg>
</span>
</button>
</span>
<span class="arco-input-wrapper">
<input id="gradient-lenght"
onchange="gradientLenghtChange(this)"
class="arco-input arco-input-size-mini" value="1"
type="text" min="1" max="10" disabled />
</span>
<span class="arco-input-append">
<button
class="arco-btn arco-btn-secondary arco-btn-shape-square arco-btn-size-mini arco-btn-status-normal arco-btn-only-icon arco-input-number-step-button"
type="button">
<span class="arco-btn-icon">
<svg viewBox="0 0 48 48" fill="none"
xmlns="http://www.w3.org/2000/svg"
stroke="currentColor"
class="arco-icon arco-icon-plus"
stroke-width="4" stroke-linecap="butt"
stroke-linejoin="miter">
<path d="M5 24h38M24 5v38"></path>
</svg>
</span>
</button>
</span>
</span>
<span style="margin-left: 5px;"
class="arco-input-outer arco-input-outer-size-mini arco-input-search">
<span class="arco-input-wrapper">
<input class="arco-input arco-input-size-mini" type="text"
id="gradient-color-text" value="#f2f3f5" disabled />
</span>
<span class="arco-input-append" style="width:30px">
<input value="#f2f3f5" type="color" id="gradient-color"
class="color arco-btn-size-mini" />
</span>
</span>
</div>
</div>
</div>
</div>
</div>
<div class="arco-col arco-col-24">
<div
class="arco-row arco-row-nowrap arco-row-align-start arco-row-justify-start arco-form-item arco-form-item-layout-horizontal">
<div class="arco-col arco-form-item-label-col" style="flex: 0 0 110px">
<label class="arco-form-item-label">滚动条渐变位</label>
</div>
<div class="arco-col arco-form-item-wrapper-col">
<div class="arco-form-item-content-wrapper">
<div class="arco-form-item-content arco-form-item-content-flex">
<span class="arco-input-wrapper eliauk-input-number"
style="flex: 0 0 60px;">
<input id="gradient-step-text" value="0%" type="text"
class="arco-input arco-input-size-mini" disabled />
</span>
<div class="eliauk-slider-context"
style="flex: 1; margin-left: 15px;">
<input id="gradient-step" type="range" value="0" min="0" max="1"
step="1" class="eliauk-slider" />
</div>
</div>
</div>
</div>
</div>
</div>
<div class="arco-col arco-col-24">
<div
class="arco-row arco-row-nowrap arco-row-align-start arco-row-justify-start arco-form-item arco-form-item-layout-horizontal">
<div class="arco-col arco-form-item-label-col" style="flex: 0 0 110px">
<label class="arco-form-item-label">滚动条渐变方向</label>
</div>
<div class="arco-col arco-form-item-wrapper-col">
<div class="arco-form-item-content-wrapper">
<div class="arco-form-item-content arco-form-item-content-flex">
<span class="arco-input-wrapper eliauk-input-number"
style="flex: 0 0 60px;">
<input id="gradient-direction-text" value="0deg"
class="arco-input arco-input-size-mini" type="text"
disabled />
</span>
<div class="eliauk-slider-context"
style="flex: 1; margin-left: 15px;">
<input id="gradient-direction" type="range" value="0" min="0"
max="360" step="1" class="eliauk-slider" />
</div>
</div>
</div>
</div>
</div>
</div>
</div>
</form>
<div class="arco-divider arco-divider-horizontal arco-divider-with-text">
<span class="arco-divider-text arco-divider-text-left">预览</span>
</div>
<div class="context">
<div class="context-x"></div>
<div class="context-y"></div>
</div>
</div>
</div>
</div>
<div class="arco-modal-footer">
<button type="button"
class="close-button arco-btn arco-btn-secondary arco-btn-shape-square arco-btn-size-mini arco-btn-status-normal">
取消
</button>
<button type="button"
class="confirm-button arco-btn arco-btn-primary arco-btn-shape-square arco-btn-size-mini arco-btn-status-normal">
确定
</button>
</div>
</div>
`;
try {
document.body.appendChild(Container);
} catch (e) {
debug(e);
}
}
function closePanel() {
debug("触发取消按钮");
document.querySelector("#eliauk-container").style.display = "none";
}
// 注册脚本菜单
function registerMenuCommand() {
const currentList = GM_getValue(CURRENT_LIST, []);
// 清除之前的
clearCommand(currentList);
currentList.push(GM_registerMenuCommand(`⚙️网页美化设置`, () => {
$one("#eliauk-container").style.display = "block";
}));
currentList.push(GM_registerMenuCommand(`🔄脚本重置 - 修复脚本`, () => {
GM_setValue(ELCONFIG, undefined);
location.reload();
}))
GM_setValue(CURRENT_LIST, currentList);
}
// 清除所有菜单项
function clearCommand(list) {
while (list.length) {
GM_unregisterMenuCommand(list.shift());
}
}
// 检查是否存在于名单中 返回配置
function getStyle(configs, url) {
debug(configs);
const conf = [];
Object.entries(configs).forEach(([reg, config]) => {
debug(reg, config);
debug(new RegExp(reg), url)
debug(new RegExp(reg).test(url), parseBoolean(config.enable))
if (new RegExp(reg).test(url) && parseBoolean(config.enable)) {
conf.push(config.style);
}
});
const style = conf.join(" ");
debug(conf);
return style;
}
// 选择开关切换事件
function arcoSwitchChange(target, element) {
// debug(target);
const btn = $one(`.arco-switch[data-v="${target.dataset.v}"]`, element);
if (!btn) return;
if (btn.classList.contains("arco-switch-checked")) {
btn.classList.remove("arco-switch-checked");
btn.setAttribute("aria-checked", false);
} else {
btn.classList.add("arco-switch-checked");
btn.setAttribute("aria-checked", true);
}
btn.dispatchEvent(new Event('change'));
}
function isClickSwitch(target) {
return [
'arco-switch',
'arco-switch-handle',
'arco-switch-handle-icon'
].some(item => target.classList.contains(item));
}
function isClickCollapseItemHeader(target) {
return [
'arco-collapse-item-header',
'arco-collapse-item-header-title',
'arco-collapse-item-expand-icon'
].some(item => target.classList.contains(item));
}
function arcoCollapseItemHeaderClick(target, coll) {
const header = $one(`.arco-collapse-item[data-v="${target.dataset.v}"]`, coll)
if (!header) return;
const isActive = header.classList.contains("arco-collapse-item-active");
if (isActive) {
header.classList.remove("arco-collapse-item-active");
} else {
const active = $one(".arco-collapse-item.arco-collapse-item-active", coll);
if (active) {
active.classList.remove("arco-collapse-item-active");
}
header.classList.add("arco-collapse-item-active");
}
}
// debug(data.rule.titles.join(";\n"));
// 清除子节点
function clearChilrenNodes(ele) {
var len = ele.childNodes.length; // 子元素的个数
for (var i = len - 1; i >= 0; i--) { // 从后往前
ele.removeChild(ele.childNodes[i]); // 从第一个元素开始删除
}
}
//生成n位数字字母混合字符串
function generateMixed(n) {
const chars = ['0', '1', '2', '3', '4', '5', '6', '7', '8', '9',
'A', 'B', 'C', 'D', 'E', 'F', 'G', 'H', 'I', 'J', 'K', 'L', 'M',
'N', 'O', 'P', 'Q', 'R', 'S', 'T', 'U', 'V', 'W', 'X', 'Y', 'Z'
];
let res = "";
for (let i = 0; i < n; i++) {
const id = Math.floor(Math.random() * chars.length);
res += chars[id];
}
return res;
}
function preloadGMStyle() {
function loadResource(resourceName) {
const data = GM_getResourceText(resourceName)
GM_addStyle(data);
}
loadResource("ArcoDesignStyle");
// 添加全局样式
GM_addStyle(globalStyle);
}
function addCustomStyle(isCodeSetting) {
if (!isCodeSetting) {
safeRemove("style[class='eliauk-user-style']");
} else {
const customStyle = getStyle(data.rule.config, hostname)
if (customStyle) {
context.flushFragment.insert(createStyleElement(customStyle, "eliauk-user-style"), 'head', {
isReload: true
})
}
}
context.flushFragment.insert(createStyleElement(aniStyle, "eliauk-animation-style"))
}
})();