您现在的位置是:网站首页> 编程资料编程资料

深入解析CSS中的自定义属性_CSS_网页制作_

2021-09-13 1231人已围观

简介 这篇文章主要介绍了CSS中的自定义属性,谈到了媒体查询的响应式特性、预处理器的使用等重要知识,干货十足,需要的朋友可以参考下

CSS 预处理器是一个非常出色的工具,但是它们的变量是静态的,有语法作用域。Native CSS 变量,从另一面来看,它们是一个完全不同类型的变量:因为它们是动态的,他们的作用域是DOM,事实上,这也是困惑该不该称他们为变量,它们实际上是CSS 属性,这也给了他们一个机会,来解决这个功能完全不同的问题。

在这篇文章中,我将讨论一些CSS 自定义属性这个功能,而且不用CSS 预处理器来做。当然我还演示一些新的设计模式,自定义功能的启用。文章最后讨论一下,我认为在未来最有可能的是预处理变量和自定义变量一起使用,两个东西取长补短,珠联璧合。


预处理器变量的限制
在继续写之前,我想强调的是,我真的很喜欢CSS 预处理器,我的所有项目都在使用它。预处理器做了一件非常了不起的事情,即时你知道他最终出来的就是原始的CSS,任然可以感受这个神器的时代。

任何工具,都有他的局限性,有一个炫酷的外观会让人惊喜而忽略了其中的限制,特别是新用户。

Preprocessor variables aren’t live
也许受预处理限制,在媒体查询中,最常见的新手也无力吐槽定义变量或使用@extend

CSS Code复制内容到剪贴板
  1. $gutter: 1em;   
  2.   
  3. @media (min-width: 30em) {   
  4.   $gutter: 2em;   
  5. }   
  6.   
  7. .Container {   
  8.   padding: $gutter;   
  9. }  

如果你编译上面的代码,你得到是:

CSS Code复制内容到剪贴板
  1. .Container {   
  2.   padding: 1em;   
  3. }  

如你所见,媒体查询被废弃,变量赋值被忽略。

从理论上讲,虽然sass 负责申明条件变量,但这样做也是一个挑战,枚举所有Permutations—exponentially 会增加CSS的最终大小。

预处理器变量不能级联(层叠)
每当你使用变量,作用域的范围不可避免,这个变量应该全局吗?应该是file/module?还是块作用域?

CSS 最终是为HTML的样式,事实证明还有另外一种有用的方法是变量的范围:DOM 元素,但是preprocessors不能运行在浏览器且从未看见标记。

参考一个网站,试图给 的元素添加一个 class user-setting-large-text 他们更倾向于更大的文本大小 。
一旦这个class设置,更大$font-size变量赋值就会运用:

CSS Code复制内容到剪贴板
  1.   $font-size: 1em;   
  2.   
  3. .user-setting-large-text {   
  4.   $font-size: 1.5em;   
  5. }   
  6.   
  7. body {   
  8.   font-size: $font-size;   
  9. }  

但是,就像上面媒体查询例子,Sass 直接忽略变量赋值, 意味着这种事是不可能的。他输出的:

CSS Code复制内容到剪贴板
  1.   body {   
  2.   font-size: 1em;   
  3. }  

预处理器变量不继承
虽然继承是级联的一部分,但是我还是要提一下,因为很多次我想使用这个功能都未能用成。

有一种情况,你有Dom元素在颜色风格基础上有什么的变化时候,你可以用在他们的父元素上。

CSS Code复制内容到剪贴板
  1. .alert { background-color: lightyellow; }   
  2. .alert.info { background-color: lightblue; }   
  3. .alert.error { background-color: orangered; }   
  4.   
  5. .alert button {   
  6.   border-color: darken(background-color, 25%);   
  7. }  

上面的Sass代码是无效的,但你应该能理解这代码试图要完成什么。

它最后试图使用sass的darken函数用在background-color属性,但button元素继承它的父class元素.alert。如果class info或者error已经添加到alert(或者通过JavaScript设置背景颜色或用户样式),按钮元素希望能够得到这两个颜色。

现在这个虽然不会在sass 运行,因为预处理器不知道DOM结构,但还是希望搞清楚这类型的东西可能有哪些用处。

说一个特定的用例:这也是在继承DOM属性的可访问性运行color 函数的原因。举个栗子,为了确保文本总是可读,且充分与背景颜色形成鲜明对比。通过自定义属性和新的CSS Color函数,这将很快成为可能!

预处理器变量不能相互协作
这是一个明显呈下降趋势的预处理器,如果你用PostCSS 建立一个网站,你想使用第三方组件,不好意思,你只有通过Sass的themeable。

与第三方分享预处理器变量在不同的工具集成或第三方托管的CND样式与都非常困难(至少不容易)。

本地CSS自定义属性将与任何CSS预处理或者原CSS正好相反。

