使用JS来动态操作CSS,你知道几种方法?
(给前端大学加星标,提升前端技能.)
译者:前端小智
https://segmentfault.com/a/1190000021169316
JavaScript 可以说是交互之王,它作为脚本语言加上许多 Web Api 进一步扩展了它的特性集,更加丰富界面交互的可操作性。这类 API 的例子包括WebGL API、Canvas API、DOM API,还有一组不太为人所知的 CSS API。
由于JSX
和无数JS框架
的出现,使通过JS API与DOM交互的想法真正流行起来,但是在 CSS 中使用类似技术似乎并没有很多。 当然,存在像CSS-in-JS这类解决方案,但是最流行的解决方案还是基于转译(transpilation),无需额外运行即可生成 CSS。 这肯定对性能有好处,因为CSS API的使用可能导致额外的重绘,这与DOM API
的使用一样。 但这不是咱们想要的。 如果哪天公司要求咱们,既要操纵 DOM 元素的样式和 CSS 类,还要像使用 HTML 一样使用 JS 创建完整的样式表,该怎么办?
.style
属性来编辑给定的HTMLElement
的内联样式。直接在const el = document.createElement('div')
el.style.backgroundColor = 'red'
// 或者
el.style.cssText = 'background-color: red'
// 或者
el.setAttribute('style', 'background-color: red')
.style
对象上设置样式属性将需要使用驼峰式命名作为属性键,而不是使用短横线命名
。 如果咱们需要设置更多的内联样式属性,则可以通过设置.style.cssText
属性,以更加高效的方式进行设置 。请记住,给cssText设置后原先的css样式被清掉了,因此,要求咱们一次死一堆样式 。如果这种设置内联样式过于繁琐,咱们还可以考虑将.style
与Object.assign()
一起使用,以一次设置多个样式属性。这些“基本知识”比咱们想象的要多得多。// ...
Object.assign(el.style, {
backgroundColor: "red",
margin: "25px"
})
.style
对象实现CSSStyleDeclaration
接口。 这说明它带还有一些有趣的属性和方法,这包括刚刚使用的.cssText
,还包括.length
(设置属性的数量),以及.item()
、.getPropertyValue()
和.setPropertyValue()
之类的方法:这里有个小窍门-在遍历过程中// ...
const propertiesCount = el.style.length
for(let i = 0; i < propertiesCount; i++) {
const name = el.style.item(i) // 'background-color'
const value = el.style.getPropertyValue(name) // 're'
const priority = el.style.getPropertyPriority(name) // 'important'
if(priority === 'important') {
el.style.removeProperty()
}
}
.item()
方法具有按索引访问形式的备用语法。// ...
el.style.item(0) === el.style[0]; // true
CSS 类
接着,来看看更高级的结构——CSS类,它在检索和设置时具有字符串形式是.classname
。设置类字符串的另一种方法是设置// ...
el.className = "class-one class-two";
el.setAttribute("class", "class-one class-two");
class
属性(与检索相同)。 但是,就像使用.style.cssText
属性一样,设置.className
将要求咱们在字符串中包括给定元素的所有类,包括已更改和未更改的类。当然,可以使用一些简单的字符串操作来完成这项工作,还有一种就是使用较新的.classList
属性,这个属性,IE9 不支持它,而 IE10 和 IE11 仅部分支持它。classlist
属性实现了DOMTokenList
,有一大堆有用的方法。例如.add()
、.remove()
、.toggle()和.replace()
允许咱们更改当前的 CSS 类集合,而其他的,例如.item()
、.entries()
或.foreach()
则可以简化这个索引集合的遍历过程。// ...
const classNames = ["class-one", "class-two", "class-three"];
classNames.forEach(className => {
if(!el.classList.contains(className)) {
el.classList.add(className);
}
});
Stylesheets
一直以来,Web Api 还有一个StyleSheetList
接口,该接口由document.styleSheets
属性实现。 document.styleSheets
只读属性,返回一个由 StyleSheet
对象组成的 StyleSheetList
,每个 StyleSheet
对象都是一个文档中链接或嵌入的样式表。通过打印结果咱们可以知道,每次循环打印的是 CSSStyleSheet 对象,每个 CSSStyleSheet 对象由下列属性组成:for(styleSheet of document.styleSheets){
console.log(styleSheet);
}
属性 | 描述 |
---|---|
media | 获取当前样式作用的媒体。 |
disabled | 打开或禁用一张样式表。 |
href | 返回 CSSStyleSheet 对象连接的样式表地址。 |
title | 返回 CSSStyleSheet 对象的title值。 |
type | 返回 CSSStyleSheet 对象的type值,通常是text/css。 |
parentStyleSheet | 返回包含了当前样式表的那张样式表。 |
ownerNode | 返回CSSStyleSheet对象所在的DOM节点,通常是或 |