forked from aixan/RuoYi-Vue
		
	
		
			
	
	
		
			175 lines
		
	
	
		
			4.9 KiB
		
	
	
	
		
			Vue
		
	
	
	
	
	
		
		
			
		
	
	
			175 lines
		
	
	
		
			4.9 KiB
		
	
	
	
		
			Vue
		
	
	
	
	
	
|   | <template> | ||
|  |   <el-color-picker | ||
|  |     v-model="theme" | ||
|  |     :predefine="['#409EFF', '#1890ff', '#304156','#212121','#11a983', '#13c2c2', '#6959CD', '#f5222d', ]" | ||
|  |     class="theme-picker" | ||
|  |     popper-class="theme-picker-dropdown" | ||
|  |   /> | ||
|  | </template> | ||
|  | 
 | ||
|  | <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 { | ||
|  |       chalk: '', // content of theme-chalk css
 | ||
|  |       theme: '' | ||
|  |     } | ||
|  |   }, | ||
|  |   computed: { | ||
|  |     defaultTheme() { | ||
|  |       return this.$store.state.settings.theme | ||
|  |     } | ||
|  |   }, | ||
|  |   watch: { | ||
|  |     defaultTheme: { | ||
|  |       handler: function(val, oldVal) { | ||
|  |         this.theme = val | ||
|  |       }, | ||
|  |       immediate: true | ||
|  |     }, | ||
|  |     async theme(val) { | ||
|  |       const oldVal = this.chalk ? this.theme : ORIGINAL_THEME | ||
|  |       if (typeof val !== 'string') return | ||
|  |       const themeCluster = this.getThemeCluster(val.replace('#', '')) | ||
|  |       const originalCluster = this.getThemeCluster(oldVal.replace('#', '')) | ||
|  | 
 | ||
|  |       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 | ||
|  |         } | ||
|  |       } | ||
|  | 
 | ||
|  |       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) | ||
|  |       }) | ||
|  | 
 | ||
|  |       this.$emit('change', 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 | ||
|  |     } | ||
|  |   } | ||
|  | } | ||
|  | </script> | ||
|  | 
 | ||
|  | <style> | ||
|  | .theme-message, | ||
|  | .theme-picker-dropdown { | ||
|  |   z-index: 99999 !important; | ||
|  | } | ||
|  | 
 | ||
|  | .theme-picker .el-color-picker__trigger { | ||
|  |   height: 26px !important; | ||
|  |   width: 26px !important; | ||
|  |   padding: 2px; | ||
|  | } | ||
|  | 
 | ||
|  | .theme-picker-dropdown .el-color-dropdown__link-btn { | ||
|  |   display: none; | ||
|  | } | ||
|  | </style> |