使用JS来动态操作css的几种方法

合集下载
  1. 1、下载文档前请自行甄别文档内容的完整性,平台不提供额外的编辑、内容补充、找答案等附加服务。
  2. 2、"仅部分预览"的文档,不可在线预览部分如存在完整性等问题,可反馈申请退款(可完整预览的文档不适用该条件!)。
  3. 3、如文档侵犯您的权益,请联系客服反馈,我们会尽快为您处理(人工客服工作时间:9:00-18:30)。

使⽤JS来动态操作css的⼏种⽅法
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 对象都是⼀个⽂档中链接或嵌⼊的样式表。

for(styleSheet of document.styleSheets){
console.log(styleSheet);
}
通过打印结果咱们可以知道,每次循环打印的是 CSSStyleSheet 对象,每个 CSSStyleSheet 对象由下列属性组成:
属性描述
media获取当前样式作⽤的媒体。

disabled打开或禁⽤⼀张样式表。

href返回 CSSStyleSheet 对象连接的样式表地址。

title返回 CSSStyleSheet 对象的title值。

type返回 CSSStyleSheet 对象的type值,通常是text/css。

parentStyleSheet返回包含了当前样式表的那张样式表。

ownerNode返回CSSStyleSheet对象所在的DOM节点,通常是<link>或
<style>。

cssRules返回样式表中所有的规则。

ownerRule如果是通过@import导⼊的,属性就是指向表⽰导⼊的规则的指针,
否则值为null。

IE不⽀持这个属性。

CSSStyleSheet对象⽅法:
⽅法描述
insertRule()在当前样式表的 cssRules 对象插⼊CSS规则。

deleteRule()在当前样式表删除 cssRules 对象的CSS规则。

有了StyleSheetList的全部内容,咱们来CSSStyleSheet本⾝。

在这⾥就有点意思了, CSSStyleSheet扩展了StyleSheet接⼝,并且只有这种只读属性,如.ownerNode,.href,.title或.type,它们⼤多直接从声明给定样式表的地⽅获取。

回想⼀下加载外部CSS⽂件的标准HTML代码,咱们就会明⽩这句话是啥意思:
<head>
<link rel="stylesheet" type="text/css" href="style.css" rel="external nofollow" title="Styles">
</head>
现在,咱们知道HTML⽂档可以包含多个样式表,所有这些样式表都可以包含不同的规则,甚⾄可以包含更多的样式表(当使⽤@import时)。

CSSStyleSheet有两个⽅法:、.insertrule()和.deleterule() 来增加和删除 Css 规则。

// ...
const ruleIndex = styleSheet.insertRule("div {background-color: red}");
styleSheet.deleteRule(ruleIndex);
.insertRule(rule,index):此函数可以在cssRules规则集合中插⼊⼀个指定的规则,参数rule是标⽰规则的字符串,参数index是值规则字符串插⼊的位置。

deleteRule(index):此函数可以删除指定索引的规规则,参数index规定规则的索引。

CSSStyleSheet也有⾃⼰的两个属性:.ownerRule和.cssRule。

虽然.ownerRule与@import相关,但⽐较有趣的是.cssRules。

简单地说,它是CSSRuleList的CSSRule,可以使⽤前⾯提到的.insertrule()和.deleterule()⽅法修改它。

请记住,有些浏览器可能会阻⽌咱们从不同的来源(域)访问外部CSSStyleSheet的.cssRules属性。

那么什么是 CSSRuleList?
CSSRuleList是⼀个数组对象包含着⼀个有序的CSSRule对象的集合。

每⼀个CSSRule可以通过rules.item(index)的形式访问,或者rules[index]。

这⾥的rules是⼀个实现了CSSRuleList接⼝的对象, index是⼀个基于0开始的,顺序与CSS样式表中的顺序是⼀致的。

样式对象的个数是通过rules.length表达。

对于CSSStyleRule对象:
每⼀个样式表CSSStyleSheet对象可以包含若⼲CSSStyleRule对象,也就是css样式规则,如下:
<style type="text/css">
h1{color:red}
div{color:green}
</style>
在上⾯的代码中style标签是⼀个CSSStyleSheet样式表对象,这个样式表对象包含两个CSSStyleRule对象,也就是两个css 样式规则。

CSSStyleRule对象具有下列属性:
1.type:返回0-6的数字,表⽰规则的类型,类型列表如下:
0:CSSRule.UNKNOWN_RULE。

1:CSSRule.STYLE_RULE (定义⼀个CSSStyleRule对象)。

2:CSSRule.CHARSET_RULE (定义⼀个CSSCharsetRule对象,⽤于设定当前样式表的字符集,默认与当前⽹页相同)。

3:CSSRule.IMPORT_RULE (定义⼀个CSSImportRule对象,就是⽤@import引⼊其他的样式表)
4:CSSRule.MEDIA_RULE (定义⼀个CSSMediaRule对象,⽤于设定此样式是⽤于显⽰器,打印机还是投影机等等)。

5:CSSRule.FONT_FACE_RULE (定义⼀个CSSFontFaceRule对象,CSS3的@font-face)。

