zoukankan      html  css  js  c++  java
  • 【CSS】519- grid-auto-flow深入理解

    作者:张鑫旭

    原文:https://www.zhangxinxu.com/wordpress/2020/01/css-grid-auto-flow/

    一、grid-auto-flow出现的背景

    Grid布局要想玩得666,grid-auto-flow是绕不开的需要深度掌握的CSS属性。

    Grid布局就像农村土改分土地。

    政府会把土地分成一块一块的,然后这一块是张三的,那一块是王二的。

    这个划分就是通过grid-template属性。

    比方说:

    .fileds {
        display: grid;
        grid-template: 1fr 1fr / 1fr 1fr;
    }

    就是把这块土地分成两行,两列4块田地。

    然而应用到具体的场景,事情往往就出乎意料的复杂。

    例如:

    1. 如果日后村里搬迁了几户新住户,那该怎么划分?

    2. 4块田地分给张三和王二,你说是横的两块地给张三呢,还是竖的两块地给张三?

    3. 孤寡老人李大爷过世了,他的田地是否可以分给后面的人?

    这些意料之外的场景单靠grid-template属性肯定捉襟见肘,此时,就需要grid-auto-flow属性出马。

    grid-auto-flow属性可以定义Grid布局中每一个Grid子项“自动流动”状态。

    二、了解grid-auto-flow的语法

    grid-auto-flow: [ row | column ] || dense

    因此,下面这些写法都是合法的:

    grid-auto-flow: row;
    grid-auto-flow: column;
    grid-auto-flow: dense;
    grid-auto-flow: row dense;
    grid-auto-flow: column dense;

    其中:

    • row

    • 多的格子一行一行陈列。默认值。

    • column

    • 多的格子一列一列排列。

    • dense

    • 多的格子空白填充。

    各个关键字值具体什么意思,我们还得看案例才能明白。

    三、row和column属性值基础

    1. 默认值row

    从天上掉下来一个宝贝容器,定睛一看,容器元素标签名是zxx-grid,好巧,正好应用的CSS声明就是display:grid,如下:

    zxx-grid {
        display: grid;
    }

    然后很巧的是还有若干子元素,如下:

    <zxx-grid>
        <zxx-item>格子1</zxx-item>
        <zxx-item>格子2</zxx-item>
        <zxx-item>格子3</zxx-item>
        <zxx-item>格子4</zxx-item>
        <zxx-item>格子5</zxx-item>
        <zxx-item>格子6</zxx-item>
        <zxx-item>格子7</zxx-item>
        <zxx-item>格子8</zxx-item>
        <zxx-item>格子9</zxx-item>
    </zxx-grid>

    <zxx-grid>元素除了应用了display:grid没有任何其他CSS,因此,所有的子元素的grid-auto-flow“自动流动”的状态是其默认值。

    grid-auto-flow的默认值是rowrow的中文意思是“行”,也就是所有的子元素一行一行显示,优先水平排列。

    于是,最终的网格轮廓结构会是下图这样:

    更进一步

    如果我们使用grid-template属性指定Grid的列数为2列,如下:

    zxx-grid {
        display: grid;
        grid-template: auto / 1fr 1fr;
        line-height: 40px;
    }

    大家就会看到Grid子元素的排列流向是下面这样的,先是水平排列,然后再到下一行再水平排列,一直到最后不断循环:

    大家可能觉得理所当然就应该这样排列,太天真了!

    大家稍后对比column属性值的效果就知道Grid子项的排列顺序并不是一成不变的了。

    2. 关键字值column

    如果我们设置grid-auto-flow的属性值是columncolumn的中文意思是“列”,则表示“自动流动”状态的子元素全部都一列一列显示。

    假设容器元素CSS如下:

    zxx-grid {
        display: grid;
        grid-auto-flow: column;
    }

    则几个子元素的布局效果会是下面截图这样,全部纵排了:

    发现没有?使用grid-auto-flow属性可以非常简单实现子元素个数不固定的等分布局效果,就是值设置为column就可以了。

    更进一步

    这里,我们同样指定我们的Grid布局是2行,CSS代码如下:

    zxx-grid {
        display: grid;
        grid-template-rows: 1fr 1fr;
        grid-auto-flow: column;
        line-height: 40px;
    }

    大家就会发现Grid子元素的排列流向和row属性值就不一样了:

    是优先垂直方向排列,然后再到下一列往下排列,一直循环往复到最后一列。

    由此,我们可以得出结论,grid-auto-flow属性会影响Grid布局的排列顺序。

    3. 排序与布局实例

    有人在微信粉丝群里面问了下面这个问题,实现下图所示布局:

    这个布局效果,最简单的方法就是用浮动3个图片都设置float:left就可以了,就这么简单。

    但是如果我们这里的图片不是3个,而是5个(如下图),那么浮动就无能为力了。

    此时,则可以试试Grid布局,让第一个子项横跨左侧两个格子,然后后面4个格子按照grid-auto-flow:column的流向排列即可。

    HTML代码如下:

    <zxx-grid>
        <zxx-item><img src="./xugou-1.jpg"></zxx-item>
        <zxx-item><img src="./xugou-2.jpg"></zxx-item>
        <zxx-item><img src="./xugou-3.jpg"></zxx-item>
        <zxx-item><img src="./xugou-4.jpg"></zxx-item>
        <zxx-item><img src="./xugou-5.jpg"></zxx-item>
    </zxx-grid>

    CSS代码如下:

    zxx-grid {
        display: grid;
        grid-auto-flow: column;
        grid-template: 1fr 1fr / 1fr 1fr 1fr;
        grid-template-areas: "a b d" "a c e";
        grid-gap: 6px;
    }
    zxx-item:first-child {
        grid-area: a;
    }

    则有如下图所示的布局效果:

    而且整个布局是弹性变化的,如下视频所示(不动可以点击播放):

    对吧,布局效果还是很酷的。


    最后,测试下grid-auto-flow:column在这个例子中所起的作用,我们打开控制台,注释掉grid-auto-flow:column这句CSS声明,则我们会得到下图所示的布局效果,图2到图5都是水平平铺呈现,这个效果Float浮动就可以实现。

    对应的Demo页面

    上面的例子有专门的demo演示页面,您可以狠狠的点击这里:grid布局实现图像A|B+C布局demo

    //zxx: Chrome浏览器的图片height:100%似乎有bug,Firefox浏览器没有这个问题

    四、row和column属性值深入

    第三小节演示的几个案例都是我们所有的格子都已经规定好的,但是实际开发,Grid格子的数量可能是动态的,无论是grid-template还是grid-template-areas往往就没法兼顾全部的子元素,此时,Grid布局又该如何表现?

    为了更进一步展示关键字值column的效果,我们来看一个稍微再复杂一点的例子:

    我们设定网格模板是 2 x 2,也就是前4个元素按照指定网格陈列,后面的子元素auto-flow自由流动。相关CSS如下:

    zxx-grid {
        display: grid;
        grid-template: 1fr 1fr / 1fr 1fr;
        grid-auto-flow: column;
    }

    则最终效果如下:

    为什么会有上图这样的效果呢?

    首先大家一定要搞清楚一点,1fr 1fr并不等于50% 50%,虽然很多时候表现是一样的,但只是grid格子正好和设定数量一致的时候。1fr 1fr更准确的含义是,可用剩余空间分别50% 50%

    在本例中,格子5~格子9属于“放任自流”的盒子,因为grid-template:1fr 1fr / 1fr 1fr只指定了4个格子的尺寸,那剩余的5个格子怎么办呢?

    那就是grid-auto-flow属性大发神威的时候。

    这里grid-auto-flow设置的属性值是column,也就是没有指定模板的剩余5个格子垂直按需排列。如下图标注:

    由于没有指定宽度,因此,格子的尺寸就fit-content,根据内容的尺寸来。


    在实际开发的时候,我们往往只要指定好垂直列的个数和宽度分配比,无论多少个grid子元素,都会依次排列好,其最底层最根本的原因其实是grid-auto-flow:row在起作用。

    例如:

    zxx-grid {
        display: grid;
        grid-template-columns: 1fr 1fr;
    }

    效果是下面这样,9个格子了,一个一个排的非常整齐,因为“自动流动”默认是水平的。

    五、不太好懂的dense属性值

    最后,再讲讲dense属性值。

    dense单从字面含义是不太好理解的,dense的中文意思是“密集的”,“稠密的”,“浓密的”。

    密集的是树林,稠密的是奶昔,浓密的是头发,跟Grid布局好像不搭边吧。

    实际上,这里dense应该用一种更感性的方式去体会这个词的含义。

    在本文一开始提到了一个引子“孤寡老人李大爷过世了,他的田地是否可以分给后面的人?”

    比如说村里9户人家,每一户都分得了一块田地,大家的田地都是紧密相连的。

    结果孤寡老人李大爷和王大爷相继去世。于是,就有两块田地是空缺的,因为有空缺,所以这个时候,如果我们使用无人机从上方拍一个照片,则田地是稀疏的,不是紧密的。

    这个时候,村里又来了两户新人家,也要分田地。

    如果是dense,则优先分配之前李大爷和王大爷留下的空缺的土地,因为这样会让整片田地没有空缺,是“密集的”,“紧密的”。

    grid-auto-flow:dense也是类似的意思。

    dense理解案例

    我们可以人为制造出稀疏布局结构(前面几个Grid格子留空),代码如下:

    zxx-grid {
        display: grid;
        grid-template-columns: 1fr 1fr;
    }
    zxx-item:first-child {
        grid-column-start: 2;
    }

    效果如下,第一个格子空缺了:

    如果我们希望第一个格子被充分利用,让整个排列是紧密相连的,则可以使用dense关键字值。

    zxx-grid {
        display: grid;
        grid-template-columns: 1fr 1fr;
        grid-auto-flow: dense;
    }
    zxx-item:first-child {
        grid-column-start: 2;
    }

    此时浏览器的渲染表现是怎样的呢?请看下图:

    原本流出来的第1个格子的窟窿,被格子2给填上了。

    这个Grid布局又是“紧密的”了,这就是dense属性的作用。

    dense实际开发的时候有什么作用呢?

    比方说我希望某两个相邻的格子视觉顺序和DOM顺序是相反的,也就是两者调换位置,则可以使用dense实现。

    其他

    dense可以和row以及column这两个关键字只是同时使用,例如:

    grid-auto-flow: row dense;  // 等同于dense
    grid-auto-flow: column dense;

    六、最后的总结么么哒

    本文虽然只有5000字,不过花的时间还挺多,算上今天,断断续续写了一周。

    为什么这篇文章跨度时间比较长呢?除了用心做demo,用心创作外,还有其他两方面主要原因:

    1. 边研究边测试边写,结果发现了自己一开始想当然的错误理解,然后不少内容重新组织推翻重来。因此可见,写文章其实对深度学习、了解细节,理清思路非常有帮助,是非常好的一种学习方法。

    2. 现在上年纪了,体力没以前好了。以前都是写到1点,这种程度的文章2~3天就完成。现在顶不住了,12点左右就困得不行,关机睡觉去了,所以写作时间跨度就更长了。

    CSS文章已经连更十几篇了,小本本上还有好几篇CSS文章要写的。

    光CSS基础API只是更新迭代的东西就这么多,再考虑到吃了激素一路狂奔年年暴走的ES规范,现在的前端开发想成为通才基本上不可能了,只能选择合适的领域,一点一点发光发热吧。

    哎呀呀,就说这么多了。

    用心之作,欢迎分享!

    行文匆忙,如果文章有表述不准确的地方,欢迎指正!

    原创系列推荐

    1. JavaScript 重温系列(22篇全)

    2. ECMAScript 重温系列(10篇全)

    3. JavaScript设计模式 重温系列(9篇全)

    4. 正则 / 框架 / 算法等 重温系列(16篇全)

    5. Webpack4 入门(上)|| Webpack4 入门(下)

    6. MobX 入门(上) ||  MobX 入门(下)

    7. 59篇原创系列汇总

    回复“加群”与大佬们一起交流学习~

    点这,与大家一起分享本文吧~

    个人博客:http://www.pingan8787.com 微信公众号【前端自习课】和千万网友一起,每日清晨,享受一篇前端优秀文章。 目前已连续推送文章 600+ 天,愿每个人的初心都能一直坚持下去!
  • 相关阅读:
    网页打开速度的心理学
    UML建模——用例图(Use Case Diagram)
    漫谈干系人识别管理
    干系人分析的3个方法:除了目标用户还要考虑谁?
    计算机网络-复习笔记
    项目经理必掌握的国际项目管理知识体系结构及内容
    剑指Offer面试题:5.重建二叉树
    剑指Offer面试题:4.从尾到头打印链表
    剑指Offer面试题:3.替换空格
    剑指Offer面试题:2.二维数组中的查找
  • 原文地址:https://www.cnblogs.com/pingan8787/p/13069475.html
Copyright © 2011-2022 走看看