我的编程空间,编程开发者的网络收藏夹
学习永远不晚

vue-element如何实现动态换肤存储

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

vue-element如何实现动态换肤存储

需要实现的效果

选择颜色块或者颜色选择器切换网站主题色,选择主题后保存到本地,下次打开页面是缓存的主题色

在这里插入图片描述

在这里插入图片描述

在这里插入图片描述

在这里插入图片描述

原理

根据ElementUI官网的自定义主题,新建一个样式文件:element-variables(或者参考官网生成),在文件中写入:


$--color-primary: #409eff;
$--color-success: #67c23a;



$--font-path: '~element-ui/lib/theme-chalk/fonts';

@import "~element-ui/packages/theme-chalk/class="lazy" data-src/index";

//主要的一步:
:export {
  theme: $--color-primary;
}

在项目store文件夹中新建theme.js文件,用于存储主题色变化状态,代码如下

//引入element-variables文件,文件路径根据自己项目存放位置来
import variables from '@/assets/css/element-variables.scss'
const state = {
  theme: variables.theme
}

const getters = {
	theme: function (state) {
		return state.theme;
	}
};

const mutations = {
  CHANGE_SETTING: (state, { key, value }) => {
    // eslint-disable-next-line no-prototype-builtins
    if (state.hasOwnProperty(key)) {
      state[key] = value
    }
  }
}

const actions = {
  changeSetting({ commit }, data) {
    commit('CHANGE_SETTING', data)
  }
}

export default {
  namespaced: true,
  state,
  mutations,
  actions,
  getters
}

然后在store.js中引入theme.js

import Vue from 'vue';
import Vuex from 'vuex';
import theme from '@/store/theme.js';
Vue.use(Vuex);
export default new Vuex.Store({
	modules: {
		theme
	}
});

切换主图部分template代码如下:

<el-form-item label="可选主题" label-width="140px">
	<div class="color-box">
		<div 
			v-for="(item,i) in themeArr"
			:key="i"
			:class="['color-item',theme===item?'active':'']"
			:style="{backgroundColor:item}"
			@click="changeTheme(item)">
			<span v-if="theme===item" class="iconfont icon-f-right"></span>	
		</div>
	</div>
</el-form-item>
<el-form-item label="自定义主题" label-width="140px">
	<el-color-picker
		v-model="theme"
		:predefine="['#409EFF', '#1890ff', '#304156','#212121','#11a983', '#13c2c2', '#6959CD', '#f5222d', ]"
		class="theme-picker"
		popper-class="theme-picker-dropdown"
	/>
</el-form-item>

script代码

const version = require('element-ui/package.json').version // element-ui version from node_modules
const ORIGINAL_THEME = '#409EFF' // default color

