JavaScript 和 CSS 并存了 20 多年。然而,在它们之间共享数据非常困难。当然,已经进行了大规模的尝试。但是,我有一些简单直观的想法——不涉及结构更改,而是使用 CSS 自定义属性甚至 Sass 变量。
CSS 自定义属性和 JavaScript:
自定义属性在这里应该不足为奇。自从浏览器开始支持它们以来,它们一直能够做的一件事是与 JavaScript 一起设置和操作值。
不过,具体来说,我们可以通过几种方式使用带有自定义属性的 JavaScript。我们可以使用以下方法设置自定义属性的值setProperty
document.documentElement.style.setProperty("--padding", 124 + "px"); // 124px
我们还可以使用getComputedStyleJavaScript 检索 CSS 变量。这背后的逻辑相当简单:自定义属性是样式的一部分,因此,它们是计算样式的一部分。
getComputedStyle(document.documentElement).getPropertyValue('--padding') // 124px
同样的处理getPropertyValue
。这让我们从 HTML 标记的内联样式中获取自定义属性值。
document.documentElement.style.getPropertyValue("--padding'"); // 124px
请注意,自定义属性是有范围的。这意味着我们需要从特定元素中获取计算样式。正如我们之前定义的变量一样,
:root
我们将它们放在 HTML 元素上。
Sass 变量和 JavaScript
Sass 是一种预处理语言,这意味着它在成为网站的一部分之前就已经变成了 CSS。出于这个原因,以与 CSS 自定义属性相同的方式从 JavaScript 中访问它们(在 DOM 中可作为计算样式访问)是不可能的。
我们需要修改我们的构建过程来改变这一点。我怀疑在大多数情况下并没有太大的需求,因为加载程序通常已经是构建过程的一部分。但如果你的项目不是这样,我们需要三个能够导入和翻译 Sass 模块的模块。
这是在 webpack 配置中的样子:
module.exports = {
// ...
module: {
rules: [
{
test: /\.scss$/,
use: ["style-loader", "css-loader", "sass-loader"]
},
// ...
]
}
};
为了使 Sass(或本例中的 SCSS)变量可用于 JavaScript,我们需要“导出”它们。
// variables.scss
$primary-color: #fe4e5e;
$background-color: #fefefe;
$padding: 124px;
:export {
primaryColor: $primary-color;
backgroundColor: $background-color;
padding: $padding;
}
该:export
块是 webpack 用来导入变量的魔法酱。这种方法的好处是我们可以使用 camelCase 语法重命名变量并选择我们公开的内容。
然后我们将 Sass 文件 ( variables.scss
) 文件导入 JavaScript,让我们可以访问文件中定义的变量。
import variables from './variables.scss';
/*
{
primaryColor: "#fe4e5e"
backgroundColor: "#fefefe"
padding: "124px"
}
*/
document.getElementById("app").style.padding = variables.padding;
该:export
块是 webpack 用来导入变量的魔法酱。这种方法的好处是我们可以使用 camelCase 语法重命名变量并选择我们公开的内容。
然后我们将 Sass 文件 ( variables.scss
) 文件导入 JavaScript,让我们可以访问文件中定义的变量。
import variables from './variables.scss';
/*
{
primaryColor: "#fe4e5e"
backgroundColor: "#fefefe"
padding: "124px"
}
*/
document.getElementById("app").style.padding = variables.padding;
语法上有一些限制:export
值得一提:
- 它必须位于顶层,但可以位于文件中的任何位置。
- 如果文件中有多个,则将键和值组合在一起并一起导出。
- 如果某个特定
exportedKey
重复,则最后一个(按源顺序)优先。 - An
exportedValue
可以包含任何在 CSS 声明值中有效的字符(包括空格)。 - An
exportedValue
不需要被引用,因为它已经被视为文字字符串。
在 JavaScript 中访问 Sass 变量有很多方法可以派上用场。我倾向于采用这种方法来共享断点。这是我的breakpoints.scs
文件,稍后我将其导入 JavaScript,这样我就可以使用该matchMedia()
方法来设置一致的断点。
// Sass variables that define breakpoint values
$breakpoints: (
mobile: 375px,
tablet: 768px,
// etc.
);
// Sass variables for writing out media queries
$media: (
mobile: '(max-width: #{map-get($breakpoints, mobile)})',
tablet: '(max-width: #{map-get($breakpoints, tablet)})',
// etc.
);
// The export module that makes Sass variables accessible in JavaScript
:export {
breakpointMobile: unquote(map-get($media, mobile));
breakpointTablet: unquote(map-get($media, tablet));
// etc.
}
动画是另一个用例。动画的持续时间通常存储在 CSS 中,但更复杂的动画需要借助 JavaScript 来完成。
// animation.scss
$global-animation-duration: 300ms;
$global-animation-easing: ease-in-out;
:export {
animationDuration: strip-unit($global-animation-duration);
animationEasing: $global-animation-easing;
}
请注意,我strip-unit在导出变量时使用了自定义函数。这让我可以轻松地解析 JavaScript 端的内容。
// main.js
document.getElementById('image').animate([
{ transform: 'scale(1)', opacity: 1, offset: 0 },
{ transform: 'scale(.6)', opacity: .6, offset: 1 }
], {
duration: Number(variables.animationDuration),
easing: variables.animationEasing,
});