«
Flex 详解

时间:2022-4   


由于 flexbox 是一个完整的模块而不是单个属性,它涉及到很多东西,包括它的整个属性集。其中一些是要设置在容器上(父元素,称为“flex container”),而其他是要设置在子级(称为“flex items”)上。

如果“常规”布局基于块流方向和内联流方向,则弹性布局基于“弹性流方向”。请看一下规范中的这张图,解释了 flex 布局背后的主要思想。

解释 flexbox 术语的图表。 穿过flexbox主轴的尺寸称为主尺寸,另一个方向是横向尺寸。 这些尺寸有一个主开始、主结束、交叉开始和交叉结束。

项目将按照main axis(from main-startto main-end) 或横轴 (from cross-startto cross-end) 排列。

弹性盒属性

父级 (弹性容器)的属性

展示

这定义了一个弹性容器;内联或块取决于给定的值。它为其所有直接子级启用了弹性上下文。

.container {
  display: flex; /* or inline-flex */
}

请注意,CSS 列对 flex 容器没有影响。

弹性方向

这建立了主轴,从而定义了弹性项目放置在弹性容器中的方向。Flexbox 是(除了可选的包装)一个单向布局的概念。将弹性项目视为主要以水平行或垂直列布局。

.container {
  flex-direction: row | row-reverse | column | column-reverse;
}

弹性包装

默认情况下,弹性项目都将尝试适合一行。您可以更改它并允许使用此属性根据需要包装项目。

.container {
  flex-wrap: nowrap | wrap | wrap-reverse;
}

弹性流动

这是flex-directionandflex-wrap属性的简写,它们共同定义了 flex 容器的主轴和交叉轴。默认值为row nowrap。

.container {
  flex-flow: column wrap;
}

证明内容

这定义了沿主轴的对齐方式。当一行上的所有 flex 项目都不灵活,或者是灵活的但已达到最大大小时,它有助于分配额外的可用空间。当项目出行时,它还会对项目的对齐方式施加一些控制。

.container {
  justify-content: flex-start | flex-end | center | space-between | space-around | space-evenly | start | end | left | right ... + safe | unsafe;
}

请注意,浏览器对这些值的支持是有细微差别的。例如,space-between某些版本的 Edge 从未获得过支持,并且 Chrome 还没有开始/结束/左/右。MDN有详细的图表。最安全的值是flex-start、flex-end和center。

还有两个额外的关键字可以与这些值配对:safe和unsafe. 使用safe确保无论您如何进行这种类型的定位,您都不能推动一个元素,使其呈现在屏幕外(例如,离开顶部),这样内容也不能滚动(称为“数据丢失”) .

对齐项目

这定义了弹性项目如何沿当前行的交叉轴布局的默认行为。将其视为justify-content横轴(垂直于主轴)的版本。

.container {
  align-items: stretch | flex-start | flex-end | center | baseline | first baseline | last baseline | start | end | self-start | self-end + ... safe | unsafe;
}

和修饰符关键字可以与所有其他关键字一起使用(尽管注意safe浏览器支持),并帮助您防止对齐元素以使内容变得不可访问。unsafe

对齐内容

当横轴上有额外空间时,这将对齐 flex 容器的线,类似于在justify-content主轴内对齐单个项目的方式。

注意:该属性只对多行灵活容器生效,这里flex-wrap设置为wrap或wrap-reverse)。单行灵活容器(即 whereflex-wrap设置为其默认值no-wrap)不会反映align-content.

.container {
  align-content: flex-start | flex-end | center | space-between | space-around | space-evenly | stretch | start | end | baseline | first baseline | last baseline + ... safe | unsafe;
}

和修饰符关键字可以与所有其他关键字一起使用(尽管注意safe浏览器支持),并帮助您防止对齐元素以使内容变得不可访问。unsafe

间隙、行间隙、列间隙

该gap属性明确控制弹性项目之间的空间。它仅在不在外边缘的项目之间应用该间距。

.container {
  display: flex;
  ...
  gap: 10px;
  gap: 10px 20px; /* row-gap column gap */
  row-gap: 10px;
  column-gap: 20px;
}

该行为可以被认为是最小排水沟,就好像排水沟以某种方式更大(因为类似justify-content: space-between;),那么只有当该空间最终变得更小时,差距才会生效。

它不仅适用于 flexbox,也gap适用于网格和多列布局。

子项的属性 (弹性项目)

命令

默认情况下,弹性项目按源顺序排列。但是,该order属性控制它们在弹性容器中出现的顺序。

.item {
  order: 5; /* default is 0 */
}

具有相同的项目order恢复到源订单。

弹性成长

这定义了弹性项目在必要时增长的能力。它接受用作比例的无单位值。它规定了项目应该占用的弹性容器内的可用空间量。

如果所有项目都flex-grow设置为1,则容器中的剩余空间将平均分配给所有子项。如果其中一个孩子的值为2,则该孩子将占用其他孩子之一的两倍空间(或者至少会尝试)。

.item {
  flex-grow: 4; /* default 0 */
}

负数无效。

弹性收缩

这定义了弹性项目在必要时收缩的能力。

.item {
  flex-shrink: 3; /* default 1 */
}

负数无效。

弹性基础

这定义了在分配剩余空间之前元素的默认大小。它可以是长度(例如 20%、5rem 等)或关键字。关键字的auto意思是“看看我的宽度或高度属性”(这是由main-size关键字临时完成的,直到被弃用)。关键字的content意思是“根据项目的内容调整大小”——这个关键字还没有得到很好的支持,所以很难测试,也很难知道它的兄弟max-content、、min-content和fit-content做什么。

