网页美化
// ==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")) } })();