前端(Vue)动态换肤的通用解决方案及原理分析(1)

news2025/2/6 4:13:09

动态换肤原理分析

比如此处将来会实现换肤功能,所以我们不能直接写死,而需要通过一个动态的值进行指定。

 <el-menu
    :default-active="activeMenu"
    :collapse="!$store.getters.sidebarOpened"
    :background-color="$store.getters.cssVar.menuBg"
    :text-color="$store.getters.cssVar.menuText"
    :active-text-color="$store.getters.cssVar.menuActiveText"
    :unique-opened="true"
    router
  >

那么换句话而言,想要实现 动态换肤 的一个前置条件就是:色值不可以写死!【处理国际化也是同理,title 名称绝对不能写死】
动态换肤的实现方式 =>
scss 中,我们可以通过 $变量名:变量值 的方式定义 css 变量 ,然后通过该 css 来去指定某一块 DOM 对应的颜色。

$menuText: #bfcbd9;
$menuActiveText: #ffffff;
$subMenuActiveText: #f4f4f5;

$menuBg: #304156;
$menuHover: #263445;

$subMenuBg: #1f2d3d;
$subMenuHover: #001528;

$sideBarWidth: 210px;
$hideSideBarWidth: 54px;
$sideBarDuration: 0.28s;

如果我此时改变了该 css 变量的值,那么对应的 DOM 颜色是不是也会同步发生变化。
当大量的 DOM 都依赖这个 css 变量 设置颜色时,我们是不是只需要改变这个 css 变量 ,那么所有 DOM 的颜色是不是都会发生变化,所谓的 动态换肤 是不是就可以实现了!
这个就是 动态换肤 的实现原理。
一般应用到项目中,主要是两方面内容:

  1. 第三方 UI 组件库的主题(如element-plus 主题)
  2. 自定义主题(自己项目组件的主题)

动态换肤实现方案

从原理中可以得到以下两个关键信息:

  1. 动态换肤的关键是修改 css 变量 的值
  2. 换肤需要同时兼顾
    1. 项目中第三方或自定义组件库的主题(如element-plus)
    2. 非第三方或自定义组件库

【假设使用element-plus
那么根据以上关键信息,就可以得出对应的实现方案

  1. 创建一个组件 ThemeSelect 用来处理修改之后的 css 变量 的值
  2. 根据新值修改 element-plus 主题色
  3. 根据新值修改非 element-plus 主题色

创建 ThemeSelect 组件

image.png

<template>
  <!-- 主题图标  -->
  <el-dropdown
    v-bind="$attrs"
    trigger="click"
    class="theme"
    @command="handleSetTheme"
    >
    <!-- 图标及tooltip -->
    <div>
      <!-- 国际化处理 -->
      <el-tooltip :content="$t('msg.navBar.themeChange')">
        <svg-icon icon="change-theme" />
      </el-tooltip>
    </div>
    <!-- menu,使用具名插槽 -->
    <template #dropdown>
      <el-dropdown-menu>
        <!-- 内部再使用它的item绑定选择项,我们这里只有一个选择项。 -->
        <el-dropdown-item command="color">
          {{ $t('msg.theme.themeColorChange') }}
        </el-dropdown-item>
      </el-dropdown-menu>
    </template>
  </el-dropdown>
  <!-- 展示弹出层 (暂空) -->
  <div></div>
</template>

<script setup>
  // 点击事件
  const handleSetTheme = command => {}
</script>

<style lang="scss" scoped></style>

使用:

<div class="right-menu">
  <theme-picker class="right-menu-item hover-effect"></theme-picker>

