美文网首页
grid布局

grid布局

作者: AAA前端 | 来源:发表于2021-11-09 14:04 被阅读0次

    网格布局(Grid)

    它将网页划分成一个个网格,可以任意组合不同的网格,做出各种各样的布局。以前,只能通过复杂的 CSS 框架达到的效果,现在浏览器内置了。

    Grid 布局与 Flex 布局有一定的相似性,都可以指定容器内部多个项目的位置。但是,它们也存在重大区别。

    Flex 布局是轴线布局,只能指定"项目"针对轴线的位置,可以看作是一维布局。Grid 布局则是将容器划分成"行"和"列",产生单元格,然后指定"项目所在"的单元格,可以看作是二维布局。Grid 布局远比 Flex 布局强大。

    比如 flex布局中两侧对齐,最后一排,如果元素不够,则左对齐。flex中往往采用生成一个元素,页面不可见实现,而grid布局则可以简单的实现。

    浏览器兼容性

    很多人总觉得 CSS Grid 的兼容性不行。其实不然,下图的统计数据能够看出大部分浏览器对 CSS Grid 的支持还是不错的,要知道 Flex 在 IE10 上面可以使用的属性也非常有限。据统计目前 CSS Grid 使用率和 Flex 已经基本持平了。所以兼容性并不能回答开篇提出的问题。


    image.png image.png

    基本概念

    1.容器---有容器属性(container) 和 项目---有项目属性(items)

    采用网格布局的区域,称为"容器"(container)。容器内部采用网格定位的子元素,称为"项目"(item)。

    
    <div>
      <div><p>1</p></div>
      <div><p>2</p></div>
      <div><p>3</p></div>
    </div>
    

    上面代码中,最外层的<div>元素就是容器,内层的三个<div>元素就是项目。

    注意:项目只能是容器的顶层子元素,不包含项目的子元素,比如上面代码的<p>元素就不是项目。Grid 布局只对项目生效。

    1. 行和列
      容器里面的水平区域称为"行"(row),垂直区域称为"列"(column)。


      image.png

    水平的深色区域就是"行",垂直的深色区域就是"列"。

    1. 单元格
      行和列的交叉区域,称为"单元格"(cell)。

    正常情况下,n行和m列会产生n x m个单元格。比如,3行3列会产生9个单元格。

    1. 网格线
      划分网格的线,称为"网格线"(grid line)。水平网格线划分出行,垂直网格线划分出列。

    正常情况下,n行有n + 1根水平网格线,m列有m + 1根垂直网格线,比如三行就有四根水平网格线。


    image.png

    上图是一个 4 x 4 的网格,共有5根水平网格线和5根垂直网格线。


    image.png

    容器属性

    Grid 布局的属性分成两类。一类定义在容器上面,称为容器属性;另一类定义在项目上面,称为项目属性。这部分先介绍容器属性。

    display 属性

    display: grid指定一个容器采用网格布局。

    grid-template-columns 属性,grid-template-rows 属性

    容器指定了网格布局以后,接着就要划分行和列。grid-template-columns属性定义每一列的列宽,grid-template-rows属性定义每一行的行高

     <div class="box">
        <div>
          <p>1</p>
        </div>
        <div>
          <p>2</p>
        </div>
        <div>
          <p>3</p>
        </div>
        <div>
          <p>4</p>
        </div>
        <div>
          <p>5</p>
        </div>
        <div>
          <p>6</p>
        </div>
        <div>
          <p>7</p>
        </div>
        <div>
          <p>8</p>
        </div>
      </div>
    
      .box {
        display: grid;
         grid-template-columns: 50px 100px 100px;
        grid-template-rows: 100px 100px 150px;
      }
    
      div {
        border: 1px solid #f00;
        margin: 5px;
      }
    
    image.png

    repeat()

    有时候,重复写同样的值非常麻烦,尤其网格很多时。这时,可以使用repeat()函数,简化重复的值。上面的代码用repeat()改写如下。

      .box {
        display: grid;
         grid-template-columns: repeat(3, 100px);
        grid-template-rows: 100px 100px 150px;
      }
    
      div {
        border: 1px solid #f00;
        margin: 5px;
      }
    
    image.png

    auto-fill 关键字

    有时,单元格的大小是固定的,但是容器的大小不确定。如果希望每一行(或每一列)容纳尽可能多的单元格,这时可以使用auto-fill关键字表示自动填充。

      .box {
        display: grid;
        grid-template-columns: repeat(auto-fill, 100px);
        grid-template-rows: 100px 100px 150px;
      }
    
      div {
        border: 1px solid #f00;
        margin: 5px;
      }
    
    
    image.png

    fr 关键字

    为了方便表示比例关系,网格布局提供了fr关键字(fraction 的缩写,意为"片段")。如果两列的宽度分别为1fr和2fr,就表示后者是前者的两倍。

    .box{
         grid-template-columns: 1fr 2fr 1fr 1fr;
        grid-template-rows: 100px 100px 150px;
      }
    
      div {
        border: 1px solid #f00;
        margin: 5px;
      }
    
    image.png

    minmax()

    minmax()函数产生一个长度范围,表示长度就在这个范围之中。它接受两个参数,分别为最小值和最大值。

    .box{
          grid-template-columns: 1fr 1fr minmax(100px, 1fr);
        grid-template-rows: 100px 100px 150px;
      }
    
      div {
        border: 1px solid #f00;
        margin: 5px;
      }
    
    image.png
    image.png

    auto 关键字

    auto关键字表示由浏览器自己决定长度。

    .box{
        grid-template-columns: 100px auto 100px;
        grid-template-rows: 100px 100px 150px;
      }
    
      div {
        border: 1px solid #f00;
        margin: 5px;
      }
    
    image.png
    image.png

    网格线的名称

    grid-template-columns属性和grid-template-rows属性里面,还可以使用方括号,指定每一根网格线的名字,方便以后的引用。

    .container {
      display: grid;
      grid-template-columns: [c1] 100px [c2] 100px [c3] auto [c4];
      grid-template-rows: [r1] 100px [r2] 100px [r3] auto [r4];
    }
    

    上面代码指定网格布局为3行 x 3列,因此有4根垂直网格线和4根水平网格线。方括号里面依次是这八根线的名字。

    网格布局允许同一根线有多个名字,比如[fifth-line row-5]。

    布局实例

    grid-template-columns属性对于网页布局非常有用。两栏式布局只需要一行代码。

    .wrapper {
      display: grid;
      grid-template-columns: 70% 30%;
    }
    

    grid-row-gap 属性,grid-column-gap 属性,grid-gap 属性

    grid-row-gap属性设置行与行的间隔(行间距),grid-column-gap属性设置列与列的间隔(列间距)。
    grid-gap属性是grid-column-gap和grid-row-gap的合并简写形式,语法如下。
    grid-gap: <grid-row-gap> <grid-column-gap>;

      .box {
        display: grid;
         grid-template-columns: repeat(3, 100px);
        grid-template-rows: 100px 100px 150px;
        grid-row-gap: 10px;
        grid-column-gap: 20px;
      }
    
      div {
        border: 1px solid #f00;
      }
    
    image.png

    grid-template-areas 属性

    网格布局允许指定"区域"(area),一个区域由单个或多个单元格组成。grid-template-areas属性用于定义区域。

    .container {
      display: grid;
      grid-template-columns: 100px 100px 100px;
      grid-template-rows: 100px 100px 100px;
      grid-template-areas: 'a b c'
                           'd e f'
                           'g h i';
    }
    

    面代码先划分出9个单元格,然后将其定名为a到i的九个区域,分别对应这九个单元格。

    grid-auto-flow 属性

    划分网格以后,容器的子元素会按照顺序,自动放置在每一个网格。默认的放置顺序是"先行后列",即先填满第一行,再开始放入第二行,即下图数字的顺序。


    image.png

    这个顺序由grid-auto-flow属性决定,默认值是row,即"先行后列"。也可以将它设成column,变成"先列后行"。

    grid-auto-flow: column;
    
    image.png

    justify-items 属性,align-items 属性,place-items 属性

    justify-items属性设置单元格内容的水平位置(左中右),align-items属性设置单元格内容的垂直位置(上中下)。

    .container {
      justify-items: start | end | center | stretch;
      align-items: start | end | center | stretch;
    }
    

    start:对齐单元格的起始边缘。
    end:对齐单元格的结束边缘。
    center:单元格内部居中。
    stretch:拉伸,占满单元格的整个宽度(默认值)。

    place-items属性是align-items属性和justify-items属性的合并简写形式。
    place-items: <align-items> <justify-items>;

    justify-content 属性,align-content 属性,place-content 属性

    justify-content属性是整个内容区域在容器里面的水平位置(左中右),align-content属性是整个内容区域的垂直位置(上中下)。

    .container {
      justify-content: start | end | center | stretch | space-around | space-between | space-evenly;
      align-content: start | end | center | stretch | space-around | space-between | space-evenly;  
    }
    

    下面实现 左右两侧对齐,最后一排元素不足,左对齐

      .box {
        display: grid;
         justify-content: space-between;
         grid-template-columns: repeat(3, 100px);
        grid-template-rows: 100px 100px 150px;
        grid-row-gap: 10px;
        grid-column-gap: 20px;
      }
    
      div {
        border: 1px solid #f00;
      }
    
    image.png

    grid-auto-columns 属性,grid-auto-rows 属性

    有时候,一些项目的指定位置,在现有网格的外部。比如网格只有3列,但是某一个项目指定在第5行。这时,浏览器会自动生成多余的网格,以便放置项目。

    grid-auto-columns属性和grid-auto-rows属性用来设置,浏览器自动创建的多余网格的列宽和行高。它们的写法与grid-template-columnsgrid-template-rows完全相同。如果不指定这两个属性,浏览器完全根据单元格内容的大小,决定新增网格的列宽和行高。

    <!DOCTYPE html>
    <html>
    <head>
      <meta charset="utf-8">
      <meta name="viewport" content="width=device-width">
      <title>JS Bin</title>
    </head>
    <body>
    <div id="container">
      <div class="item item-1">1</div>
      <div class="item item-2">2</div>
      <div class="item item-3">3</div>
      <div class="item item-4">4</div>
      <div class="item item-5">5</div>
      <div class="item item-6">6</div>
      <div class="item item-7">7</div>
      <div class="item item-8">8</div>
      <div class="item item-9">9</div>
    </div>
    </body>
    </html>
    
    #container{
      display: grid;
      grid-template-columns: 100px 100px 100px;
      grid-template-rows: 100px 100px 100px;
      grid-auto-rows: 50px; 
    }
    
    .item {
      font-size: 2em;
      text-align: center;
      border: 1px solid #e5e4e9;
    }
    
    .item-1 {
      background-color: #ef342a;
    }
    
    .item-2 {
      background-color: #f68f26;
    }
    
    .item-3 {
      background-color: #4ba946;
    }
    
    .item-4 {
      background-color: #0376c2;
    }
    
    .item-5 {
      background-color: #c077af;
    }
    
    .item-6 {
      background-color: #f8d29d;
    }
    
    .item-7 {
      background-color: #b5a87f;
    }
    
    .item-8 {
      background-color: #d0e4a9;
      grid-row-start: 4;
      grid-column-start: 2;
    }
    
    .item-9 {
      background-color: #4dc7ec;
      grid-row-start: 5;
      grid-column-start: 3;
    }
    
    image.png

    上面代码指定新增的行高统一为50px(原始的行高为100px)。

    grid-template 属性,grid 属性

    grid-template属性是grid-template-columns、grid-template-rows和grid-template-areas这三个属性的合并简写形式。

    grid属性是grid-template-rows、grid-template-columns、grid-template-areas、 grid-auto-rows、grid-auto-columns、grid-auto-flow这六个属性的合并简写形式。

    从易读易写的角度考虑,还是建议不要合并属性,所以这里就不详细介绍这两个属性了。

    项目属性

    下面这些属性定义在项目上面。

    grid-column-start 属性,grid-column-end 属性,grid-row-start 属性,grid-row-end 属性

    项目的位置是可以指定的,具体方法就是指定项目的四个边框,分别定位在哪根网格线。

    grid-column-start属性:左边框所在的垂直网格线
    grid-column-end属性:右边框所在的垂直网格线
    grid-row-start属性:上边框所在的水平网格线
    grid-row-end属性:下边框所在的水平网格线

    .item-1 {
      grid-column-start: 2;
      grid-column-end: 4;
    }
    

    上面代码指定,1号项目的左边框是第二根垂直网格线,右边框是第四根垂直网格线。

    image.png

    只指定了1号项目的左右边框,没有指定上下边框,所以会采用默认位置,即上边框是第一根水平网格线,下边框是第二根水平网格线。

    除了1号项目以外,其他项目都没有指定位置,由浏览器自动布局,这时它们的位置由容器的grid-auto-flow属性决定,这个属性的默认值是row,因此会"先行后列"进行排列。读者可以把这个属性的值分别改成column、row dense和column dense,看看其他项目的位置发生了怎样的变化。

    grid-column 属性,grid-row 属性

    grid-column属性是grid-column-start和grid-column-end的合并简写形式,grid-row属性是grid-row-start属性和grid-row-end的合并简写形式。

    .item {
      grid-column: <start-line> / <end-line>;
      grid-row: <start-line> / <end-line>;
    }
    

    grid-area 属性

    grid-area属性指定项目放在哪一个区域。

    <!DOCTYPE html>
    <html>
    <head>
      <meta charset="utf-8">
      <meta name="viewport" content="width=device-width">
      <title>JS Bin</title>
    </head>
    <body>
    <div id="container">
      <div class="item item-1">1</div>
      <div class="item item-2">2</div>
      <div class="item item-3">3</div>
      <div class="item item-4">4</div>
      <div class="item item-5">5</div>
      <div class="item item-6">6</div>
      <div class="item item-7">7</div>
      <div class="item item-8">8</div>
      <div class="item item-9">9</div>
    </div> 
    </body>
    </html>
    
    #container{
      display: grid;
      grid-template-columns: 100px 100px 100px;
      grid-template-rows: 100px 100px 100px;
      grid-template-areas: 'a b c'
                         'd e f'
                         'g h i';
    }
    
    .item {
      font-size: 4em;
      text-align: center;
      border: 1px solid #e5e4e9;
    }
    
    .item-1 {
      background-color: #ef342a;
      grid-area: e;
    }
    
    .item-2 {
      background-color: #f68f26;
    }
    
    .item-3 {
      background-color: #4ba946;
    }
    
    .item-4 {
      background-color: #0376c2;
    }
    
    .item-5 {
      background-color: #c077af;
    }
    
    .item-6 {
      background-color: #f8d29d;
    }
    
    .item-7 {
      background-color: #b5a87f;
    }
    
    .item-8 {
      background-color: #d0e4a9;
    }
    
    .item-9 {
      background-color: #4dc7ec;
    }
    
    
    image.png

    grid-area属性还可用作grid-row-start、grid-column-start、grid-row-end、grid-column-end的合并简写形式,直接指定项目的位置。
    grid-area: <row-start> / <column-start> / <row-end> / <column-end>;

    justify-self 属性,align-self 属性,place-self 属性

    justify-self属性设置单元格内容的水平位置(左中右),跟justify-items属性的用法完全一致,但只作用于单个项目。

    align-self属性设置单元格内容的垂直位置(上中下),跟align-items属性的用法完全一致,也是只作用于单个项目。

    
    .item {
      justify-self: start | end | center | stretch;
      align-self: start | end | center | stretch;
    }
    

    start:对齐单元格的起始边缘。
    end:对齐单元格的结束边缘。
    center:单元格内部居中。
    stretch:拉伸,占满单元格的整个宽度(默认值)。

    place-self

    place-self属性是align-self属性和justify-self属性的合并简写形式
    place-self: <align-self> <justify-self>;

    参考:https://www.ruanyifeng.com/blog/2019/03/grid-layout-tutorial.html
    https://zhuanlan.zhihu.com/p/279972433
    https://www.jianshu.com/p/3762f214cd6f

    相关文章

      网友评论

          本文标题:grid布局

          本文链接:https://www.haomeiwen.com/subject/pxbmzltx.html