自定义属性有何不同
你可能已经猜到了,我上面列出的适用于CSS 自定义属性没有任何限制,但也许更重要的不是说他们不适用,而是为什么他们不用。

CSS自定义属性就像常规的CSS属性一样,他们的操作方式完全相同。

像普通的CSS属性,自定义属性是动态的,他们可以在运行时修改,也可以在媒体查询时通过更改DOM添加一个新类,同时也可以指派内联元素和一个常规CSS里申明选择器。还可以通过正常的cascade规则或者使用JavaScript覆盖。最主要的是,他们的可以继承的,所以当他们应用到DOM元素的时候,他们的子元素也会继承属性。

为了更简洁,预处理器变量是语法作用域和编译后静态。自定义属性作用域是DOM,他们都很灵活。

实际案例
如果你仍然不确定自定义属性可以做到这一点,而预处理器不行,我这里给一些例子。

不论真假,有大量非常好的例子我都很想展示,但为了不让这篇文章太丑,我选了两个。

我选择这些例子不仅仅因为它们的理论,它们也是我们过去实际面临的挑战,我依然记得试图用预处理器,但这是不可能的。现在好了,直接自定义属性走起。

媒体查询的响应式特性
很多网站在项目布局使用“gap”和“gutter” 定义默认间距和填充页面各个部分,很多时候,你想要这个“gutter”的值根据浏览器窗口的大小而不同。在大屏幕上你想要每一项之间有足够的空间,但小屏幕又负担不起那么大的空间,所以“gutter”的值要较小。

正如我上面提到的,在媒体查询里面Sass 不能正常运行,所以你必须每个单独处理。

下面的例子定义了变量$gutterSm, $gutterMd和$gutterLg,然后给每个变量申明一个单独的规则:

CSS Code复制内容到剪贴板
  1. /* Declares three gutter values, one for each breakpoint */  
  2.   
  3. $gutterSm: 1em;   
  4. $gutterMd: 2em;   
  5. $gutterLg: 3em;   
  6.   
  7. /* Base styles for small screens, using $gutterSm. */  
  8.   
  9. .Container {   
  10.   margin: 0 auto;   
  11.   max-width: 60em;   
  12.   padding: $gutterSm;   
  13. }   
  14. .Grid {   
  15.   display: flex;   
  16.   margin: -$gutterSm 0 0 -$gutterSm;   
  17. }   
  18. .Grid-cell {   
  19.   flex: 1;   
  20.   padding: $gutterSm 0 0 $gutterSm;   
  21. }   
  22.   
  23. /* Override styles for medium screens, using $gutterMd. */  
  24.   
  25. @media (min-width: 30em) {   
  26.   .Container {   
  27.     padding: $gutterMd;   
  28.   }   
  29.   .Grid {   
  30.     margin: -$gutterMd 0 0 -$gutterMd;   
  31.   }   
  32.   .Grid-cell {   
  33.     padding: $gutterMd 0 0 $gutterMd;   
  34.   }   
  35. }   
  36.   
  37. /* Override styles for large screens, using $gutterLg. */  
  38.   
  39. @media (min-width: 48em) {   
  40.   .Container {   
  41.     padding: $gutterLg;   
  42.   }   
  43.   .Grid {   
  44.     margin: -$gutterLg 0 0 -$gutterLg;   
  45.   }   
  46.   .Grid-cell {   
  47.     padding: $gutterLg 0 0 $gutterLg;   
  48.   }   
  49. }  

使用自定义属性来完成相同的东西,你只需要定义样式即可。你可以使用一个 gutter 属性,然后随着媒体查询的变化,更新gutter 的值,它就会做出相应的变化。

CSS Code复制内容到剪贴板
  1. :root { --gutter: 1.5em; }   
  2.   
  3. @media (min-width: 30em) {   
  4.   :root { --gutter: 2em; }   
  5. }   
  6. @media (min-width: 48em) {   
  7.   :root { --gutter: 3em; }   
  8. }   
  9.   
  10. /*  
  11.  * Styles only need to be defined once because  
  12.  * the custom property values automatically update.  
  13.  */  
  14.   
  15. .Container {   
  16.   margin: 0 auto;   
  17.   max-width: 60em;   
  18.   padding: var(--gutter);   
  19. }   
  20. .Grid {   
  21.   --gutterNegative: calc(-1 * var(--gutter));   
  22.   display: flex;   
  23.   margin-left: var(--gutterNegative);   
  24.   margin-top: var(--gutterNegative);   
  25. }   
  26. .Grid-cell {   
  27.   flex: 1;   
  28.   margin-left: var(--gutter);   
  29.   margin-top: var(--gutter);   
  30. }  

虽然有额外增加的自定义属性语法,但是相比冗长的代码完成同样的事明显好很多。这里只考虑了三个变量,如果变量越多,这将节省更多的代码。

下面的演示使用的是上面的代码自动构建的一个基本的网站布局,gutter的值跟随窗口的变化而变化,浏览器的支持自定义属性的话,效果

-六神源码网