SASS 预处理器

什么是 CSS 预处理器?

CSS 预处理器定义了一种新的语言,其基本思想是,用一种专门的编程语言,为 CSS 增加了一些编程的特性,将 CSS 作为目标生成文件,然后开发者就只要使用这种语言进行编码工作。

  • 通过工具编译成 CSS:开发后,这些特定文件被编译成任何浏览器都可以理解的常规 CSS。
  • 能提升 CSS 文件的组织方式:预处理程序有助于在 CSS 中编写可重用,易于维护和可扩展的代码。
  • 基于 CSS 的另一种语言:CSS 预处理器定义了一种新的语言,其基本思想是,用一种专门的编程语言,为 CSS 增加了一些编程的特性,将 CSS 作为目标生成文件,然后开发者就只要使用这种语言进行编码工作
  • 扩展 CSS:CSS 预处理器为 CSS 增加一些编程的特性,无需考虑浏览器的兼容性问题,例如你可以在 CSS 中使用变量、简单的逻辑程序、函数、插值、mixin 等等在编程语言中的一些基本特性,可以让你的 CSS 更加简洁、适应性更强、可读性更佳,更易于代码的维护等。

目前主流的预处理器里有:Sass(scss)LessStylusPostCSS(后预处理器)

下面我们将来介绍 Sass 预处理器。详细内容查阅 Sass 文档

什么是 Sass?

Sass 是最早的 CSS 预处理语言,有比 Less 更为强大的功能。但因其一开始的缩进式语法并不能被开发者们接受,所以使用率不高,不过由于其强大的功能和 Ruby on Rails 的大力推动,逐渐被更多开发者使用。

Sass 是采用的 Ruby 语言编写的一款 CSS 预处理语言,它诞生于 2007 年,是最早成熟 CSS 预处理脚本语言,它被解释或编译成级联样式表。它允许您编写可维护的 CSS,并提供变量、嵌套、混合、扩展、函数、循环、条件等功能。最初它是为了配合 HAML 而设计的,因此有着和 HAML 一样的缩进式风格。

Sass 和 SCSS 有什么区别?

Sass 从第三代开始,放弃了缩进式风格,并且完全向下兼容普通的 CSS 代码,这一代的 Sass 也被称为 ScssScss 完全继承了 Sass 中的功能,并引入了一些新语法特性。

主要有两点不同:

  • 文件扩展:Sass 的后缀扩展名为 .sass,而 SCSS 的 后缀为扩展名为 .scss
  • 语法书写:Sass 是以严格的缩进式语法规则来书写,不带大括号 {} 和分号 ;,而 SCSS 书写方式更像 CSS。

变量

Sass 变量很简单:您可以为以 $ 开头的名称分配一个值,然后可以引用该名称而不是值本身。

$red: #833;

body {
  color: $red;
}

变量可以减少重复,进行复杂的数学运算,配置库等。

CSS 变量与 Sass 变量的区别

  • Sass 变量都是由 Sass 编译出来的。CSS 变量包含在 CSS 中输出。
  • CSS 变量对于不同的元素可以有不同的值,但是 Sass 变量一次只有一个值。
  • Sass 变量是必需的,这意味着如果您使用一个变量然后更改它的值,那么前面的使用将保持不变。CSS 变量是声明性的,这意味着如果您更改该值,它将影响前面使用和后面使用的。

嵌套

Sass 允许开发人员以嵌套的方式使用 CSS。

.markdown-body {
  a {
    color: blue;
    &:hover {
      color: red;
    }
  }
}

引用父级选择器 &

Sass 允许在嵌套的代码块内,使用 & 引用父元素

a {
  font-size: 14px;
  font-weight: bold;
  text-decoration: none;
  &:hover { 
    text-decoration: underline; 
    font-size: 20px;
  }
}

嵌套属性

CSS 许多属性都位于相同的命名空间,例如 text-aligntext-decorationtext-transform 都位于 font 命名空间下,Sass 当中只需要编写命名空间一次,后续嵌套的子属性都将会位于该命名空间之下。

text: {
  align: center;          // 将编译为 text-align: center
  decoration: underline;  // 将编译为 text-decoration: underline
  transform: uppercase;   // 将编译为 text-transform: uppercase
}

