In diesem Artikel lernen Sie CSS-Variablen kennen, führen in die Verwendung von CSS-Variablen ein und erfahren, wie Sie CSS-Variablen geschickt einsetzen, um Ihr CSS spannender und Ihr Projekt cooler zu machen!
CSS-Variablen werden auch „benutzerdefinierte CSS-Eigenschaften“ genannt. Warum wird dieses Ding, das nur wenige Leute verwenden, plötzlich erwähnt? Da ich kürzlich meine persönliche offizielle Website neu erstellt habe, weiß ich nicht, warum ich plötzlich gerne CSS-Variablen verwende. Vielleicht bewundere ich sie wegen ihres verborgenen Charmes. Wenn ich darüber spreche, warum Variablen in CSS verwendet werden, hier ist ein Beispiel, ich denke, jeder wird es auf einen Blick verstehen.
/* 不使用CSS变量 */ .title { background-color: red; } .desc { background-color: red; } /* 使用CSS变量 */ :root { --bg-color: red; } .title { background-color: var(--bg-color); } .desc { background-color: var(--bg-color); }
Nachdem Sie dies gelesen haben, haben Sie vielleicht das Gefühl, dass die Menge an Code, die
CSS-Variablen verwendet, etwas zu viel ist, aber haben Sie jemals daran gedacht, dass der böse Planungsbruder und das Design-Mädchen eines Tages plötzlich sagten, dass sie einen machen wollen? Hautveränderungsfunktion. Der üblichen Denkweise zufolge werden einige Schüler wahrscheinlich eine entsprechende Neues Farbthema
-CSS-Datei entsprechend dem Standardfarbthema
hinzufügen. Es wäre mühsam, bei jeder neuen Anforderung mehrere Sätze von Themenfarben gleichzeitig beizubehalten. Hier kommen 默认颜色主题
增加一份对照的新颜色主题
CSS文件。这样每次新增需求都同时维护几套主题颜色多麻烦啊。
此时CSS变量就派上用场了,提前跟设计小姐姐规范好各种需要变换的颜色并通过CSS变量进行定义,通过JS批量操作这些定义好的CSS变量即可。这也是变换主题颜色的一种解决方案之一,好处在于只需写一套CSS代码。
["red", "blue", "green"].forEach(v => { const btn = document.getElementById(`${v}-theme-btn`); btn.addEventListener("click", () => document.body.style.setProperty("--bg-color", v)); });
在此总结下CSS使用变量的好处:
<ul><li>减少样式代码的重复性<li>增加样式代码的扩展性<li>提高样式代码的灵活性<li>增多一种CSS与JS的通讯方式<li>不用深层遍历DOM改变某个样式可能有些同学会问,Sass和Less早就实现了变量这个特性,何必再多此一举呢。可是细想一下,CSS变量对比Sass和Less的变量,又有它的过人之处。
<ul><li>浏览器原生特性,无需经过任何转译就可直接运行<li>DOM对象一员,极大便利了CSS与JS之间的联系本来打算用一半篇幅讲述CSS变量的规范和用法,但是网上一搜一大把就感觉没必要了,贴上阮一峰老师写的教程《CSS变量教程》。同时笔者也对CSS变量的细节地方进行一个整理,方便大家记忆。
<ul><li>声明:--变量名
<li>读取:var(--变量名, 默认值)
<li>类型<ul><li>普通:只能用作属性值
不能用作属性名
<li>字符:与字符串拼接 "Hello, "var(--name)
<li>数值:使用calc()
与数值单位连用 var(--width) * 10px
<li>作用域<ul><li>范围:在当前元素块作用域
及其子元素块作用域
下有效<li>优先级别:内联样式 > ID选择器 > 类选择器 = 属性选择器 = 伪类选择器 > 标签选择器 = 伪元素选择器
接下来使用几个特别的场景展示CSS变量的魅力。还是那句话,一样东西有使用的场景,那自然就会有它的价值,那么用的人也会越来越多。
其实CSS变量有一个特别好用的场景,那就是结合List元素集合使用。如果不明白这是什么,请继续往下看。
以下所有演示代码基于vue文件,但HTML、CSS和JS分开书写,为了简化CSS的书写而使用Sass进行预处理,方便代码演示
条形加载条
一个条形加载条通常由几条线条组成,并且每条线条对应一个存在不同时延的相同动画,通过时间差运行相同的动画,从而产生加载效果。估计大部分的同学可能会把CSS代码写成以下这样。
<ul class="strip-loading flex-ct-x"> <li v-for="v in 6" :key="v"></li> </ul>
.loading { width: 200px; height: 200px; li { border-radius: 3px; width: 6px; height: 30px; background-color: #f66; animation: beat 1s ease-in-out infinite; & + li { margin-left: 5px; } &:nth-child(2) { animation-delay: 200ms; } &:nth-child(3) { animation-delay: 400ms; } &:nth-child(4) { animation-delay: 600ms; } &:nth-child(5) { animation-delay: 800ms; } &:nth-child(6) { animation-delay: 1s; } } }
分析代码发现,每个<li>
只是存在animation-delay
不同,而其余代码则完全相同,换成其他类似的List元素集合场景,那岂不是有10个<li>
就写10个:nth-child
。
显然这种方法不灵活也不容易封装成组件,如果能像JS那样封装成一个函数,并根据参数输出不同的样式效果,那就更棒了。说到这里,很明显就是为了铺垫CSS变量的开发技巧了。
对于HTML部分的修改,让每个<li>
拥有一个自己作用域下的CSS变量。对于CSS部分的修改,就需要分析哪些属性是随着index
CSS-Variablen zum Einsatz. Sie können die verschiedenen Farben, die geändert werden müssen, im Voraus über CSS-Variablen
<ul class="strip-loading flex-ct-x"> <li v-for="v in 6" :key="v" :style="`--line-index: ${v}`"></li> </ul>
--Variablenname
<li>Lesen: var(--Variablenname, Standardwert)
<li>Typ<ul><li>Normal: kann nur als Attributwert
verwendet werden und kann nicht als Attributname
<li>Zeichen: und verwendet werden String-Splicing "Hallo, "var(--name)
<li>numerische Werte: Verwenden Sie calc()
mit der numerischen Einheit var(- -width ) * 10px
<li>Bereich <ul><li>Bereich: innerhalb des aktuellen Elementblockbereichs
und seines Gültig im Bereich des untergeordneten Elementblocks
<li>Prioritätsstufe: Inline-Stil> Attributselektor = Pseudoklassenselektor> code>🎜 Als nächstes werden wir mehrere spezielle Szenarien verwenden, um den Charme von 🎜CSS-Variablen🎜 zu demonstrieren. Auch hier gilt: Wenn etwas ein Nutzungsszenario hat, hat es natürlich seinen Wert und immer mehr Menschen werden es nutzen. 🎜Verwendungsszenarien
🎜Tatsächlich gibt es für 🎜CSS-Variablen🎜 ein besonders nützliches Szenario, nämlich ihre Verwendung in Verbindung mit einer Sammlung von Listenelementen. Wenn Sie nicht wissen, was das ist, lesen Sie bitte weiter. 🎜.strip-loading {
width: 200px;
height: 200px;
li {
--time: calc((var(--line-index) - 1) * 200ms);
border-radius: 3px;
width: 6px;
height: 30px;
background-color: #f66;
animation: beat 1.5s ease-in-out var(--time) infinite;
& + li {
margin-left: 5px;
}
}
}
Nach dem Login kopierenNach dem Login kopieren🎜🎜Bar Loading Bar🎜🎜🎜Eine Bar Loading Bar besteht normalerweise aus mehreren Zeilen, und jede Zeile entspricht einer anderen Verzögerung. Die gleiche Animation ist Durchlaufen Sie die Zeitdifferenz, um einen Ladeeffekt zu erzeugen. Es wird geschätzt, dass die meisten Schüler CSS-Code wie folgt schreiben können. 🎜🎜🎜/* flex属性无效 */
.loading {
display: flex;
align-items: center;
flex: var(--line-index);
}
Nach dem Login kopierenNach dem Login kopieren<div class="heart-loading flex-ct-x">
<ul style="--line-count: 9;">
<li v-for="v in 9" :key="v" :class="`line-${v}`" :style="`--line-index: ${v}`"></li>
</ul>
</div>
Nach dem Login kopierenNach dem Login kopieren🎜 Nach der Analyse des Codes wurde festgestellt, dass sich jeder <li>
nur in der animation-delay
unterscheidet, während der Rest des Codes genau gleich ist Was würde passieren, wenn 10 <li>
vorhanden wären, anstatt 10 :nth-child
zu schreiben? 🎜🎜Offensichtlich ist diese Methode nicht flexibel und kann nicht einfach in Komponenten gekapselt werden. Es wäre sogar noch besser, wenn sie in eine Funktion wie JS gekapselt werden könnte und entsprechend den Parametern unterschiedliche Stileffekte ausgeben könnte. Allerdings dient es offensichtlich dazu, den Weg für die Entwicklung von 🎜CSS-Variablen🎜 zu ebnen. 🎜🎜Für die Änderung des HTML-Teils lassen Sie jeden <li>
eine 🎜CSS-Variable🎜 in seinem eigenen Bereich haben. Für die Änderung des CSS-Teils müssen Sie analysieren, welche Attribute sich regelmäßig ändern, wenn der index
zunimmt. Verwenden Sie 🎜CSS-Variablen🎜Ausdrücke, um die sich regelmäßig ändernden Teile zu ersetzen. 🎜<ul class="strip-loading flex-ct-x">
<li v-for="v in 6" :key="v" :style="`--line-index: ${v}`"></li>
</ul>
Nach dem Login kopierenNach dem Login kopieren.strip-loading {
width: 200px;
height: 200px;
li {
--time: calc((var(--line-index) - 1) * 200ms);
border-radius: 3px;
width: 6px;
height: 30px;
background-color: #f66;
animation: beat 1.5s ease-in-out var(--time) infinite;
& + li {
margin-left: 5px;
}
}
}
Nach dem Login kopierenNach dem Login kopieren代码中的变量--line-index
和--time
使每个<li>
拥有一个属于自己的作用域。例如第2个<li>
,--line-index
的值为2,--time
的计算值为200ms
,换成第3个<li>
后这两个值又会不同了。
这就是CSS变量的作用范围所致(在当前元素块作用域及其子元素块作用域下有效
),因此在.strip-loading
的块作用域下调用--line-index
是无效的。
/* flex属性无效 */
.loading {
display: flex;
align-items: center;
flex: var(--line-index);
}
Nach dem Login kopierenNach dem Login kopieren通过妙用CSS变量,也把CSS代码从29行
缩减到15行
,对于那些含有List元素集合越多的场景,效果就更明显。而且这样写也更加美观更加容易维护,某天说加载效果的时间差不明显,直接将calc((var(--line-index) - 1) * 200ms)
里的200ms
调整成400ms
即可。就无需对每个:nth-child(n)
进行修改了。
心形加载条
前段时间刷掘金看到陈大鱼头兄
的心形加载条,觉得挺漂亮的,很带感觉。
通过动图分析,发现每条线条的背景色和动画时延不一致,另外动画运行时的高度也不一致。细心的你可能还会发现,第1条和第9条的高度一致,第2条和第8条的高度一致,依次类推,得到高度变换相同类
的公式:对称index = 总数 + 1 - index
。
背景色使用了滤镜的色相旋转hue-rotate
函数,目的是为了使颜色过渡得更加自然;动画时延的设置和上面条形加载条
的设置一致。下面就用CSS变量根据看到的动图实现一番。
<div class="heart-loading flex-ct-x">
<ul style="--line-count: 9;">
<li v-for="v in 9" :key="v" :class="`line-${v}`" :style="`--line-index: ${v}`"></li>
</ul>
</div>
Nach dem Login kopierenNach dem Login kopieren.heart-loading {
width: 200px;
height: 200px;
ul {
display: flex;
justify-content: space-between;
width: 150px;
height: 10px;
}
li {
--Θ: calc(var(--line-index) / var(--line-count) * .5turn);
--time: calc((var(--line-index) - 1) * 40ms);
border-radius: 5px;
width: 10px;
height: 10px;
background-color: #3c9;
filter: hue-rotate(var(--Θ));
animation-duration: 1s;
animation-delay: var(--time);
animation-iteration-count: infinite;
}
.line-1,
.line-9 {
animation-name: line-move-1;
}
.line-2,
.line-8 {
animation-name: line-move-2;
}
.line-3,
.line-7 {
animation-name: line-move-3;
}
.line-4,
.line-6 {
animation-name: line-move-4;
}
.line-5 {
animation-name: line-move-5;
}
}
Nach dem Login kopieren一波操作后就有了下面的效果。和陈大鱼头兄
的心形加载条对比一下,颜色、波动曲线和跳动频率有点不一样,在暖色调的蔓延和肾上腺素的飙升下,这是一种心动的感觉。想起自己曾经写的一首诗:我见犹怜,爱不释手,雅俗共赏,君子好逑
。
标签导航栏
上面通过两个加载条演示了CSS变量在CSS中的运用以及一些妙用技巧,现在通过标签导航栏演示CSS变量在JS中的运用。
JS中主要有3个操作CSS变量的API,看上去简单易记,分别如下:
<ul><li>读取变量:elem.style.getPropertyValue()
<li>设置变量:elem.style.setProperty()
<li>删除变量:elem.style.removeProperty()
先上效果图,效果中主要是使用CSS变量标记每个Tab的背景色和切换Tab的显示状态。
<div class="tab-navbar">
<nav>
<a v-for="(v, i) in list" :key="v" :class="{ active: index === i }" @click="select(i)">标题{{i + 1}}</a>
</nav>
<div>
<ul ref="tabs" :style="`--tab-count: ${list.length}`">
<li v-for="(v, i) in list" :key="v" :style="`--bg-color: ${v}`">内容{{i + 1}}</li>
</ul>
</div>
</div>
Nach dem Login kopieren.tab-navbar {
display: flex;
overflow: hidden;
flex-direction: column-reverse;
border-radius: 10px;
width: 300px;
height: 400px;
nav {
display: flex;
height: 40px;
background-color: #f0f0f0;
line-height: 40px;
text-align: center;
a {
flex: 1;
cursor: pointer;
transition: all 300ms;
&.active {
background-color: #66f;
font-weight: bold;
color: #fff;
}
}
}
div {
flex: 1;
ul {
--tab-index: 0;
--tab-width: calc(var(--tab-count) * 100%);
--tab-move: calc(var(--tab-index) / var(--tab-count) * -100%);
display: flex;
flex-wrap: nowrap;
width: var(--tab-width);
height: 100%;
transform: translate3d(var(--tab-move), 0, 0);
transition: all 300ms;
}
li {
display: flex;
justify-content: center;
align-items: center;
flex: 1;
background-color: var(--bg-color);
font-weight: bold;
font-size: 20px;
color: #fff;
}
}
}
Nach dem Login kopierenexport default {
data() {
return {
index: 0,
list: ["#f66", "#09f", "#3c9"]
};
},
methods: {
select(i) {
this.index = i;
this.$refs.tabs.style.setProperty("--tab-index", i);
}
}
};
Nach dem Login kopieren在<ul>
上定义--tab-index
表示Tab当前的索引,当点击按钮时重置--tab-index
的值,就可实现不操作DOM来移动<ul>
的位置显示指定的Tab。不操作DOM而可移动<ul>
是因为定义了--tab-move
,通过calc()
计算--tab-index
与--tab-move
的关系,从而操控transform: translate3d()
来移动<ul>
。
另外在<li>
上定义--bg-color
表示Tab的背景色,也是一种比较简洁的模板赋值方式,总比写<li :style="backgroundColor: ${color}">
要好看。如果多个CSS属性依赖一个变量赋值,那么使用CSS变量赋值到style上就更方便了,那些CSS属性可在CSS文件里进行计算与赋值,这样可帮助JS分担一些属性计算工作。
当然,这个标签导航栏也可通过纯CSS实现,有兴趣的同学可看看笔者之前一篇文章里的纯CSS标签导航栏。
悬浮跟踪按钮
通过几个栗子实践了CSS变量在CSS和JS上的运用,相信大家已经掌握了其用法和技巧。之前在某个网站看过一个比较酷炫的鼠标悬浮特效,好像也是使用CSS变量实现的。笔者凭着记忆也使用CSS变量实现一番。
其实思路也比较简单,先对按钮进行布局和着色,然后使用伪元素标记鼠标的位置,定义--x
和--y
表示伪元素在按钮里的坐标,通过JS获取鼠标在按钮上的offsetLeft
和offsetLeft
分别赋值给--x
和--y
,再对伪元素添加径向渐变的背景色,大功告成,一个酷炫的鼠标悬浮跟踪特效就这样诞生了。
<a class="track-btn pr tac" @mousemove="move">
<span>妙用CSS变量,让你的CSS变得更心动</span>
</a>
Nach dem Login kopieren.track-btn {
display: block;
overflow: hidden;
border-radius: 100px;
width: 400px;
height: 50px;
background-color: #66f;
line-height: 50px;
cursor: pointer;
font-weight: bold;
font-size: 18px;
color: #fff;
span {
position: relative;
}
&::before {
--size: 0;
position: absolute;
left: var(--x);
top: var(--y);
width: var(--size);
height: var(--size);
background-image: radial-gradient(circle closest-side, #09f, transparent);
content: "";
transform: translate3d(-50%, -50%, 0);
transition: all 200ms ease;
}
&:hover::before {
--size: 400px;
}
}
Nach dem Login kopierenexport default {
name: "track-btn",
methods: {
move(e) {
const x = e.pageX - e.target.offsetLeft;
const y = e.pageY - e.target.offsetTop;
e.target.style.setProperty("--x", `${x}px`);
e.target.style.setProperty("--y", `${y}px`);
}
}
};
Nach dem Login kopieren其实可结合鼠标事件来完成更多的酷炫效果,例如动画关联
、事件响应
等操作。没有做不到,只有想不到,尽情发挥你的想象力啦。
之前在CodePen上还看到一个挺不错的栗子,一个悬浮视差按钮,具体代码涉及到一些3D变换的知识。看完源码后,按照其思路自己也实现一番,顺便对代码稍加改良并封装成Vue组件,存放到本课件示例代码中。感觉录制的GIF有点别扭,显示效果不太好,有兴趣的同学可下载本课件示例代码,自己运行看看效果。
兼容
对于现代浏览器来说,CSS变量的兼容性其实还是蛮好的,所以大家可放心使用。毕竟现在都是各大浏览器厂商快速迭代的时刻,产品对于用户体验来说是占了很大比重,因此在条件允许的情况下还是大胆尝新,不要被一些过去的所谓的规范所约束着。
试问现在还有多少人愿意去维护IE6~IE9的兼容性,如果一个产品的用户体验受限于远古浏览器的压制(可能政务Web应用和金融Web应用除外吧
),相信这个产品也不会走得很远。
我们在完成一个产品的过程中,不仅仅是为了完成工作任务,如果在保证进度的同时能花点心思点缀一下,可能会有意外的收获。用心写好每一段代码,才是享受写代码的真谛。
总结
本文通过循序渐进的方式探讨了CSS变量的运用和技巧,对于一个这么好用的特性,当然是不能放过啦。其实多多思考,就能把CSS变量用在很多场景上。
更多编程相关知识,请访问:编程入门!!
Das obige ist der detaillierte Inhalt vonSetzen Sie CSS-Variablen geschickt ein, um Ihr Projekt cooler zu machen!. Für weitere Informationen folgen Sie bitte anderen verwandten Artikeln auf der PHP chinesischen Website!