.item {
  flex-basis:  | auto; /* default auto */
}

如果设置为0,则不考虑内容周围的额外空间。如果设置为auto,则根据其flex-grow值分配额外空间。请参阅此图。

柔性

这是flex-grow, flex-shrink和flex-basis组合的简写。第二个和第三个参数 (flex-shrink和flex-basis) 是可选的。默认值为0 1 auto,但如果您使用单个数值设置它,例如flex: 5;,则会将 更改flex-basis为 0%,所以它就像设置flex-grow: 5; flex-shrink: 1; flex-basis: 0%;。

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

建议您使用此速记属性,而不是设置单个属性。速记智能地设置其他值。

对齐自我

align-items这允许为单个弹性项目覆盖默认对齐方式(或由 指定的对齐方式)。

请参阅align-items说明以了解可用值。

.item {
  align-self: auto | flex-start | flex-end | center | baseline | stretch;
}

请注意float,clear和vertical-align对弹性项目没有影响。

为 Flexbox 添加前缀

Flexbox 需要一些供应商前缀来支持尽可能多的浏览器。它不仅包括带有供应商前缀的属性,而且实际上还有完全不同的属性和值名称。这是因为 Flexbox 规范随着时间的推移发生了变化,创建了“旧”、“补间”和“新”版本。

也许处理这个问题的最好方法是编写新的(也是最终的)语法并通过Autoprefixer运行你的 CSS ,它可以很好地处理回退。

或者,这里有一个 Sass@mixin来帮助处理一些前缀,它也让你知道需要做什么样的事情:

@mixin flexbox() {
  display: -webkit-box;
  display: -moz-box;
  display: -ms-flexbox;
  display: -webkit-flex;
  display: flex;
}

@mixin flex($values) {
  -webkit-box-flex: $values;
  -moz-box-flex:  $values;
  -webkit-flex:  $values;
  -ms-flex:  $values;
  flex:  $values;
}

@mixin order($val) {
  -webkit-box-ordinal-group: $val;  
  -moz-box-ordinal-group: $val;     
  -ms-flex-order: $val;     
  -webkit-order: $val;  
  order: $val;
}

.wrapper {
  @include flexbox();
}

.item {
  @include flex(1 200px);
  @include order(2);
}css

例子

让我们从一个非常简单的例子开始,解决一个几乎每天都会遇到的问题:完美居中。如果你使用 flexbox,它再简单不过了。

.parent {
  display: flex;
  height: 300px; /* Or whatever */
}

.child {
  width: 100px;  /* Or whatever */
  height: 100px; /* Or whatever */
  margin: auto;  /* Magic! */
}

这依赖于auto弹性容器中设置的边距吸收额外空间的事实。因此,设置边距auto将使项目在两个轴上完美居中。

现在让我们使用更多的属性。考虑一个包含 6 个项目的列表,所有项目都具有固定尺寸,但可以自动调整大小。我们希望它们在水平轴上均匀分布,这样当我们调整浏览器大小时,一切都可以很好地缩放,并且没有媒体查询。

.flex-container {
  /* We first create a flex layout context */
  display: flex;

  /* Then we define the flow direction 
     and if we allow the items to wrap 
   * Remember this is the same as:
   * flex-direction: row;
   * flex-wrap: wrap;
   */
  flex-flow: row wrap;

  /* Then we define how is distributed the remaining space */
  justify-content: space-around;
}

完毕。其他一切都只是一些造型问题。

让我们试试别的。想象一下,我们网站顶部有一个右对齐的导航元素,但我们希望它在中型屏幕上居中,在小型设备上为单列。很容易。

/* Large */
.navigation {
  display: flex;
  flex-flow: row wrap;
  /* This aligns items to the end line on main-axis */
  justify-content: flex-end;
}

/* Medium screens */
@media all and (max-width: 800px) {
  .navigation {
    /* When on medium sized screens, we center it by evenly distributing empty space around items */
    justify-content: space-around;
  }
}

/* Small screens */
@media all and (max-width: 500px) {
  .navigation {
    /* On small screens, we are no longer using row direction but column */
    flex-direction: column;
  }
}

让我们通过玩弹性项目的灵活性来尝试更好的东西!带有全宽页眉和页脚的移动优先 3 列布局怎么样?并且独立于源顺序。

.wrapper {
  display: flex;
  flex-flow: row wrap;
}

/* We tell all items to be 100% width, via flex-basis */
.wrapper > * {
  flex: 1 100%;
}

/* We rely on source order for mobile-first approach
 * in this case:
 * 1. header
 * 2. article
 * 3. aside 1
 * 4. aside 2
 * 5. footer
 */

/* Medium screens */
@media all and (min-width: 600px) {
  /* We tell both sidebars to share a row */
  .aside { flex: 1 auto; }
}

/* Large screens */
@media all and (min-width: 800px) {
  /* We invert order of first sidebar and main
   * And tell the main element to take twice as much width as the other two sidebars 
   */
  .main { flex: 2 0px; }
  .aside-1 { order: 1; }
  .main    { order: 2; }
  .aside-2 { order: 3; }
  .footer  { order: 4; }
}

浏览器支持

此浏览器支持数据来自Caniuse,其中包含更多详细信息。数字表示浏览器支持该版本及更高版本的功能。

错误

Flexbox 当然也不是没有缺陷。我见过的最好的集合是 Philip Walton 和 Greg Whitworth 的Flexbugs。这是一个跟踪所有这些的开源地方,所以我认为最好只链接到它。