深入新版BS4源码 探索flex和工程化sass奥秘

你可能已经听说了一个“大新闻”:Bootstrap4 合并了代号为#21389的PR,宣布放弃支持IE9,并默认使用flexbox弹性盒模型布局。
这标志着:
1)前端开发全面步入“现代浏览器”的时代进一步来临;
2)样式处理也再一次面向未来,拥抱更加灵活的弹性盒模型-Flex布局。

这篇文章会带你深入Bootstrap最新版源码,窥探其架构组织奥秘,并解析最具亮点的栅格化系统。
同时,你也会了解到sass的高阶用法和flex最新语法的奥秘。

BS4的新特性

在开启我们的探索之前,有必要先梳理一下BS4添加的新特性:
1)从Less迁移到Sass:
现在,Bootstrap已加入Sass的大家庭中。得益于Libsass(Sass 解析器),Bootstrap的编译速度比以前更快;

2)改进网格系统:
新增一个网格层适配移动设备,并整顿语义混合。

3)默认弹性盒模型(flexbox):
这是项划时代的变动,利用flexbox的优势快速布局。

4)废弃了wells、thumbnails和panels,使用cards代替。

5)新的自定义选项:
不再像上个版本一样,将渐变、淡入淡出、阴影等效果分放在单独的样式表中。而是将所有选项都移到一个Sass变量中。

6)使用rem和em单位。

7)重构所有JavaScript插件:
Bootstrap 4用ES6重写了所有插件。现在提供UMD支持、泛型拆解方法、选项类型检查等特性。

8)改进工具提示和popovers自动定位:
这部分要感谢Tether(A positioning engine to make overlays, tooltips and dropdowns better)工具的帮助,
如果你还不知道Tether是什么,可以去他家Github地址了解一下,源码也短小精悍,值得一读。

BS4栅格系统揭秘

了解了以上新特性,我们主要研究BS从诞生以来最大的“卖点” — 栅格系统。

一个栅格实例

我们选取代表性的BS4官网范例,可以在线参考, 或者参看以下截图,
在宽屏幕下,我们看到:

宽屏幕状态下

当屏幕宽度小于576px时候,我们有:

iphone5屏幕下

对应代码:

<div class="col-6 col-sm-3">
    ...
</div>
<div class="col-6 col-sm-3">
    ...
</div>
<div class="col-6 col-sm-3">
    ...
</div>
<div class="col-6 col-sm-3">
    ...
</div>

.col-6 class样式在源码里面可以简单归纳(不完全)为:

.col-6 {
    -webkit-box-flex: 0;
    -webkit-flex: 0 0 50%;
    -ms-flex: 0 0 50%;
    flex: 0 0 50%;
    max-width: 50%;
}

.col-sm-3 class在源码里面可以归纳为:

.col-sm-3{
    -webkit-box-flex: 0;
    -webkit-flex: 0 0 25%;
        -ms-flex: 0 0 25%;
            flex: 0 0 25%;
    max-width: 25%;
}

两种类的共存和交替作用

我们看到,代码里设置了这两个class进行样式声明,很明显他们的样式属性是有冲突的,那么他们是如何做到“和平共处”交替发挥作用的呢?

1)在屏幕宽度大于576px时候,我们发现.col-sm-3并没有起作用,这时候起作用的是.col-6。
我们在源码里发现.col-sm-的样式声明全部在@media (min-width: 576px) {...}的媒体查询中,
这就保证了在576px宽度以上的屏幕,只有在媒体查询之外的.col-
样式声明发挥了作用。

2)在屏幕宽度小于576px时候,命中媒体查询,命中.col-sm-3的样式声明。他的优先级一定大于.col-6(媒体查询优先级高),这时候就保证了移动端的样式“占上风”。

flex讲解

从上面样式代码里看到类似flex: 0 0 25%的声明,为了理解它,我们从flex属性入手:
flex属性是flex-grow, flex-shrink 和 flex-basis的简写(类似backgroud是很多背景属性的简写一样),
它的默认值为0 1 auto。后两个属性可选。语法格式如下:

.item {
    flex: none | [ <'flex-grow'> <'flex-shrink'>? || <'flex-basis'> ]
}

1)flex-grow:属性定义项目的放大比例,默认为0。我们看到BS代码里这个值一直为0,即如果存在剩余空间,也不放大。

2)flex-shrink:属性定义了项目的缩小比例,默认为1,即如果空间不足,该项目将缩小。

3)flex-basis:属性定义了在分配多余空间之前,项目占据的主轴空间(main size)。
浏览器根据这个属性,计算主轴是否有多余空间。它可以设为跟width或height属性一样的值(比如350px),则项目将占据固定空间。
当然BS4这里设置为比例值,这也是响应式自然而然实现的基础。

SASS在BS4工程化中的伟大作用