6:CSSRule.PAGE_RULE (定义⼀个CSSPageRule对象)。

2.cssText:返回⼀个字符串,表⽰的是当前规则的内容,例如:
div{color:green}
3.parentStyleSheet:返回所在的CSSStyleRule对象。

4.parentRule:如果规则位于另⼀规则中,该属性引⽤另⼀个CSSRule对象。

5.selectorText:返回此规则的选择器,如上⾯的div就是选择器。

6.style:返回⼀个CSSStyleDeclaration对象。

// ...
const ruleIndex = styleSheet.insertRule("div {background-color: red}");
const rule = styleSheet.cssRules.item(ruleIndex);
rule.selectorText; // "div"
rule.style.backgroundColor; // "red"
实现
现在,咱们对 CSS 相关的 JS Api有了⾜够的了解,可以创建咱们⾃⼰的、⼩型的、基于运⾏时的CSS-in-JS实现。

咱们的想法是创建⼀个函数,它传递⼀个简单的样式配置对象,⽣成⼀个新创建的CSS类的哈希名称供以后使⽤。

实现流程很简单,咱们需要⼀个能够访问某种样式表的函数,并且只需使⽤.insertrule()⽅法和样式配置就可以运⾏了。

先从样式表部分开始:
function createClassName(style) {
// ...
let styleSheet;
for (let i = 0; i < document.styleSheets.length; i++) {
if (document.styleSheets[i].CSSInJS) {
styleSheet = document.styleSheets[i];
break;
}
}
if (!styleSheet) {
const style = document.createElement("style");
document.head.appendChild(style);
styleSheet = style.sheet;
styleSheet.CSSInJS = true;
}
// ...
}
如果你使⽤的是ESM或任何其他类型的JS模块系统,则可以在函数外部安全地创建样式表实例,⽽不必担⼼其他⼈对其进⾏访问。

但是,为了演⽰例,咱们将stylesheet上的.CSSInJS属性设置为标志的形式,通过标志来判断是否要使⽤它。

现在,如果如果还需要创建⼀个新的样式表怎么办?最好的选择是创建⼀个新的<style/>标记,并将其附加到HTML⽂档的<head/>上。

这会⾃动将新样式表添加到document.styleSheets列表,并允许咱们通过<style/>标记的.sheet属性对其进⾏访问,是不是很机智?
function createRandomName() {
const code = Math.random().toString(36).substring(7);
return `css-$[code]`;
}
function phraseStyle(style) {
const keys = Object.keys(style);
const keyValue = keys.map(key => {
const kebabCaseKey =
key.replace(/([a-z])([A-Z])/g, "$1-$2").toLowerCase();
const value =
`${style[key]}${typeof style[key] === "number" ? "px" : ""}`;
return `${kebabCaseKey}:${value};`;
});
return `{${keyValue.join("")}}`;
}
除了上⾯的⼩窍门之外。

⾃然,咱们⾸先需要⼀种为CS S类⽣成新的随机名称的⽅法。

然后,将样式对象正确地表达为可⾏的CSS字符串的形式。

这包括驼峰命名和短横线全名之间的转换,以及可选的像素单位(px)转换的处理。

function createClassName(style) {
const className = createRandomName();
let styleSheet;
// ...
styleSheet.insertRule(`.${className}${phraseStyle(style)}`);
return className;
}
完整代码如下:
HTML
<div id="el"></div>
JS
function createRandomName() {
const code = Math.random().toString(36).substring(7);
return `css-$[code]`;
}
function phraseStyle(style) {
const keys = Object.keys(style);
const keyValue = keys.map(key => {
const kebabCaseKey = key.replace(/([a-z])([A-Z])/g, "$1-$2").toLowerCase();
const value = `${style[key]}${typeof style[key] === "number" ? "px" : ""}`;
return `${kebabCaseKey}:${value};`;
});
return `{${keyValue.join("")}}`;
}
function createClassName(style) {
const className = createRandomName();
let styleSheet;
for (let i = 0; i < document.styleSheets.length; i++) {
if (document.styleSheets[i].CSSInJS) {
styleSheet = document.styleSheets[i];
break;
}
}
if (!styleSheet) {
const style = document.createElement("style");
document.head.appendChild(style);
styleSheet = style.sheet;
styleSheet.CSSInJS = true;
}
styleSheet.insertRule(`.${className}${phraseStyle(style)}`);
return className;
}
const el = document.getElementById("el");
const redRect = createClassName({
width: 100,
height: 100,
backgroundColor: "red"
});
el.classList.add(redRect);
运⾏效果:
总结
正如本⽂咱们所看到的,使⽤ JS 操作CSS 是⼀件⾮常有趣的事,咱们可以挖掘很多好⽤的 API,上⾯的例⼦只是冰⼭⼀⾓,在CSS API(或者更确切地说是API)中还有更多⽅法,它们正等着被揭开神秘⾯纱。

以上就是本⽂的全部内容,希望对⼤家的学习有所帮助,也希望⼤家多多⽀持。

相关文档
最新文档