// ==UserScript== // @name 1panel 日志颜色覆写 // @description 1panel 管理面板的容器日志界面太唐了,此脚本用于强行覆盖样式 // @version 2025-12-08-v1.0.1 // @author WIFI连接超时 // @match https://demo.1panel.cn/ // @license MIT // @icon  // @namespace https://greasyfork.org/users/1405510 // @downloadURL https://update.greasyfork.icu/scripts/558283/1panel%20%E6%97%A5%E5%BF%97%E9%A2%9C%E8%89%B2%E8%A6%86%E5%86%99.user.js // @updateURL https://update.greasyfork.icu/scripts/558283/1panel%20%E6%97%A5%E5%BF%97%E9%A2%9C%E8%89%B2%E8%A6%86%E5%86%99.meta.js // ==/UserScript== (function () { 'use strict' // ----------------< 配置区 >---------------- // 日志区域背景色 const backgroundColor = '#fff' // 日志区域文字颜色匹配对照表 const textColorMatchMap = { // 普通文本要深黑色 '#dadada': '#333333', '#bbbbbb': '#333333', '#919191': '#333333', '#6c6c6c': '#333333', '#4f4f4f': '#333333', // 其他颜色覆写 '#ff0000': '#ce0b48', '#ff7d00': '#dc6f01', '#03ad00': '#02c464', '#1daf66': '#02c464', '#4db9b9': '#057aef', '#4b709b': '#057aef', '#9e44c0': '#b212e3', } // 匹配色列表 const textColorMatchList = [...Object.keys(textColorMatchMap)] // 匹配彩色列表 const matchColorList = [] // 匹配中性色列表 const matchNeutralList = [] for (const color of textColorMatchList) { const rgb = hexToRgb(color) if (isNeutral(rgb)) { matchNeutralList.push(color) } else { matchColorList.push(color) } } // ----------------------------------------- // 将 rgb(xx, xx, xx) 转成 { r, g, b } function parseRGB(rgbStr) { const nums = rgbStr.match(/\d+/g).map(Number) return { r: nums[0], g: nums[1], b: nums[2] } } // 将 hex 转成 { r, g, b } function hexToRgb(hex) { hex = hex.replace('#', '') if (hex.length === 3) { hex = hex.split('').map(c => c + c).join('') } return { r: Number.parseInt(hex.slice(0, 2), 16), g: Number.parseInt(hex.slice(2, 4), 16), b: Number.parseInt(hex.slice(4, 6), 16), } } // 判断颜色是否为中性色 function isNeutral(color) { const { r, g, b } = color const mean = (r + g + b) / 3 const variance = ((r - mean) ** 2 + (g - mean) ** 2 + (b - mean) ** 2) / 3 const std = Math.sqrt(variance) // 灰度阈值 return std < 10 } // 计算颜色距离 (欧氏距离) function colorDistance(c1, c2) { return Math.sqrt( (c1.r - c2.r) ** 2 + (c1.g - c2.g) ** 2 + (c1.b - c2.b) ** 2, ) } // 返回最接近的颜色 function findClosestHex(rgbStr) { const target = parseRGB(rgbStr) const targetIsNeutral = isNeutral(target) // 根据输入属于哪类,选对应的列表 const compareList = targetIsNeutral ? matchNeutralList : matchColorList // 如果某一类刚好为空,则 fallback 回全列表 const finalList = compareList.length > 0 ? compareList : textColorMatchList let minDist = Infinity let closestHex = null for (const hex of finalList) { const rgb = hexToRgb(hex) const dist = colorDistance(target, rgb) if (dist < minDist) { minDist = dist closestHex = hex } } return closestHex } // 计算颜色 function computeColor(spanStyleColor) { // console.log('%c输入颜色 ', `background: ${spanStyleColor}`) if (spanStyleColor) { const closestHex = findClosestHex(spanStyleColor) // console.log('%c匹配颜色 ', `background: ${closestHex}`) if (closestHex) { // console.log('%c输出颜色 ', `background: ${textColorMatchMap[closestHex]}`) return textColorMatchMap[closestHex] } } // 默认返回第一个 // console.log('%c默认返回 ', `background: ${textColorMatchMap[textColorMatchList[0]]}`) return textColorMatchMap[textColorMatchList[0]] } // 执行颜色覆写 function doOverwrite() { // 日志容器 const logContainer = document.querySelector('.log-container') // 覆写背景色 logContainer.style.backgroundColor = backgroundColor // 对日志元素执行颜色覆写 logContainer.querySelectorAll('span') .forEach((span) => { if (span.style.color) { span.style.color = computeColor(span.style.color) } }) } function loop() { setTimeout(() => { try { doOverwrite() } catch {} loop() }, 100) } loop() })()