Greasy Fork

Greasy Fork is available in English.

1panel 日志颜色覆写

1panel 管理面板的容器日志界面太唐了,此脚本用于强行覆盖样式

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Greasemonkey 油猴子Violentmonkey 暴力猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Violentmonkey 暴力猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Userscripts ,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展后才能安装此脚本。

(我已经安装了用户脚本管理器,让我安装!)

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

(我已经安装了用户样式管理器,让我安装!)

// ==UserScript==
// @name         1panel 日志颜色覆写
// @description  1panel 管理面板的容器日志界面太唐了,此脚本用于强行覆盖样式
// @version      2025-12-08-v1.0.1
// @author       WIFI连接超时
// @match        https://demo.1panel.cn/
// @license MIT
// @icon         
// @namespace http://greasyfork.icu/users/1405510
// ==/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()
})()