看到此,不难明白BS4栅格是如何实现的,但是这并不是此文的最终目的。我们可以深入更多:比如,BS4的栅格系统里,一行一共是12栏。他的媒体查询断点又包括:xs: 0, sm: 576px, md: 768px, lg: 992px, xl: 1200px。
我们会想组织生成如此大量的CSS样式,不用预处理器简直是反人类的。而BS4却是把sass用到了极致。

参考其源码dist/css目录下样式代码,我们进行分析:

.col-sm-*是如何生成的

我们深入其scss目录下,scss/mixins/_grid.scss文件:

@if $enable-grid-classes {
    @include make-grid-columns();
}

在enable-grid-classes变量为true的情况下(默认为true),调用make-grid-columns,make-grid-columns()这个mixin定义在scss/mixins/_grid-reamework.scss文件中(找的我好心累。。。):

@mixin make-grid-columns($columns: $grid-columns, $gutters: $grid-gutter-widths, $breakpoints: $grid-breakpoints) {
    ...
}

这个mixin接受三个参数:
1)$columns表示栅格数目,默认为12;
2)$gutters默认为30
3)$breakpoints表示断点设置,这是一个全局变量,为map类型。
这些你可以在scss/mixins/_breakpoints.scss文件中和scss/_variables.scss文件中找到。

认识了这些参数,我们看.col-sm-具体实现,下面代码我已经进行过大范围精简,只保留col-sm-相关部分,并且加了注释:

@each $breakpoint in map-keys($breakpoints) {
    // Returns a blank string if smallest breakpoint, otherwise returns the name with a dash infront.
    $infix: breakpoint-infix($breakpoint, $breakpoints);
    // Media of at least the minimum breakpoint width. No query for the smallest breakpoint.
    // Makes the @content apply to the given breakpoint and wider.
    @include media-breakpoint-up($breakpoint, $breakpoints) {
        @for $i from 1 through $columns {
            .col#{$infix}-#{$i} {
                @include make-col($i, $columns);
            }
        }
    }
}

我们一步一步来分析:
1)@each $breakpoint in map-keys($breakpoints),对每一个断点进行遍历;
2)breakpoint-infix是一个函数,它定义在css/mixins/_breakpoints.scss文件当中, 返回一个带破折号的断点标识字符串,比如这里我们关系的就是“-sm”;
3)media-breakpoint-up是一个mixin:

@mixin media-breakpoint-up($name, $breakpoints: $grid-breakpoints) {
    $min: breakpoint-min($name, $breakpoints);
    @if $min {
        @media (min-width: $min) {
        @content;
    }
    } @else {
        @content;
    }
}

4)breakpoint-min又是一个函数,它返回了断点的具体数值。这里是用来拼媒体查询条件的。
5)最后最关键样式的生成又使用了另外一个定义在css/mixins/_grid.scss文件当中的mixin:

@mixin make-col($size, $columns: $grid-columns) {
    flex: 0 0 percentage($size / $columns);
    max-width: percentage($size / $columns);
}

到此为止,我们深入了Bootstrap V4的scss/目录下的源码,研究涉及了:
css/mixins/_grid-framework.scss文件;
css/mixins/_breakpoints.scss文件;
css/mixins/_grid.scss文件;
css/_variables.scss文件;
css/bootstrap-gris.scss文件;
......

如果你理解了这些,那么再去读bootstrap新版源码就不会存在任何难度。相信你也能够在全局上,以“上帝视角”了解sass所起的作用,大型样式框架的架构组织。

总结

通过阅读源码的栅格系统部分,我们应该认识到sass对于这样大型样式框架系统的重要意义:
1)css模块化在管理组织上的灵活性;
2)复用的意义:使用了大量的mixin,function,全局变量;
3)像JS一样神奇的语法,包括条件、遍历等等等等。
同时,你也应该对flex这一神器有了更加深刻的认识。

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

推荐阅读更多精彩内容

  • 声明变量 定义变量的语法: 在有些编程语言中(如,JavaScript)声明变量都是使用关键词“var”开头,但是...
    Junting阅读 1,469评论 0 6
  • 想想学习Sass时间也有差不多一年的光景了。在这一年来的时间中,在GitHub不断阅读Sass相关的源码。也在国外...
    小豌豆书吧阅读 7,242评论 1 24
  • Sass 是对 CSS 的扩展,让 CSS 语言更强大、优雅。 它允许你使用变量、嵌套规则、mixins、导入等众...
    八宝君阅读 2,012评论 0 1
  • 学习Sass(官网:Sass)之前需要了解什么是Sass,Sass全称:Syntactically Awesome...
    haoxilu阅读 487评论 0 3
  • 《礼记·中庸》有言:“凡是预则立,不预则废。言前定则不跲,事前定则不困,行前定则不疚,道前定则不穷。”意即凡事如果...
    上海Jimmy刘阅读 323评论 0 1