Vue通過js控制css變量一鍵修改全局樣式的完整過程
背景
最近接到一個(gè)需求:在Vue(版本是2.x)頁面有個(gè)控制字體大小的選項(xiàng),修改后全局表格內(nèi)的字體會(huì)變化,且要新打開的頁面也會(huì)變化。
需求整理
- 在頁面上方設(shè)置一個(gè)修改布局大小的選項(xiàng)
- 點(diǎn)擊后會(huì)出現(xiàn)滾動(dòng)條,字體范圍為9到23px,默認(rèn)值為14px
- 拖動(dòng)字體滾動(dòng)條時(shí),頁面表格字體會(huì)隨著滾動(dòng)條值的變化而變化
- 用戶每次打開頁面時(shí)會(huì)讀取上次設(shè)置的字體大小
實(shí)現(xiàn)效果如圖:


本文章會(huì)記錄完成這個(gè)需求的思考過程,如果需要直接看代碼請(qǐng)拉至"代碼實(shí)現(xiàn)"章節(jié)。
系統(tǒng)現(xiàn)狀
技術(shù)框架:vue2+ElementUI,基于若依管理系統(tǒng)的二次開發(fā)系統(tǒng)
實(shí)現(xiàn)思路
ElementUI有個(gè)控制全局大小的size變量,只要使用的控件不添加size屬性,就會(huì)讀取全局的size變量進(jìn)行相應(yīng)大小的展示。- 但該設(shè)置只能設(shè)置為預(yù)設(shè)的4種大小(
default,meduim,small,mini)無法滿足需求中的設(shè)置字體大小為指定像素(px)。
- 但該設(shè)置只能設(shè)置為預(yù)設(shè)的4種大小(
- 通過修改全局樣式
scss樣式文件,需要在每個(gè)class聲明fontSize樣式
由上述思考后,決定采用方法2,但如果我用js每次都更新一大堆class中的fontSize,那太痛苦了,如果使用到的class中fontSize屬性能動(dòng)態(tài)讀取同一個(gè)變量,用戶修改該變量的值后引用該變量的class值都能改變,那我只用js修改一個(gè)變量即可,那有沒有這種方法呢?
css如何讀取某個(gè)變量
CSS變量(也稱為 CSS 自定義屬性)允許你在 CSS 中定義可重用的值。具體特點(diǎn)如下:定義:通過
--前綴命名,在:root或其他選擇器(如其他的class)中定義。
使用:通過var()函數(shù)在CSS屬性值中引用。
作用:提高樣式的可維護(hù)性和一致性,便于全局修改樣式。
/* 可由js控制的css傳參*/
:root {
--table-font-size: 14px;
--ux-grid-height: 31px;
}
/* 使用var引用--table-font-size和--ux-grid-height變量,后文的ux-grid使用了.elx-table樣式類*/
.elx-table.size--default {
font-size: var(--table-font-size);
height: var(--ux-grid-height);
padding-top: 2px;
/* 使用var引用--table-font-size變量*/
.el-button {
font-size: var(--table-font-size);
/* 結(jié)合calc和min函數(shù)一起使用*/
padding-top: min(calc(var(--ux-grid-height) - var(--table-font-size) - 13px), 20px);
}
}
既然css已經(jīng)提供了這種方法,那么最后一個(gè)小需求實(shí)現(xiàn)起來就更容易了:每次修改字體大小時(shí)觸發(fā)值改變事件,修改字體大小后將值保存在cookie和vuex中,用戶每次登陸時(shí)獲取cookie中的值,如果沒有值則默認(rèn)取:root的大小。
代碼實(shí)現(xiàn)
<template>
<el-tooltip content="調(diào)整字體" effect="dark" placement="bottom">
<el-popover class="right-menu-item hover-effect"
placement="bottom"
width="200"
trigger="click">
<el-slider :min="9" :max="23" v-model="fontSize" :format-tooltip="(value)=>value + '像素'" @change="changeFontSize"></el-slider>
<i slot="reference" class="el-icon-s-operation"></i>
</el-popover>
</el-tooltip>
</template>
created() {
this.fontSize = parseFloat(this.$store.state.app.tableFontSize)
this.updateCssSize(CSS_FONT_SIZE_NAME, this.fontSize)
},
data() {
return {
fontSize: 0
}
},
/**
* 值變化事件
*/
methods: {
/** 修改字體大小事件*/
changeFontSize(amount) {
this.updateCssSize(CSS_FONT_SIZE_NAME, amount)
// 保存到vuex中
this.$store.dispatch('app/setTableFontSize', amount)
// 表格的行高為fontSize+17,讓行高隨著字體的變化而變化
let uxGridHeight = '--ux-grid-height'
this.updateCssSize(uxGridHeight, amount + 17)
this.refreshView()
},
/** 修改樣式屬性值 */
updateCssSize(prop, value) {
const newVal = parseFloat(value)
document.documentElement.style.setProperty(prop, newVal + 'px');
},
/** 重新加載已打開的其他標(biāo)簽頁*/
refreshView() {
this.$store.dispatch('tagsView/delAllCachedViews', this.$route)
const { fullPath } = this.$route
this.$nextTick(() => {
this.$router.replace({
path: '/redirect' + fullPath
})
})
}
}
// 在vuex中的代碼
const mutations = {
TOGGLE_SIDEBAR: state => {
SET_TABLE_FONT_SIZE: (state, fontSize) => {
state.tableFontSize = fontSize
Cookies.set('tableFontSize', fontSize)
}
}
}
// 在app.js中的代碼
const state = {
tableFontSize: Cookies.get('tableFontSize') || window.getComputedStyle(document.documentElement).getPropertyValue('--table-font-size')
}
總結(jié)
作為一個(gè)常年寫java代碼的開發(fā)人員真的感覺css出神入化,之前還停留在css只能寫靜態(tài)屬性的印象中,沒想到已經(jīng)這么方便
到此這篇關(guān)于Vue通過js控制css變量一鍵修改全局樣式的文章就介紹到這了,更多相關(guān)Vue通過js控制css變量?jī)?nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
解決vuejs 使用value in list 循環(huán)遍歷數(shù)組出現(xiàn)警告的問題
今天小編就為大家分享一篇解決vuejs 使用value in list 循環(huán)遍歷數(shù)組出現(xiàn)警告的問題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧2018-09-09
vue使用拖拽方式創(chuàng)建結(jié)構(gòu)樹
這篇文章主要為大家詳細(xì)介紹了vue使用拖拽方式創(chuàng)建結(jié)構(gòu)樹,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2021-10-10
element修改form的el-input寬度,el-select寬度的方法實(shí)現(xiàn)
有時(shí)候像form表單這樣,頁面的input、select等寬度不一定會(huì)是一樣的,可能有些長,有些短,本文就介紹了如何element修改form的el-input寬度,el-select寬度的方法實(shí)現(xiàn),感興趣的可以了解一下2022-02-02
簡(jiǎn)單聊聊vue2.x的$attrs和$listeners
vue組件之間的通信方式有很多種,props/emit,event bus,vuex,provide/inject等,還有一種通信方式就是$attrs和$listeners,下面這篇文章主要給大家介紹了關(guān)于vue2.x中$attrs和$listeners的相關(guān)資料,需要的朋友可以參考下2022-03-03
Vue使用distpicker插件實(shí)現(xiàn)省市級(jí)下拉框三級(jí)聯(lián)動(dòng)
這篇文章主要介紹了Vue使用distpicker插件實(shí)現(xiàn)省市級(jí)下拉框三級(jí)聯(lián)動(dòng),比如通過JSON文件生成對(duì)應(yīng)的區(qū)域下拉框,element-china-are插件,包括distpicker插件,通過代碼講解如何使用distpicker插件實(shí)現(xiàn)省市級(jí)三聯(lián)跳動(dòng),需要的朋友可以參考下2023-02-02
vue中this.$router.push()路由傳值和獲取的兩種常見方法匯總
這篇文章主要介紹了vue中this.$router.push()路由傳值和獲取的兩種常見方法,本文結(jié)合示例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2022-12-12
vue使用@scroll監(jiān)聽滾動(dòng)事件時(shí),@scroll無效問題的解決方法詳解
這篇文章主要介紹了vue使用@scroll監(jiān)聽滾動(dòng)事件時(shí),@scroll無效問題的解決方法,結(jié)合實(shí)例形式分析了@scroll監(jiān)聽滾動(dòng)事件無效問題的原因及相應(yīng)的解決方法,需要的朋友可以參考下2019-10-10
Vue+ElementUI啟動(dòng)vue卡死的問題及解決
這篇文章主要介紹了Vue+ElementUI啟動(dòng)vue卡死的問題及解決,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2024-03-03

