目录
插值。啥玩意儿?
Sass的插值
什么时候应该使用插值
字符串 (Strings)
CSS函数 (Functions)
CSS指令 (Directives)
选择器 (Selectors)
结语
首页 web前端 html教程 你需要知道的Sass插值_html/css_WEB-ITnose

你需要知道的Sass插值_html/css_WEB-ITnose

Jun 24, 2016 am 11:56 AM

你也许会不时地写写 Sass 玩玩,你也会很享受它带给你各种便利。但还有一件事,你并不一定完全了解:插值 (interpolation) - 将一个占位符,替换成一个值。好了,你们都很幸运,因为今天我将把这种问题说清楚。

插值。啥玩意儿?

插值,通常是指变量插值,或者变量替换。这不是Sass独有的。实际上,你可以在很多编程语言中,发现这种特性。比如 PHP, Perl, Ruby, Tcl, Groovy, Unix shells, 等等。我们经常说的是,插入一个变量,或者插入一个表达式。

我们还是先来看看一个例子吧。如果你有PHP的基础,接下来应该会很容易了解。比如说,你想打印一个包含变量的字符串,最常见的方式:

  $description = "awesome";  echo "Tuts+ is " . $description . "!";
登录后复制

这就不是插值的方式,你这是字符串的连接。这其中,连接了三个字符串:"Tuts+ is ",awesome(被 $description所引用) 以及 "!"。现在,我们就来看看使用插值而非字符串连接的方式:

  $description = "awesome";  echo "Tuts+ is ${description}!";
登录后复制

包裹在变量两边的花括号会告诉PHP,用字符串打印出变量的值。值得注意的是:它需要在双引号里面才可以工作(大多语言都是这样)。

不管怎样,这就是变量/表达式插值。是使用字符串连接,还是使用插值,这都取决于你自己。但可以说一点:插值其实就是字符串连接的语法糖而已。

Sass的插值

我们先看看在Sass中,变量替换是怎么工作。
Sass变量的命名,就像PHP一样,都有着美元符号的前缀($)。两者的对比,显得就结束了,因为谈到插值,两者的表现是不同的。有一个很好的解释是:Sass是基于Ruby的,它使用 ${}进行表达式替换。

在Sass中,你会这样做:

  $description: "awesome";  @warn "Tuts+ is #{$description}!";
登录后复制

需要注意的是,变量中的$ 不能像PHP一样丢掉。变量被#{}包裹了起来。另外值得一提的是:你可以插入任何类型的变量,不仅仅是字符串。
比如:

  $answer: 42;  @warn "The Answer to the Ultimate Question of Life, the Universe, and Everything is #{$answer}.";
登录后复制

什么时候应该使用插值

现在你应该明白什么是插值了,也知道怎么在Sass中工作的了。是时候,开始着手实际场景了。首先,我们会再次使用刚才做过的@warn指令,它将打印相应内容到控制台。

字符串 (Strings)

假设你有一组叫$colors名的颜色映射(映射是指一个存储了一系列 key/value 组合的变量),但你已经受够了一次又一次地敲 map-get($colors, ...)。所以,你写了一个简单的color()函数,使用key去获得相应的值。
把这些组合一下就是:

  // _config.scss  $colors: (    "primary": tomato,    "secondary": hotpink  );  // _function.scss  @function color($key) {    @return map-get($colors, $key);  }  // _component.scss  .el {    background-color: color(primary);  }
登录后复制

一切都很好,不是吗?现在,当你敲错名称,或者去取一个不存在的key时,你想给出错误的信息。这将通过 @warn指令来完成。color()函数如下:

  @function color($key) {    @if not map-has-key($colors, $key) {      @warn "Key not found.";    }    @return map-get($colors, $key);  }
登录后复制

还不错。如果你想知道哪个key没有找到呢?

  @function color($key) {    @if not map-has-key($colors, $key) {      @warn "Key `#{$key}` not found.";    }    @return map-get($colors, $key);  }
登录后复制

嘣~~变量插值。调用 color(awesomeness) 时,将会抛出以下信息:

Key awesomeness not found

这真的很棒,但我们却不知道上下文是什么。为使后来的方便,我们可以将映射的名称写到错误信息里面。

  @function color($key) {    @if not map-has-key($colors, $key) {      @warn "Key `#{$key}` not found in $colors map.";    }    @return map-get($colors, $key);  }
登录后复制

在这个场景里,由于这个 $colors 变量没有使用插值,它将打印以下信息:

Key awesomeness not found in $colors map.

CSS函数 (Functions)

到目前为止,我们已经见到了最常见的变量替换场景:打印字符串中变量的内容。这确实是一个好的示例,但我觉得应该还有更好的场景:CSS函数中的变量,比如 calc()。

