你可能已经据说了一个“大新闻”:Bootstrap4 合并了代号为#21389的PR,宣布放弃支持IE9,并默认使用flexbox弹性盒模型。
这标志着:
1)前端开发全面步入“现代浏览器”的时代进一步来临;
2)样式处理也再一次面向将来,拥抱更加灵活的弹性盒模型-Flex布局。css
这篇文章会带你深刻Bootstrap最新版源码,窥探其架构组织奥秘,并解析最具亮点的栅格化系统。
同时,你也会了解到sass的高阶用法和flex最新语法的奥秘。前端
在开启咱们的探索以前,有必要先梳理一下BS4添加的新特性:
1)从Less迁移到Sass:
如今,Bootstrap已加入Sass的你们庭中。得益于Libsass(Sass 解析器),Bootstrap的编译速度比之前更快;git
2)改进网格系统:
新增一个网格层适配移动设备,并整顿语义混合。github
3)默认弹性盒模型(flexbox):
这是项划时代的变更,利用flexbox的优点快速布局。web
4)废弃了wells、thumbnails和panels,使用cards代替。bootstrap
5)新的自定义选项:
再也不像上个版本同样,将渐变、淡入淡出、阴影等效果分放在单独的样式表中。而是将全部选项都移到一个Sass变量中。
想要给全局或考虑不到的角落定义一个默认效果?很简单,只要更新变量值,而后从新编译就能够了。浏览器
6)使用rem和em单位。sass
7)重构全部JavaScript插件:
Bootstrap 4用ES6重写了全部插件。如今提供UMD支持、泛型拆解方法、选项类型检查等特性。架构
8)改进工具提示和popovers自动定位:
这部分要感谢Tether(A positioning engine to make overlays, tooltips and dropdowns better)工具的帮助,
若是你还不知道Tether是什么,能够去他家Github地址。app
了解了以上新特性,咱们主要研究BS从诞生以来最大的“卖点”-栅格系统。
咱们选取表明性的BS4官网范例,能够在线参考, 或者参看如下截图,
在宽屏幕下,咱们看到:
当屏幕宽度小于576px时候,咱们有:
对应代码:
<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: 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这里设置为比例值,这也是响应式天然而然实现的基础。
看到此,不难明白BS4栅格的实现,可是这并非此文的最终目的。咱们能够深刻更多,好比,BS4的栅格系统里,一行一共是12栏。他的媒体查询断点又包括:xs: 0, sm: 576px, md: 768px, lg: 992px, xl: 1200px。
参考其源码dist/css目录下样式代码,咱们会想组织生成如此大量的CSS样式,不用预处理器简直是反人类的。而BS4倒是把sass用到了极致。
咱们深刻其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这一神器有了更加深入的认识。