基于 Unocss 的后台系统 SVG 图标方案实践

在后台管理系统中,图标通常并不是简单的装饰元素,而是菜单结构、功能语义和状态表达的一部分 。为了在保证灵活性的同时降低维护成本,我在项目中基于 UnoCSS 的 presetIcons 实现了一套本地 SVG 图标方案。本文主要记录该方案的实现方式及其背后的机制设计。

一:实现思路

  • 使用 本地 SVG 文件 作为唯一图标源

  • 借助 UnoCSS presetIcons 将 SVG 转换为 CSS 图标

  • 通过统一的预处理逻辑,使 SVG 图标:

    • 尺寸可控
    • 颜色可继承
    • 支持动态渲染
  • 解决 UnoCSS 按需生成机制下,动态图标类名无法被扫描的问题

二:UnoCSS presetIcons 与本地图标集合

配置:

php 复制代码
import { defineConfig, presetIcons } from "unocss";
import { FileSystemIconLoader } from "@iconify/utils/lib/loader/node-loaders";
​
const iconDir = "./src/assets/icons";
​
export default defineConfig({
  presets: [
    presetIcons({
      extraProperties: {
        display: "inline-block",
        width: "1em",
        height: "1em",
      },
      collections: {
        svg: FileSystemIconLoader(iconDir),
      },
    }),
  ],
});
  1. src/assets/icons 注册为一个图标集合
  2. 使用 svg 作为集合名,对应类名格式为:i-svg:icon-name
  3. 统一图标尺寸为 1em,使其行为与文字一致

三:SVG 预处理:注入fill="currentColor"

在实际使用中,SVG 图标最大的痛点之一是 颜色控制不统一

常见问题

  • 未声明 fill:浏览器默认渲染为黑色
  • 写死 fill="#000":无法通过 CSS 控制
  • 多状态(hover / active / disabled)需要多份 SVG

currentColor 的作用

currentColor 并不是 SVG 私有属性,而是 CSS 颜色关键字 ,表示当前元素的 color 值。

css 复制代码
color: red;
/* currentColor === red */

如果 SVG 的 fill 使用 currentColor,即:SVG 图标的颜色,完全由外层元素的 color 决定。

注入逻辑

javascript 复制代码
FileSystemIconLoader(iconDir, (svg) => {
  return svg.includes('fill="')
    ? svg
    : svg.replace(/^<svg /, '<svg fill="currentColor" ')
});

注入后,图标即可通过普通的文本颜色控制:

ini 复制代码
<i class="i-svg:user text-gray-400"></i>
<i class="i-svg:user text-blue-500 hover:text-blue-600"></i>
<i class="i-svg:user text-red-500"></i>

四:动态场景下的关键问题:UnoCSS 的按需生成机制

UnoCSS 是 按需生成(JIT) 的工具,它只会为 在构建阶段扫描到的类名 生成 CSS。

在后台系统中,一个非常典型的场景是:

ruby 复制代码
const iconClass = `i-svg:${menu.icon}`;
<i :class="iconClass"></i>

此时:

  • menu.icon 来自后端
  • 构建阶段无法确定具体类名
  • UnoCSS 无法扫描到真实的 i-svg:xxx

结果就是: 运行时 DOM 上有 class,但 CSS 不存在,图标无法显示。

五:safelist:解决动态图标的方案

ini 复制代码
import fs from "fs";
​
const iconDir = "./src/assets/icons";
​
const generateIconSafeList = () => {
  return fs
    .readdirSync(iconDir)
    .filter(file => file.endsWith(".svg"))
    .map(file => `i-svg:${file.replace(".svg", "")}`);
};  
arduino 复制代码
export default defineConfig({
  safelist: generateIconSafeList(),
});

动态菜单图标在生产环境也能稳定渲染

六:使用

less 复制代码
<i class="i-svg:home text-sm"></i>
<i :class="`i-svg:${menu.icon}`"></i>

完整代码示例:

javascript 复制代码
import { defineConfig } from "unocss";
​
import { FileSystemIconLoader } from "@iconify/utils/lib/loader/node-loaders";
import fs from "fs";
​
// 本地SVG图标目录
const iconDir = "./src/assets/icons";
​
// 读取本地 SVG 目录,自动生成 safelist
const generateIconSafeList = () => {
  try {
    return fs
      .readdirSync(iconDir)
      .filter((file) => file.endsWith(".svg"))
      .map((file) => `i-svg:${file.replace(".svg", "")}`);
  } catch (error) {
    console.error("无法读取图标:", error);
    return [];
  }
};
​
export default defineConfig({
  presets: [
    presetIcons({
      extraProperties: {
        display: "inline-block",
        width: "1em",
        height: "1em",
      },
      // 图标集合
      collections: {
        // svg 是图标集合名称,使用 `i-svg:图标名` 调用
        svg: FileSystemIconLoader(iconDir, (svg) => {
          // 如果 `fill` 没有定义,则添加 `fill="currentColor"`
          console.log(svg, "svgsvgsvsgvsgvsgvsg");
          return svg.includes('fill="') ? svg : svg.replace(/^<svg /, '<svg fill="currentColor" ');
        }),
      },
    }),
  ],
  safelist: generateIconSafeList(),
});
相关推荐
小夏卷编程2 小时前
ant-design-vue 2.0 a-table 中实现特殊行样式,选中样式,鼠标悬浮样式不一样
前端·javascript·vue.js
wulijuan8886662 小时前
前端性能优化之图片webp
前端
一颗烂土豆2 小时前
ECharts 水球图不够炫?试试 RayChart 的创意可视化玩法
前端·vue.js·数据可视化
天才熊猫君2 小时前
Vue 3 命令式弹窗组件
前端
NEXT062 小时前
CSS基础-标准盒模型与怪异盒模型
前端·css
DaMu2 小时前
Dreamcore3D ARPG IDE “手搓”游戏引擎,轻量级实时3D创作工具,丝滑操作,即使小白也能轻松愉快的创作出属于你自己的游戏世界!
前端·架构·three.js
代码猎人2 小时前
什么是尾调用,使用尾调用有什么好处?
前端
AI_56782 小时前
Webpack从“配置到提速”,4步解决“打包慢、体积大”问题
前端·javascript·vue.js
pinkQQx2 小时前
手把手搭建前端跨平台服务(IPlatform + iOS / Android / Web)
前端·javascript