广告
返回顶部
首页 > 资讯 > 前端开发 > JavaScript >vue-element如何实现动态换肤存储
  • 644
分享到

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

vueelementelement换肤存储动态换肤存储 2023-05-17 14:05:17 644人浏览 八月长安
摘要

目录需要实现的效果原理总结需要实现的效果 选择颜色块或者颜色选择器切换网站主题色,选择主题后保存到本地,下次打开页面是缓存的主题色 原理 根据ElementUI官网的自定义主

需要实现的效果

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

在这里插入图片描述

在这里插入图片描述

在这里插入图片描述

在这里插入图片描述

原理

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


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



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

@import "~element-ui/packages/theme-chalk/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作者 本实现方案借鉴于此

总结

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

--结束END--

本文标题: vue-element如何实现动态换肤存储

本文链接: https://www.lsjlt.com/news/211258.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

本篇文章演示代码以及资料文档资料下载

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

下载Word文档
猜你喜欢
  • vue-element如何实现动态换肤存储
    目录需要实现的效果原理总结需要实现的效果 选择颜色块或者颜色选择器切换网站主题色,选择主题后保存到本地,下次打开页面是缓存的主题色 原理 根据ElementUI官网的自定义主...
    99+
    2023-05-17
    vue element element换肤存储 动态换肤存储
  • vue+element实现动态换肤的示例代码
    有时候一个项目的主题并不能满足所有人的审美,这时候换肤功能就很友好,本项目基于vue+element实现后台管理项目的换肤功能 1.创建换肤组件 <template>...
    99+
    2022-11-12
  • vue如何实现换肤功能
    今天小编给大家分享一下vue如何实现换肤功能的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一下吧。首先我们用vue-element-...
    99+
    2023-07-04
  • vue如何实现动态的选中状态切换效果
    目录动态选中状态切换效果vue状态转换状态展示动态选中状态切换效果  HTML中的内容为以下。 <ul class="list">      <li v-...
    99+
    2022-11-13
  • vue动态组件如何实现选项卡切换效果
    这篇文章主要介绍了vue动态组件如何实现选项卡切换效果,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇文章之后大有收获,下面让小编带着大家一起了解一下。具体内容如下导航按钮:<div&n...
    99+
    2022-10-19
  • C语言如何实现动态开辟存储杨辉三角
    本文小编为大家详细介绍“C语言如何实现动态开辟存储杨辉三角”,内容详细,步骤清晰,细节处理妥当,希望这篇“C语言如何实现动态开辟存储杨辉三角”文章能帮助大家解决疑惑,下面跟着小编的思路慢慢深入,一起来学习新知识吧。问题引入杨辉三角相必大家并...
    99+
    2023-06-29
  • Vue router动态路由如何实现
    本文小编为大家详细介绍“Vue router动态路由如何实现”,内容详细,步骤清晰,细节处理妥当,希望这篇“Vue router动态路由如何实现”文章能帮助大家解决疑惑,下面跟着小编的思路慢慢深入,一起来学习新知识吧。实...
    99+
    2023-07-05
  • 如何实现IP动态切换bat脚本
    这篇文章将为大家详细讲解有关如何实现IP动态切换bat脚本,小编觉得挺实用的,因此分享给大家做个参考,希望大家阅读完这篇文章后可以有所收获。新建"IP切换脚本.bat"文件,将下列代码复制进去,保存,并加入启动项,这样每...
    99+
    2023-06-08
  • vue中如何使用动态组件实现选项卡切换效果
    这篇文章主要为大家展示了“vue中如何使用动态组件实现选项卡切换效果”,内容简而易懂,条理清晰,希望能够帮助大家解决疑惑,下面让小编带领大家一起研究并学习一下“vue中如何使用动态组件实现选项卡切换效果”这...
    99+
    2022-10-19
  • Vue如何实现动态路由导航
    这篇文章主要介绍“Vue如何实现动态路由导航”的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇“Vue如何实现动态路由导航”文章能帮助大家解决问题。1、导航守卫“导航” 表示路由正在发生改变正如其名,vu...
    99+
    2023-07-05
  • Vue动态如何实现评分效果
    小编给大家分享一下Vue动态如何实现评分效果,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让我们一起去了解一下吧!具体内容如下1.图片分为三种on:ha...
    99+
    2022-10-19
  • VUE路由动态加载如何实现
    这篇文章主要讲解了“VUE路由动态加载如何实现”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“VUE路由动态加载如何实现”吧!首先新建vue工程,一般我们不会特殊处理路由,但当项目页面越来越多...
    99+
    2023-07-04
  • SpringBoot+Mybatis如何实现动态数据源切换
    这篇文章主要介绍了SpringBoot+Mybatis如何实现动态数据源切换,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇文章之后大有收获,下面让小编带着大家一起了解一下。springboot是什么springboot一种全...
    99+
    2023-06-14
  • Django状态保持搭配与存储如何实现
    本篇内容主要讲解“Django状态保持搭配与存储如何实现”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“Django状态保持搭配与存储如何实现”吧!为什么需要状态保持因为通常浏览器请求服务器是 无...
    99+
    2023-06-30
  • Vue如何实现异步动态加载块
    本篇内容介绍了“Vue如何实现异步动态加载块”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!首先定义组件为异步加载define(['j...
    99+
    2023-07-04
  • scrapy如何实现ip动态代理与更换ip
    这篇文章将为大家详细讲解有关scrapy如何实现ip动态代理与更换ip,小编觉得挺实用的,因此分享给大家做个参考,希望大家阅读完这篇文章后可以有所收获。请自行准备一个ip代理的平台例如我用的这个平台,每次提取10个ip从上面可以看到数据格式...
    99+
    2023-06-14
  • Vue如何实现利用vuex永久储存数据
    目录Vue用vuex永久储存数据基于vuex的数据持久化问题存在问题此问题映射在本项目上的体现解决方法:使用vuex的数据持久化总结Vue用vuex永久储存数据 首先需要在 vue项...
    99+
    2023-05-16
    Vue储存数据 vuex永久储存数据 vuex储存数据
  • 如何实现CentOS7使用Docker Overlay2存储驱动
    这篇文章主要讲解了“如何实现CentOS7使用Docker Overlay2存储驱动”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“如何实现CentOS7使用Docker Overlay2存储...
    99+
    2023-06-10
  • Vue如何实现动态渲染input输入框
    本文小编为大家详细介绍“Vue如何实现动态渲染input输入框”,内容详细,步骤清晰,细节处理妥当,希望这篇“Vue如何实现动态渲染input输入框”文章能帮助大家解决疑惑,下面跟着小编的思路慢慢深入,一起来学习新知识吧。问题描述就比如现在...
    99+
    2023-07-04
  • vue如何实现select动态控制input禁用
    这篇文章主要介绍“vue如何实现select动态控制input禁用”,在日常操作中,相信很多人在vue如何实现select动态控制input禁用问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”vue如何实现se...
    99+
    2023-07-04
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作