export default {
	data() {
		return {
			themeArr: ['#409EFF','#202225','#F56C6C','#FFAB42','#17a969','#888C8F'],
			chalk: '',
			theme: ORIGINAL_THEME,
		}
	},
	watch: {
		//异步监听theme的变化
		async theme(val,oldVal) {
	      if (typeof val !== 'string') return
	      const themeCluster = this.getThemeCluster(val.replace('#', ''))
	      const originalCluster = this.getThemeCluster(oldVal.replace('#', ''))
	      // console.log(themeCluster, originalCluster)
	
	      const $message = this.$message({
	        message: '  Compiling the theme',
	        customClass: 'theme-message',
	        type: 'success',
	        duration: 0,
	        iconClass: 'el-icon-loading'
	      })
	
	      const getHandler = (variable, id) => {
	        return () => {
	          const originalCluster = this.getThemeCluster(ORIGINAL_THEME.replace('#', ''))
	          const newStyle = this.updateStyle(this[variable], originalCluster, themeCluster)
	
	          let styleTag = document.getElementById(id)
	          if (!styleTag) {
	            styleTag = document.createElement('style')
	            styleTag.setAttribute('id', id)
	            document.head.appendChild(styleTag)
	          }
	          styleTag.innerText = newStyle
	        }
	      }
	
		  //  初次进入或刷新时动态加载CSS文件
	      if (!this.chalk) {
	        const url = `https://unpkg.com/element-ui@${version}/lib/theme-chalk/index.css`
	        await this.getCSSString(url, 'chalk')
	      }
	
	      const chalkHandler = getHandler('chalk', 'chalk-style')
	
	      chalkHandler()
	
	      const styles = [].slice.call(document.querySelectorAll('style'))
	        .filter(style => {
	          const text = style.innerText
	          return new RegExp(oldVal, 'i').test(text) && !/Chalk Variables/.test(text)
	        })
	      styles.forEach(style => {
	        const { innerText } = style
	        if (typeof innerText !== 'string') return
	        style.innerText = this.updateStyle(innerText, originalCluster, themeCluster)
	      })
	
	       // 将修改的主题保存到本地,下次打开页面是保存的主题色
	    	localStorage.setItem('theme',val)
	    	//调用vuex中改变主题色方法
			this.$store.dispatch('theme/changeSetting', {
		        key: 'theme',
		        value: val
		    })
	
	      $message.close()
	
	    },
	},
	methods: {
		updateStyle(style, oldCluster, newCluster) {
	      let newStyle = style
	      oldCluster.forEach((color, index) => {
	        newStyle = newStyle.replace(new RegExp(color, 'ig'), newCluster[index])
	      })
	      return newStyle
	    },
	    getCSSString(url, variable) {
	      return new Promise(resolve => {
	        const xhr = new XMLHttpRequest()
	        xhr.onreadystatechange = () => {
	          if (xhr.readyState === 4 && xhr.status === 200) {
	            this[variable] = xhr.responseText.replace(/@font-face{[^}]+}/, '')
	            resolve()
	          }
	        }
	        xhr.open('GET', url)
	        xhr.send()
	      })
	    },
	    getThemeCluster(theme) {
	      const tintColor = (color, tint) => {
	        let red = parseInt(color.slice(0, 2), 16)
	        let green = parseInt(color.slice(2, 4), 16)
	        let blue = parseInt(color.slice(4, 6), 16)
	
	        if (tint === 0) { // when primary color is in its rgb space
	          return [red, green, blue].join(',')
	        } else {
	          red += Math.round(tint * (255 - red))
	          green += Math.round(tint * (255 - green))
	          blue += Math.round(tint * (255 - blue))
	
	          red = red.toString(16)
	          green = green.toString(16)
	          blue = blue.toString(16)
	
	          return `#${red}${green}${blue}`
	        }
	      }
	
	      const shadeColor = (color, shade) => {
	        let red = parseInt(color.slice(0, 2), 16)
	        let green = parseInt(color.slice(2, 4), 16)
	        let blue = parseInt(color.slice(4, 6), 16)
	
	        red = Math.round((1 - shade) * red)
	        green = Math.round((1 - shade) * green)
	        blue = Math.round((1 - shade) * blue)
	
	        red = red.toString(16)
	        green = green.toString(16)
	        blue = blue.toString(16)
	
	        return `#${red}${green}${blue}`
	      }
	
	      const clusters = [theme]
	      for (let i = 0; i <= 9; i++) {
	        clusters.push(tintColor(theme, Number((i / 10).toFixed(2))))
	      }
	      clusters.push(shadeColor(theme, 0.1))
	      return clusters
	    },
		// 颜色块点击切换主题事件
		changeTheme(item) {
		  this.theme = item
	      this.$store.dispatch('theme/changeSetting', {
	        key: 'theme',
	        value: item
	      })
		},
	},
	mounted() {
		//从本地存储获取保存的主题
		if(localStorage.getItem("theme")) {
			this.theme = localStorage.getItem("theme")
			this.$store.dispatch('theme/changeSetting', {
		        key: 'theme',
		        value: this.theme
		    })
		}
	}
}

颜色块选择样式style:

.color-box {
	display: flex;
	flex-wrap: wrap;
	.color-item {
		position: relative;
		margin: 2px;
		width: 34px;
		height: 34px;
		border-radius: 3px;
		border: 2px solid transparent;
		cursor: pointer;
		span {
			position: absolute;
			right: 0;
			top: 0;
			width: 15px;
			height: 15px;
			background-color: #000;
			color: #fff;
			font-size: 10px;
    	text-align: center;
		}
		&.active {
			border-color: #000;
		}
	}
}