注意,text 后面必须加上冒号。

注释

// 单行注释,该注释方式只保留在 Sass 源文件中,编译后被省略。此注释不会包含在 CSS 中。

/* 多行注释,该注释方式会保留到编译后的文件。但压缩模式下,仍会被删除 */
/* 多行注释可以使用插值表达式 #{1 + 1} */

/*! 即使在压缩模式下也会包含此注释。*/

继承

@extend 指令用于继承另一个选择器的样式。相同样式直接继承,不会造成代码冗余;

.button {
  font-size: 14px;
  color: plum;
}

.push-button {
  @extend .button;
}

混合

混合(Mixin)允许您定义可以在整个样式表中重用的样式。

语法

@mixin  { ... }

// 带参数
@mixin name(<arguments...>) { ... }

示例

@mixin heading-font {
  font-family: sans-serif;
  font-weight: bold;
}

h1 {
  @include heading-font;
}

Mixin 特性在添加浏览器兼容性前缀的时候非常有用。

@mixin border-radius($radius) {
  border-radius: $radius;
  -ms-border-radius: $radius;
  -moz-border-radius: $radius;
  -webkit-border-radius: $radius;
}

.card {
  @include border-radius(8px);
}

参数

你可以向 Mixin 传递变量参数来让代码更加灵活。

@mixin font-size($n) {
  font-size: $n * 1.2em;
}

body {
  @include font-size(2);
}

具有默认值

@mixin pad($n: 10px) {
  padding: $n;
}

body {
  @include pad(15px);
}

具有默认变量

// 设置默认值
$default-padding: 10px;
@mixin pad($n: $default-padding) {
  padding: $n;
}

body {
  @include pad(15px);
}

引入 SCSS 模块

Sass 能够将代码分割为多个片段,并以下划线作为其命名前缀。Sass 会通过这些下划线来辨别哪些文件是 Sass 片段,并且不让片段内容直接生成为 CSS 文件,只在使用 @import 指令的位置被导入。

// _variables.scss
$gray-1: #eee;

// index.scss
@import './_variables';

.scss.sass 扩展名是可选的。

函数

颜色功能

颜色设置

Sass 提供了一些内置的颜色函数,以便生成系列颜色。

// RGBA
rgb(100, 120, 140)
rgba(100, 120, 140, .5)
rgba($color, .5)

// 修改 HSLA
complement($color)    // 像 adjust-hue(_, 180deg)
invert($color)
grayscale($color)
hsla(hue, saturation, lightness, alpha)

颜色操作

// 混合
mix($a, $b, 10%)   // 10% a, 90% b

// 修改 HSLA
darken($color, 5%)
lighten($color, 5%)
saturate($color, 5%)
desaturate($color, 5%)
adjust-hue($color, 15deg)
fade-in($color, .5)   // 又名 opacify()
fade-out($color, .5)  // aka transparentize() - 将不透明度减半
rgba($color, .5)      // 将 alpha 设置为 .5

// 调整
// 按固定数量变动
adjust-color($color, $blue: 5) 
adjust-color($color, $lightness: -30%)   // 像 darken(_, 30%)
adjust-color($color, $alpha: -0.4)       // 像 fade-out(_, .4)
adjust-color($color, $hue: 30deg)        // 像 adjust-hue(_, 15deg)

// 通过百分比更改
scale-color($color, $lightness: 50%)

// 完全更改一个属性
change-color($color, $hue: 180deg)
change-color($color, $blue: 250)

支持:$red$green$blue$hue$saturation$lightness$alpha

颜色获取

// HSLA
hue($color)         // 0deg-360deg 返回颜色在 HSL 色值中的角度值 (0deg - 255deg)。
saturation($color)  // 0%-100%
lightness($color)   // 0%-100%
alpha($color)       // 0-1 (类似 opacity())

// RGB
red($color)   // 从一个颜色中获取其中红色值(0-255)
green($color)
blue($color)

其他功能

字符串

unquote('hello')
quote(hello)
to-upper-case(hello)
to-lower-case(hello)
str-length(hello world)
str-slice(hello, 2, 5)      // "ello" - it's 1-based, not 0-based
str-insert("abcd", "X", 1)  // "Xabcd"

单位