假设你想基于侧边栏的宽度设置主容器的大小。你是一个勤奋的前端开发者,已经把这个宽度存储在一个变量中,所以,你可能会这样做:

  $sidebar-width: 250px;  .main {    width: calc(100% - $sidebar-width);  }
登录后复制

然后,你会惊讶地发现,根本不work。没有报错,容器的大小却又不正确。如果你去审查你的dom元素,你会看到这个 ? 被划掉了。因为,这是不合法的。

  .main {    width: calc(100% - $sidebar-width);  }
登录后复制

现在我们应该想到:calc() 是一个CSS函数,不是一个Sass函数。这就是说Sass会将整个表达式解释成一个字符串。你可以试试:

$type-of-expression: type-of(calc(100% - $sidebar-width)); // string
登录后复制

因为这是一个字符串,难怪Sass表现和之前@warn中的$colors字符串一样。$sidebar-width被认为是一个常规字符串,所以打出来就是它自己。但这都不是我们所想要的,是吧?我们用插值这样做。

  .main {    width: calc(100% - #{$sidebar-width});  }
登录后复制

现在当Sass编译这个样式文件时,它会用#{$sidebar-width}的值,250px替换#{$sidebar-width}。最后,便是合法的CSS表达式。

  .main {    width: calc(100% - 250px);  }
登录后复制

任务完成!我们仅仅在这里谈了calc(),但其实和其他CSS 原生函数是一样的,包括伪类。比如:url(),linear-gradient(),radial-gradient(),cubic-bezier()。

以下是另一个使用CSS函数的例子:

  @for $i from 1 through $max {    .el:nth-of-type(#{$i}) {      // ...    }  }
登录后复制

这是一个你有可能遇到过的场景:for循环和:nth-*()选择器一起使用。再一次说明,你需要使用插值变量,才能最终得到想得到的结果。

小结:Sass会把CSS函数认为是字符串,所以想要在最后获得它们的值,要求你转义所有同它们一起使用的变量。

CSS指令 (Directives)

我们将视转移到另一个有趣的变量插值场景:CSS指令,比如@support,@page,最重要的还是@media。

现在,Sass是怎样解析CSS指令,尤其是demia指令的。我已经查看过Sass的源码,当我Ruby环境有点问题的时候,我找到一些有趣的事情。

  def query_expr    interp = interpolation    return interp if interp    return unless tok(/\(/)    res = ['(']    ss    res << sass_script(:parse)    if tok(/:/)      res << ': '      ss      res << sass_script(:parse)    end    res << tok!(/\)/)    ss    res  end
登录后复制

第一行告诉Sass,如果有一个插值表达式的话,便返回media query。如果找到一个开括号((),便会一直往下走,解析所有的东西,反之就会抛出一个错误。我们在这里试试一个例子:

  $value: screen;  @media $value {    // ...  }
登录后复制

毫不惊讶的是,这失败了:

Invalid CSS after "@media (" at media) ": expected media query (e.g. print, screen, print and screen), was "$value {"

就像错误信息提示的那样,它期待一个media query。在这里,如果你的变量在 @media 字符串后面,需要使用插值才可以。比如:

  $value: screen;  @media #{$value} {    // ...  }
登录后复制

和我们之前讨论的Ruby转义规则一样,如果@media后面紧跟(()),你就不再需要插值变量了,因为Sass会求出所有在这些括号里面的值。比如:

  $value: 1336px;  @media (max-width: $value) {    // ...  }
登录后复制

在这个示例中,Sass将这个表达式(max-width: $value)转化成(max-width: 1337px),最后生成合法的CSS结果。所以,我们便没有必要再对变量转义了。

选择器 (Selectors)

好的,这将是最后一个示例:使用变量作为一个选择器,或者选择器的一部分。这不是一种常用的用法,以下的做法不太符合常识:

  $value: custom;  selector-$value {    property: value;  }
登录后复制

不幸的是,这根本无法编译成功:

Invalid CSS after "selector-": expected "{", was "$value {"

这是media query那个例子的原因,差不多。Sass有着自己的方式解析一个CSS选择器。如果它遇到了不可预知的东西,比如一个未经处理的美元符号,它就会直接崩溃。

幸运的,解决这个问题也相当简单(你也已经知道怎么做了):没错,使用插值变量!

  $value: custom;  selector-#{$value} {    property: value;  }
登录后复制

结语

在最后,Sass的插值没有看起来那么简单。在一些示例之中,你必须转义你的变量,一些示例中,你却不必。如果是那样,你有两种方式处理:

  • 要么你便等着错误信息,然后转义
  • 要么你转义除了常规CSS值(你清楚地知道,它会工作得很好)外的所有地方
  • 不管怎样,我希望你明白了插值变量是如何工作的。如果你有什么需要添加的,请在留言中指出。

    本站声明
    本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

    热AI工具

    Undresser.AI Undress

    Undresser.AI Undress

    人工智能驱动的应用程序,用于创建逼真的裸体照片

    AI Clothes Remover

    AI Clothes Remover

    用于从照片中去除衣服的在线人工智能工具。

    Undress AI Tool

    Undress AI Tool

    免费脱衣服图片

    Clothoff.io

    Clothoff.io

    AI脱衣机

    AI Hentai Generator

    AI Hentai Generator

    免费生成ai无尽的。

    热门文章

    R.E.P.O.能量晶体解释及其做什么(黄色晶体)
    2 周前 By 尊渡假赌尊渡假赌尊渡假赌
    仓库:如何复兴队友
    1 个月前 By 尊渡假赌尊渡假赌尊渡假赌
    Hello Kitty Island冒险:如何获得巨型种子
    1 个月前 By 尊渡假赌尊渡假赌尊渡假赌

    热工具

    记事本++7.3.1

    记事本++7.3.1

    好用且免费的代码编辑器

    SublimeText3汉化版

    SublimeText3汉化版

    中文版,非常好用

    禅工作室 13.0.1

    禅工作室 13.0.1

    功能强大的PHP集成开发环境

    Dreamweaver CS6

    Dreamweaver CS6

    视觉化网页开发工具

    SublimeText3 Mac版

    SublimeText3 Mac版

    神级代码编辑软件(SublimeText3)

    公众号网页更新缓存难题:如何避免版本更新后旧缓存影响用户体验? 公众号网页更新缓存难题:如何避免版本更新后旧缓存影响用户体验? Mar 04, 2025 pm 12:32 PM

    公众号网页更新缓存,这玩意儿,说简单也简单,说复杂也够你喝一壶的。你辛辛苦苦更新了公众号文章,结果用户打开还是老版本,这滋味,谁受得了?这篇文章,咱就来扒一扒这背后的弯弯绕绕,以及如何优雅地解决这个问题。读完之后,你就能轻松应对各种缓存难题,让你的用户始终体验到最新鲜的内容。先说点基础的。网页缓存,说白了就是浏览器或者服务器为了提高访问速度,把一些静态资源(比如图片、CSS、JS)或者页面内容存储起来。下次访问时,直接从缓存里取,不用再重新下载,速度自然快。但这玩意儿,也是个双刃剑。新版本上线,

    如何高效地在网页中为PNG图片添加描边效果? 如何高效地在网页中为PNG图片添加描边效果? Mar 04, 2025 pm 02:39 PM

    本文展示了使用CSS为网页中添加有效的PNG边框。 它认为,与JavaScript或库相比,CSS提供了出色的性能,详细介绍了如何调整边界宽度,样式和颜色以获得微妙或突出的效果

    如何使用HTML5表单验证属性来验证用户输入? 如何使用HTML5表单验证属性来验证用户输入? Mar 17, 2025 pm 12:27 PM

    本文讨论了使用HTML5表单验证属性,例如必需的,图案,最小,最大和长度限制,以直接在浏览器中验证用户输入。

    &lt; datalist&gt;的目的是什么。 元素? &lt; datalist&gt;的目的是什么。 元素? Mar 21, 2025 pm 12:33 PM

    本文讨论了html&lt; datalist&gt;元素,通过提供自动完整建议,改善用户体验并减少错误来增强表格。Character计数:159

    HTML5中跨浏览器兼容性的最佳实践是什么? HTML5中跨浏览器兼容性的最佳实践是什么? Mar 17, 2025 pm 12:20 PM

    文章讨论了确保HTML5跨浏览器兼容性的最佳实践,重点是特征检测,进行性增强和测试方法。

    &lt; meter&gt;的目的是什么。 元素? &lt; meter&gt;的目的是什么。 元素? Mar 21, 2025 pm 12:35 PM

    本文讨论了HTML&lt; meter&gt;元素,用于在一个范围内显示标量或分数值及其在Web开发中的常见应用。它区分了&lt; meter&gt;从&lt; progress&gt;和前

    &gt; gt;的目的是什么 元素? &gt; gt;的目的是什么 元素? Mar 21, 2025 pm 12:34 PM

    本文讨论了HTML&lt; Progress&gt;元素,其目的,样式和与&lt; meter&gt;元素。主要重点是使用&lt; progress&gt;为了完成任务和LT;仪表&gt;对于stati

    &lt; iframe&gt;的目的是什么。 标签?使用时的安全考虑是什么? &lt; iframe&gt;的目的是什么。 标签?使用时的安全考虑是什么? Mar 20, 2025 pm 06:05 PM

    本文讨论了&lt; iframe&gt;将外部内容嵌入网页,其常见用途,安全风险以及诸如对象标签和API等替代方案的目的。

    See all articles