最后感谢:vue-element-admin作者 本实现方案借鉴于此

总结

以上为个人经验,希望能给大家一个参考,也希望大家多多支持编程网。

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

vue-element如何实现动态换肤存储

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

vue-element如何实现动态换肤存储

这篇文章主要介绍了vue-element如何实现动态换肤存储问题,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
2023-05-17

vue如何实现换肤功能

今天小编给大家分享一下vue如何实现换肤功能的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一下吧。首先我们用vue-element-
2023-07-04

Vue router动态路由如何实现

本文小编为大家详细介绍“Vue router动态路由如何实现”,内容详细,步骤清晰,细节处理妥当,希望这篇“Vue router动态路由如何实现”文章能帮助大家解决疑惑,下面跟着小编的思路慢慢深入,一起来学习新知识吧。实现思路思路其实很简单
2023-07-05

C语言如何实现动态开辟存储杨辉三角

本文小编为大家详细介绍“C语言如何实现动态开辟存储杨辉三角”,内容详细,步骤清晰,细节处理妥当,希望这篇“C语言如何实现动态开辟存储杨辉三角”文章能帮助大家解决疑惑,下面跟着小编的思路慢慢深入,一起来学习新知识吧。问题引入杨辉三角相必大家并
2023-06-29

Vue如何实现动态路由导航

这篇文章主要介绍“Vue如何实现动态路由导航”的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇“Vue如何实现动态路由导航”文章能帮助大家解决问题。1、导航守卫“导航” 表示路由正在发生改变正如其名,vu
2023-07-05

VUE路由动态加载如何实现

这篇文章主要讲解了“VUE路由动态加载如何实现”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“VUE路由动态加载如何实现”吧!首先新建vue工程,一般我们不会特殊处理路由,但当项目页面越来越多
2023-07-04

如何实现IP动态切换bat脚本

这篇文章将为大家详细讲解有关如何实现IP动态切换bat脚本,小编觉得挺实用的,因此分享给大家做个参考,希望大家阅读完这篇文章后可以有所收获。新建"IP切换脚本.bat"文件,将下列代码复制进去,保存,并加入启动项,这样每次开机的时候选择你的
2023-06-08

Vue如何实现异步动态加载块

本篇内容介绍了“Vue如何实现异步动态加载块”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!首先定义组件为异步加载define([jquery
2023-07-04

Django状态保持搭配与存储如何实现

本篇内容主要讲解“Django状态保持搭配与存储如何实现”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“Django状态保持搭配与存储如何实现”吧!为什么需要状态保持因为通常浏览器请求服务器是 无
2023-06-30

SpringBoot+Mybatis如何实现动态数据源切换

这篇文章主要介绍了SpringBoot+Mybatis如何实现动态数据源切换,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇文章之后大有收获,下面让小编带着大家一起了解一下。springboot是什么springboot一种全
2023-06-14

Vue如何实现利用vuex永久储存数据

这篇文章主要介绍了Vue如何实现利用vuex永久储存数据问题,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
2023-05-16

如何实现CentOS7使用Docker Overlay2存储驱动

这篇文章主要讲解了“如何实现CentOS7使用Docker Overlay2存储驱动”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“如何实现CentOS7使用Docker Overlay2存储
2023-06-10

Vue如何实现动态渲染input输入框

本文小编为大家详细介绍“Vue如何实现动态渲染input输入框”,内容详细,步骤清晰,细节处理妥当,希望这篇“Vue如何实现动态渲染input输入框”文章能帮助大家解决疑惑,下面跟着小编的思路慢慢深入,一起来学习新知识吧。问题描述就比如现在
2023-07-04

vue如何实现select动态控制input禁用

这篇文章主要介绍“vue如何实现select动态控制input禁用”,在日常操作中,相信很多人在vue如何实现select动态控制input禁用问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”vue如何实现se
2023-07-04

C语言如何实现动态内存分配

今天小编给大家分享一下C语言如何实现动态内存分配的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一下吧。C语言动态内存分配的详解1.为
2023-06-16

编程热搜

目录