v-bind=“ a t t r s " ,这里的 ‘ " attrs" ,这里的`" attrs",这里的‘"attrs” 就是指父组件传递的属性【如这里的 theme-picker 的 class 属性】绑定到子组件el-dropdown`上。【后面有补充】
运行到页面上就可以看到了。
image.png
删掉v-bind

<el-dropdown
    trigger="click"
    class="theme"
    @command="handleSetTheme"
>

就发现el-dropdown上之前的class没了。
image.png

v-bind=“$attrs”【补充】

当我们使用该组件的时候,在父组件中给它指定的一些属性会直接被绑定到当前根元素上。
v-bind="$attrs" 被用在 el-dropdown 上,这意味着所有父组件传递给这个组件的未声明(没定义在数组 props 中)属性和事件都会直接应用到 el-dropdown 上。这样做的好处是,不需要在当前组件中手动接收和处理这些属性或事件,而是自动将它们传递给 el-dropdown,从而使组件更加简洁和灵活。
通俗理解: 在 Vue.js 中,v-bind="$attrs" 是一种常用的方式,用于将父组件传递给当前组件的所有属性($attrs)绑定到某个子组件或元素上。

例如,如果父组件传递了一个 data-test="test" 的属性和一个 @custom-event="doSomething" 的事件到当前组件,而这些没有被显式地定义为 props,那么使用 v-bind="$attrs" 会将这些属性和事件自动传递给 el-dropdown

为什么使用 $attrs

  • 灵活性: 使用 $attrs 可以使组件更灵活和通用化,避免在子组件中手动列出所有可能的 props
  • 自动传递: 在一些情况下,你希望某些属性或事件自动传递给子组件而不需要手动处理,v-bind="$attrs" 就是一个简洁的解决方案。
  • 组合组件: 在构建复杂的组合组件时,父组件可以传递一组属性和事件到子组件,而子组件不需要显式处理这些属性和事件,可以直接传递给其内部的元素或子组件。

创建 SelectColor 组件

在有了 ThemeSelect 之后,接下来去处理颜色选择的组件 SelectColor(颜色选择组件),在这里我会用到 element+ 中的 el-color-picker 组件
ColorPicker 颜色选择器
image.png
对于 SelectColor 的处理,需要分成两步进行:

  1. 完成 SelectColor 弹窗展示的双向数据绑定
  2. 把选中的色值进行本地缓存
    1. vuex
    2. localStorage

第一步:完成 **SelectColor** 弹窗展示的双向数据绑定
vue2提供两种双向数据绑定第一种是v-model
image.png
还有.sync修饰符,但是在vue3中,它已经被去除了。


创建 SelectColor
弹出层 => el-dialog 包裹

<template>
  <!-- 因为期望在themeSlect里面通过双向数据绑定,控制SelectColor,SelectColor这里就不能用v-model,而是v-bind,单向数据绑定了。modelValue是Vue3中v-model在组件中使用的时候,默认的一个值。modelValue需要从父组件中接收过来。modelValue控制弹出层展示。
  之后处理dialog关闭事件
  弹出层的宽度是22%
  -->
  <el-dialog title="提示" 
    :model-value="modelValue" 
    @close="closed" width="22%">
    <!-- 内容区 -->
    <div class="center">
      <!-- 设置标题 -->
      <p class="title">{{ $t('msg.theme.themeColorChange') }}</p>
      <!-- 颜色选择器,需要设置一个初始色值predefineColors,即预定义色值。并设置双向绑定 -->
      <el-color-picker
        v-model="mColor"
        :predefine="predefineColors"
      ></el-color-picker>
    </div>
    <!-- foot -->
    <template #footer>
      <!-- 两个按钮,取消和确定,按钮文字也使用国际化 -->
      <span class="dialog-footer">
        <el-button @click="closed">{{ $t('msg.universal.cancel') }}</el-button>
        <el-button type="primary" @click="comfirm">{{
          $t('msg.universal.confirm')
        }}</el-button>
      </span>
    </template>
  </el-dialog>
</template>

<script setup>
import { defineProps, defineEmits, ref } from 'vue'
defineProps({
  modelValue: {
    type: Boolean,
    required: true
  }
})
// 通过父组件控制子组件显示的话,需要通过modelValue。通过父组件传递的数据,在子组件中进行展示。需要传递一个事件,从子组件通知父组件,modelValue 要发生变化了。事件在 Vue3 中使用 defineEmits 进行声明。
// 事件的名称  =>   update是固定写法,后面的值是指定修改哪一个从父组件传递来的props。 
const emits = defineEmits(['update:modelValue'])

// 预定义色值
const predefineColors = [
  '#ff4500',
  '#ff8c00',
  '#ffd700',
  '#90ee90',
  '#00ced1',
  '#1e90ff',
  '#c71585',
  'rgba(255, 69, 0, 0.68)',
  'rgb(255, 120, 0)',
  'hsv(51, 100, 98)',
  'hsva(120, 40, 94, 0.5)',
  'hsl(181, 100%, 37%)',
  'hsla(209, 100%, 56%, 0.73)',
  '#c7158577'
]
// 默认色值
const mColor = ref('#00ff00')

/**
 * 关闭
 */
const closed = () => {
  // 通知父组件触发该事件,modeValue 关闭了。
  emits('update:modelValue', false)
}
/**
 * 确定
 * 1. 修改主题色
 * 2. 保存最新的主题色
 * 3. 关闭 dialog
 */
const comfirm = async () => {
  // 3. 关闭 dialog
  closed()
}
</script>

<style lang="scss" scoped>
.center {
  text-align: center;
  .title {
    margin-bottom: 12px;
  }
}
</style>

ThemePicker (父组件)中使用该组件,中调用弹出层

<template>
  ...
  <!-- 展示弹出层 -->
  <div>
    <!-- 弹出层默认不展示的,通过 v-model 完成父子组件的双向绑定。 -->
    <select-color v-model="selectColorVisible"></select-color>
  </div>
</template>

<script setup>
import SelectColor from './components/SelectColor.vue'
import { ref } from 'vue'

const selectColorVisible = ref(false)
const handleSetTheme = command => {
  // 在点击事件中触发显示弹出层
  selectColorVisible.value = true
}
</script>

handleSetTheme => el-dropdown 的点击事件

  <!-- 主题图标  -->
  <el-dropdown
    v-bind="$attrs"
    trigger="click"
    class="theme"
    @command="handleSetTheme"
  >

现在完成第一步,完成 SelectColor 弹窗展示的双向数据绑定之后,接下来处理第二步:把选中的色值进行本地缓存
缓存的方式分为两种:

  1. vuex
  2. 本地存储

创建 store/modules/theme 模块,用来处理 主题色 相关内容(完成vuex和localStorage的存储)

import { getItem, setItem } from '@/utils/storage'
import { MAIN_COLOR, DEFAULT_COLOR } from '@/constant'
export default {
  namespaced: true,
  state: () => ({
    mainColor: getItem(MAIN_COLOR) || DEFAULT_COLOR
  }),
  mutations: {
    /**
     * 设置主题色
     */
    setMainColor(state, newColor) {
      state.mainColor = newColor
      setItem(MAIN_COLOR, newColor)
    }
  }
}

store/getters 下指定快捷访问

import { MAIN_COLOR } from '@/constant'
import { getItem } from '@/utils/storage'
import { generateColors } from '@/utils/theme'

const getters = {
  token: state => state.user.token,
  userInfo: state => state.user.userInfo,
  /**
   * @returns true 表示已存在用户信息
   */
  hasUserInfo: state => {
    return JSON.stringify(state.user.userInfo) !== '{}'
  },
  // ....
  // 新增 =>
  mainColor: state => state.theme.mainColor,
}
export default getters

store/index 中导入 theme

...
import theme from './modules/theme.js'

export default createStore({
  getters,
  modules: {
    ...
      theme
  }
})

selectColor 中,设置初始色值 和 缓存色值
点击确定的时候,就可以把双向绑定的色值缓存了。

...

<script setup>
import { defineProps, defineEmits, ref } from 'vue'
import { useStore } from 'vuex'
...
const store = useStore()
// 默认色值
const mColor = ref(store.getters.mainColor)
...
/**
 * 确定
 * 1. 修改主题色
 * 2. 保存最新的主题色
 * 3. 关闭 dialog
 */
const comfirm = async () => {
  // 2. 保存最新的主题色
  store.commit('theme/setMainColor', mColor.value)
  // 3. 关闭 dialog
  closed()
}
</script>
<el-color-picker
  v-model="mColor"
  :predefine="predefineColors"
></el-color-picker>

刷新页面,色值就有了。【不做本地缓存,刷新页面后主题信息会丢失!】

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2056556.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

手机使用技巧:如何恢复Android手机不见的短信

在您的 Android 手机上丢失短信可能是一种令人沮丧的经历&#xff0c;尤其是在文本包含重要信息的情况下。幸运的是&#xff0c;有一些方法可以在Android上恢复已删除的短信。在这篇博文中&#xff0c;我们将讨论几种在Android手机上恢复已删除短信的方法。 为什么需要恢复Andr…

测绘程序设计|认识VS2017|VS2017新建项目|VS2017使用技巧

由于微信公众号改变了推送规则&#xff0c;为了每次新的推送可以在第一时间出现在您的订阅列表中&#xff0c;记得将本公众号设为星标或置顶喔~ 分享了如何使用VS2017新建项目、VS2017的项目结构以及一些使用技巧~ &#x1f33f;前言 Visual Studio作为微软旗下一款热门的编程…

C语言典型例题46

《C程序设计教程&#xff08;第四版&#xff09;——谭浩强》 题目&#xff1a; 习题3.6 企业发放的奖金根据利润提成。利润I低于或等于100 000元的&#xff0c;奖金可提成10%&#xff1b; 利润高于100 000元&#xff0c;低于200000元&…

干货分享!渗透测试成功的8个关键

01 知道为什么要测试 执行渗透测试的目的是什么&#xff1f;是满足审计要求&#xff1f;是你需要知道某个新应用在现实世界中表现如何&#xff1f;你最近换了安全基础设施中某个重要组件而需要知道它是否有效&#xff1f;或者渗透测试根本就是作为你定期检查防御健康的一项例行…

2024年【电工(高级)】试题及解析及电工(高级)复审考试

题库来源&#xff1a;安全生产模拟考试一点通公众号小程序 2024年【电工&#xff08;高级&#xff09;】试题及解析及电工&#xff08;高级&#xff09;复审考试&#xff0c;包含电工&#xff08;高级&#xff09;试题及解析答案和解析及电工&#xff08;高级&#xff09;复审…

C语言——字符函数、字符串函数和内存函数

目录 1.字符分类函数 2.字符转换函数 3.字符串函数 3.1strlen 函数 3.1.1 strlen函数的模拟实现 3.1.1.1第一种方法&#xff1a;计算器方法 3.1.1.2 第二种方法&#xff1a;指针-指针 3.1.1.3 第三种方法&#xff1a;递归 3.2 strcpy 函数 3.2.1 strcpy函数的模拟实现…

成为Python砖家(4): 装饰器的简单理解

第一次理解 Python 中的装饰器&#xff08;decorator&#xff09;&#xff0c;是Python中一个非常强大的工具&#xff0c;它是一个返回函数的函数。 上面这个定义很简洁&#xff0c;但是没说清楚。 第二次理解 装饰器&#xff0c;是一个接收函数 func、返回封装后的函数 wr…

计算机Java项目|基于SpringBoot的农商对接系统的设计与实现

作者主页&#xff1a;编程指南针 作者简介&#xff1a;Java领域优质创作者、CSDN博客专家 、CSDN内容合伙人、掘金特邀作者、阿里云博客专家、51CTO特邀作者、多年架构师设计经验、多年校企合作经验&#xff0c;被多个学校常年聘为校外企业导师&#xff0c;指导学生毕业设计并参…

基于Shader实现的UGUI描边解决方案遇到的bug

原文链接&#xff1a;https://www.cnblogs.com/GuyaWeiren/p/9665106.html 使用这边文章介绍的描边解决方案时遇到了一些问题&#xff0c;就是文字的描边经常会变粗&#xff0c;虽然有的时候也可以正常显示描边&#xff0c;但是运行一会儿描边就不正常了&#xff0c;而且不正常…

【数据分享】《新疆省统计年鉴》(2000-2022)

而今天要限时免费分享的数据就是2000-2022年间出版的《新疆省统计年鉴》并以多格式提供免费下载。&#xff08;无需分享朋友圈即可获取&#xff09; 数据介绍 《新疆省统计年鉴》是记录新疆维吾尔自治区历年来社会经济发展情况的重要资料汇编&#xff0c;涵盖了从2000年至…

海外仓物流的最后一步至关重要!电商的复购、好评全都要靠它!

在跨境电商物流链中&#xff0c;尾程派送是直接影响消费者购物体验的关键环节。作为物流流程的最后一步&#xff0c;尾程派送的效率和准确性关系到商品能否及时、安全地送达客户手中。这不仅关乎消费者的满意度&#xff0c;也关乎电商企业的品牌形象和市场竞争力。尤其是在依托…

睡眠质量不好该怎么调理

1、运动&#xff1a;睡前多做些小运动&#xff0c;但不要做太剧烈的运动&#xff0c;比如跑步、散步、打太极拳等&#xff0c;适当的运动有益于睡眠。   2、远离扰乱睡眠的食物&#xff1a;不要喝咖啡、茶和其他刺激性的重口味的东西。睡前最好喝一杯牛奶或温水&#xff0c;这…

字节序大小端

概述 1. MSB、LSB2. 最高有效字节、最低有效字节3. 大小端4. 如何判断本机大小端5. 大小端转换 1. MSB、LSB 以整数“157”为例 MSB &#xff1a;单个字节中的最高位 2^7 128 LSB &#xff1a;单个字节中的最低位 2^0 0 2. 最高有效字节、最低有效字节 以整形“0x0102030…

极速闪存启动:SD与SPI模式的智能初始化指南

最近很多客户朋友在询问我们 CS 创世 SD NAND 能不能使用 SPI 接口&#xff0c;两者使用起来有何区别&#xff0c;下面为大家详细解答。 SD MODE: CS 创世 SD NAND 支持 SD 模式和 SPI 模式&#xff0c;SD NAND 默认为 SD 模式&#xff0c;上电后&#xff0c;其初始化过程如下…

【MySQL】5.0 入门学习(五)——MySQL源码了解及MySQL初始化设置

1.0 MySQL源码目录主要包括&#xff1a;客户端代码、服务端代码、测试工具、其他库文件。当然&#xff0c;看懂源代码得有一定的C语言基础。 image image.gif ​ BUILD&#xff1a;各种平台的编译脚本&#xff0c;可以用来制作各平台的二进制版本 client&#xff1a;客户端目录…

推荐编译器插件:Fitten Code 更快更好的AI助手

✨✨ 欢迎大家来访Srlua的博文&#xff08;づ&#xffe3;3&#xffe3;&#xff09;づ╭❤&#xff5e;✨✨ &#x1f31f;&#x1f31f; 欢迎各位亲爱的读者&#xff0c;感谢你们抽出宝贵的时间来阅读我的文章。 我是Srlua小谢&#xff0c;在这里我会分享我的知识和经验。&am…

【springboot】自定义starter

自定义一个starter&#xff0c;实现获取系统和程序信息。 0. 项目结构 org.springframework.boot.autoconfigure.AutoConfiguration.imports 文件是用来加载自动配置类的&#xff0c;该文件必须放在META-INF/spring/目录下。 1. 创建项目 创建一个普通的maven项目&#xff0c;使…

在线翻译工具分享,这三款值得收藏

作为经常需要处理各种文件的人&#xff0c;我的英语又不是很好&#xff0c;但是文件中不乏需要翻译的英文PDF文件。在翻译工具的选择上&#xff0c;我尝试过不少&#xff0c;今天就来跟大家分享一下我使用过的三款工具翻译PDF文件时的体验感。 一、福昕翻译在线 网址&#xf…

GEC6818开发板的学习

1、开发板的简介 首先连接 开发板与电脑,需电脑安装串口驱动:例CH340 2、开发板的特性: 像素:800*480Pix分辨率:高,宽两个维度的像素点数目开发板色深为32位一个像素点占4个字节:分别为灰度保留位、RGB三原色各占一位3、为什么要内存映射 虽然LCD设备本质上也可以看作…

C#使用Modbus TCP通讯PLC,实现读写寄存器

一、创建一个Moudbus类&#xff0c;引入NModbus和Modbus这两个包 #region ModbusTCPpublic class NmodbusTcpHelper{// 静态成员变量&#xff0c;用于存储TcpClient实例private static TcpClient tcpClient null;// 静态成员变量&#xff0c;用于存储ModbusIpMaster实例privat…