unit(3em)        // 'em'
unitless(100px)  // false

数字

floor(3.5)
ceil(3.5)
round(3.5)
abs(3.5)
min(1, 2, 3)
max(1, 2, 3)
percentage(.5)   // 50%
random(3)        // 0..3

选择器

selector-append('.menu', 'li', 'a')   // .menu li a 
selector-nest('.menu', '&:hover li')  // .menu:hover li
selector-extend(...)
selector-parse(...)
selector-replace(...)
selector-unify(...)

其他

variable-exists(red)    // 判断变量是否在当前的作用域下。
mixin-exists(red-text)  // 检测指定混入 (mixinname) 是否存在。
function-exists(redify) // 检测指定的函数是否存在
global-variable-exists(red) // 检测某个全局变量是否定义。

循环语句

Sass 提供了以下循环:

for 循环

@for $i from 1 through 4 {
  .item-#{$i} { left: 20px * $i; }
}

each 循环(简单)

$menu-items: home about services contact;

@each $item in $menu-items {
  .photo-#{$item} {
    background: url('images/#{$item}.jpg');
  }
}

each 循环(嵌套)

$backgrounds: (home, 'home.jpg'), (about, 'about.jpg');

@each $id, $image in $backgrounds {
  .photo-#{$id} {
    background: url($image);
  }
}

while 循环

$i: 6;
@while $i > 0 {
  .item-#{$i} { width: 2em * $i; }
  $i: $i - 2;
}

条件语句

@if $position == 'left' {
   position: absolute;
   left: 0;
} @else if $position == 'right' {
   position: absolute;
   right: 0;
} @else {
   position: static;
}

插值语句

通过 #{} 插值语句(interpolation)以在选择器或属性名中使用变量:

$name: card;
$attr: border;

.#{$class} { 
  #{$attr}-color: plum; 
}

@media #{$tablet}
// 使用 #{} 可以避免 Sass 运行运算表达式,直接编译 CSS。
font: #{$size}/#{$line-height}
url("#{$background}.jpg")

列表

Sass 列表(List)函数用于处理列表,可以访问列表中的值,向列表添加元素,合并列表等。

Sass 列表是不可变的,因此在处理列表时,返回的是一个新的列表,而不是在原有的列表上进行修改。

列表的起始索引值为 1,记住不是 0。

$list: (a b c);

nth($list, 1)  // 从 1 开始
length($list)  // 返回列表的长度

// 其他属性
nth // 函数可以直接访问数组中的某一项
join // 函数可以将多个数组连接在一起
list-separator(list) // 返回一列表的分隔符类型。可以是空格或逗号。
append(list, value, [separator]) // 将单个值 value 添加到列表尾部。separator 是分隔符,默认会自动侦测,或者指定为逗号或空格
is-bracketed(list) // 判断列表中是否有中括号
index(list, value) // 返回元素 value 在列表中的索引位置。
zip($list)      // 将多个列表按照以相同索引值为一组,重新组成一个新的多维度列表。
set-nth(list, n, value) // 设置列表第 n 项的值为 value。

// 遍历列表中的每一项
@each $item in $list { ... }

Map

Sass Map(映射)对象是以一对或多对的 key/value 来表示。

Sass Map 是不可变的,因此在处理 Map 对象时,返回的是一个新的 Map 对象,而不是在原有的 Map 对象上进行修改。

$map: (key1: value1, key2: value2, key3: value3);

map-get($map, key1)       // 返回 Map 中 key 所对应的 value(值)。如没有对应的 key,则返回 null 值。
map-has-key($map, key1)   // 判断 map 是否有对应的 key,存在返回 true,否则返回 false。
map-keys($map)            // 返回 map 中所有的 key 组成的队列。
map-merge($map1, $map2)   // 合并两个 map 形成一个新的 map 类型,即将 map2 添加到 map1的尾部
map-remove($map, keys...) // 移除 map 中的 keys,多个 key 使用逗号隔开。
map-values($map)          // 返回 map 中所有的 value 并生成一个队列。

更多资料

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,723评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,003评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,512评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,825评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,874评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,841评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,812评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,582评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,033评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,309评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,450评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,158评论 5 341
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,789评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,409评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,609评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,440评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,357评论 2 352

推荐阅读更多精彩内容