Flex Box布局学习- 语法

上一篇,我学习并整理了使用flex时,需要注意的兼容性问题。那么今天就来学习一下有关flex语法的东西。

先看一下flex布局的总体图,然后我们再慢慢学习。

flex布局属性图.png

背景


我本来觉得,我写过许多的CSS样式了,对flex的应用也算娴熟了,起码经常写的布局,我都能不费劲儿的写出来了,flex布局的相关东西我也看了不少,当我看到这段代码时,我发现自己会的那点东西,简直不值得一提,不过没关系,那么让我从新学起,我就不信学不会,掌握不了。

.item-input textarea {
-webkit-box-flex: 1;
-webkit-flex: 1 0 220px;
-moz-box-flex: 1;
-moz-flex: 1 0 220px;
-ms-flex: 1 0 220px;
flex: 1 0 220px;
}


>上一篇《CSS之flex兼容性》中详细介绍了,*兼容性*的由来,那么在这我就不多做解释了。直接开始学习这些语法。

## 简介

2009年,W3C提出了一种新的方案----Flex布局,可以简便、完整、响应式地实现各种页面布局。目前,它已经得到了所有浏览器的支持,这意味着,现在就能很安全地使用这项功能。

## Flex Box 是什么?

CSS3 弹性盒子(Flex Box)弹性盒子是 CSS3 的一种新的布局模式。
CSS3 弹性盒( Flexible Box 或 flexbox),是一种当页面需要适应不同的屏幕大小以及设备类型时确保元素拥有恰当的行为的布局方式。
引入弹性盒布局模型的目的是提供一种更加有效的方式来对一个容器中的子元素进行排列、对齐和分配空白空间。

任何一个容器都可以指定为Flex布局。

**容器内可以包含一个或者多个弹性子元素。弹性子元素通常在弹性盒子内一行显示。默认情况下每个容器只有一行。**

e.g.:

.box{
display: flex;
}
// 行内元素也可以
.box{
display: inline-flex;
}

*<u>注意,设为Flex布局以后,子元素的float、clear和vertical-align属性将失效。</u>*

## 基本概念

采用Flex布局的元素,称为Flex容器(flex container),简称"容器"。它的所有子元素自动成为容器成员,称为Flex项目(flex item),简称"子元素"。

![basic](http://upload-images.jianshu.io/upload_images/2041009-d5f2aed4e59b5416.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)

**容器默认存在两根轴:**
<u>水平的主轴(main axis)和垂直的交叉轴(cross axis)。主轴的开始位置(与边框的交叉点)叫做main start,结束位置叫做main end;交叉轴的开始位置叫做cross start,结束位置叫做cross end。
项目默认沿主轴排列。单个项目占据的主轴空间叫做main size,占据的交叉轴空间叫做cross size。</u>


## 容器的属性

### 1. flex-direction属性
flex-direction属性决定主轴的方向(即子元素的排列方向)。

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

![flex-direction.png](http://upload-images.jianshu.io/upload_images/2041009-11dccceb5c388ce2.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)

它有4个值:

* row(默认值):主轴为水平方向,起点在左端。
* row-reverse:主轴为水平方向,起点在右端。
* column:主轴为垂直方向,起点在上沿。
* column-reverse:主轴为垂直方向,起点在下沿。
### 2. justify-content属性 
`justify-content`属性定义了项目在主轴上的对齐方式。

.box {
justify-content: flex-start | flex-end | center | space-between | space-around;
}

![flex-direction1.png](http://upload-images.jianshu.io/upload_images/2041009-790c37f67c9f4f84.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)

它可能取5个值,具体对齐方式与轴的方向有关。下面假设主轴为从左到右。

* flex-start(默认值):左对齐
* flex-end:右对齐
* center: 居中
* space-between:两端对齐,项目之间的间隔都相等。
* space-around:每个项目两侧的间隔相等。所以,项目之间的间隔比项目与边框的间隔大一倍。

### 3. align-items属性 
align-items属性定义项目在交叉轴上如何对齐。

.box {
align-items: flex-start | flex-end | center | baseline | stretch;
}

![align-items.png](http://upload-images.jianshu.io/upload_images/2041009-482855763e84059f.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)

它可能取5个值。具体的对齐方式与交叉轴的方向有关,下面假设交叉轴从上到下。

* flex-start:交叉轴的起点对齐。
* flex-end:交叉轴的终点对齐。
* center:交叉轴的中点对齐。
* baseline: 项目的第一行文字的基线对齐。
* stretch(默认值):如果项目未设置高度或设为auto,将占满整个容器的高度。

### 4. flex-wrap属性
默认情况下,项目都排在一条线(又称"轴线")上。flex-wrap属性定义,如果一条轴线排不下,如何换行,以及它的换行方式。

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

它可能取三个值:

* nowrap(默认):不换行。

![nowrap.png](http://upload-images.jianshu.io/upload_images/2041009-9c4905858390a65b.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)

* wrap:换行,第一行在最上方。

![wrap.jpg](http://upload-images.jianshu.io/upload_images/2041009-6e99f57b41fcaf92.jpg?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)

* wrap-reverse:换行,第一行在最下方。

![wrap-reverse.jpg](http://upload-images.jianshu.io/upload_images/2041009-ebc2e49e723efc61.jpg?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)
### 5.  align-content属性
 align-content属性用于修改flex-wrap属性的行为。类似于align-item,但它不是设置弹性元素的对齐,而是设置各个行的对齐。
如果弹性元素只有一行,该属性不起作用。

.box {
align-content: flex-start | flex-end | center | space-between | space-around | stretch;
}

![align-content.png](http://upload-images.jianshu.io/upload_images/2041009-a726e10048d78e67.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)

该属性可能取6个值:

* flex-start:与交叉轴的起点对齐。
* flex-end:与交叉轴的终点对齐。
* center:与交叉轴的中点对齐。
* space-between:与交叉轴两端对齐,轴线之间的间隔平均分布。
* space-around:每根轴线两侧的间隔都相等。所以,轴线之间的间隔比轴线与边框的间隔大一倍。
* stretch(默认值):轴线占满整个交叉轴。

## 弹性子元素的属性

以下6个属性设置在弹性子元素上。

* order
* flex-grow
* flex-shrink
* flex-basis
* flex
* align-self

### 1. order属性
order属性定义项目的排列顺序。数值越小,排列越靠前,默认为0。

.item {
order: <integer>;
}

![order.png](http://upload-images.jianshu.io/upload_images/2041009-08febe61b63ed0d1.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)

### 2. flex属性

#### 1. flex-grow属性

flex-grow属性赋予子元素在必要时伸展的能力,可指定一个不带单位的数值,作为父容器剩余空间的比例,它表示子元素在flex容器中可以分配多少可用的空间。

如果所有声明了flex-grow的子元素都指定flex-grow为1,那么父容器剩余的空间将会平均的分配到这些子元素上。如果其中一个flex-grow指定为2,那么容器将会试图为其分配一个空间,这个空间2倍于那些flex-grow为1的子元素。 

需要注意的是,我们说的剩余空间,是指除子元素内容以外的父容器可用空间,另外,父容器并不保证所有情况下都能均匀分配,但至少它会这样尝试。

flex-grow的值不能为负。 

每个弹性子元素没有设置宽度,是自由伸展的,那么子元素的宽度就是本身flex item1所占的宽度。如eg.1

![flow-grow.png](http://upload-images.jianshu.io/upload_images/2041009-666333d078397b54.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)

容器宽度:200

弹性子元素宽度:100

html:
<div class="flex-container">
  <div class="flex-item" style="flex-grow: 1;">flex item</div>
  <div class="flex-item" style="flex-grow: 2;">flex item</div>
  <div class="flex-item" style="flex-grow: 1;">flex item</div>
</div>
如果每个弹性子元素设置宽度100,那么子元素的宽度就是100,除去子元素后的部分按flex-grow比例进行分配。

![flex-grow2.png](http://upload-images.jianshu.io/upload_images/2041009-46cb48f6608f2c9d.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)

#### 2. flex-shrink属性

flex-shrink属性表示一个子元素在必要时是否收缩自己来适应当前的Flexbox,默认值是1。

注意:flex-shrink不能为负值。 

flex-shrink适合使用在固定尺寸的子元素上,默认情况下,固定大小的子元素并非始终保持设定的值,比如在父容器太小时,就会压缩子元素来适应,如果我们不想这些子元素被压缩,就可以使用flex-shrink,并设置其值为0。 

eg1:

```html
<div class="flex-container">
    <div class="flex-item" style="flex-shrink: 2;">flex item1</div>
  <div class="flex-item" style="flex-shrink: 1;">flex item2</div>
  <div class="flex-item" style="">flex item3</div>
</div>
flex-shrink.png

容器宽度:200

弹性子元素宽度:100

收缩比例为:2:1:1

所以:1003-200=100,超出100,按比例划分,就是item1被移除溢出量=100/(2+1+1) * 2=50;item2被移除溢出量=100/(2+1+1) * 1;item3被溢出量=100/(2+1+1)1。

eg2:

如果其中有一个元素设置了flex-shrink: 0;则其宽度不会收缩,其他部分按比例收缩即可。

3. flex-basis 属性

flex-basis属性告诉父容器,在剩余空间被分配之前先定义子元素的默认尺寸,可以指定为百分比或rem等长度单位或者auto关键字。

如果设置为0,那么父容器分配分配之前,对每个子元素的默认尺寸都视之为0,剩余空间也就是父容器的全部空间,其结果是,直接按照flex-grow值的比例分配子元素整体的大小;

如果设置为auto,那么父容器会将每个子元素中的内容作为子元素默认尺寸,然后再计算剩余空间,最后把剩余空间按照flex-grow值的比例平均分配到子元素除内容以外的空间,也就是”padding”。

[图片上传失败...(image-732e68-1524620541028)]

4. flex属性

flex属性是flex-grow, flex-shrink 和 flex-basis的简写,默认值为0 1 auto。后两个属性可选。

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

该属性有两个快捷值:auto (1 1 auto) 和 none (0 0 auto)。
建议优先使用这个属性,而不是单独写三个分离的属性,因为浏览器会推算相关值。

3. align-self属性

align-self属性允许单个项目有与其他项目不一样的对齐方式,可覆盖align-items属性。默认值为auto,表示继承父元素的align-items属性,如果没有父元素,则等同于stretch。

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

属性值各个值解释 - 自己的解释

一般大家都会看到好多个属性的取值中都有flex-startflex-endcenterspace-betweenspace-around这几个取值范围。那么下面就详细解释一下每个取值的含义。因为之前总是各个属性值组合起来用的,很是混乱。
刚开始看阮一峰老师的讲解时,有个地方自己理解错了, 所以特意自己截图来说明一下。可能有小伙伴和我一样也理解错了。

各属性图解

比如上图,只是看这个图,我就会认为,设置为center属性就会展示成上图中center的这个样子,其实这样的理解是有一点点偏差的。

以justify-content 属性为例进行说明。
justify-content:内容对齐(justify-content)属性应用在弹性容器上,把弹性项沿着弹性容器的主轴线(main axis)对齐。

1. flex-start

弹性项目向行头紧挨着填充。这个是默认值。第一个弹性项的main-start外边距边线被放置在该行的main-start边线,而后续弹性项依次平齐摆放。也就是左对齐
---w3c的解释

左对齐的意思就是说,从main-start边线,一个紧挨着一个向右排列,中间没有空隙的。例如,下图:


flex-start.png

2. flex-end

弹性项目向行尾紧挨着填充。第一个弹性项的main-end外边距边线被放置在该行的main-end边线,而后续弹性项依次平齐摆放。
---w3c的解释

如下图所示:

flex-end.png

3. center

弹性项目居中紧挨着填充。(如果剩余的自由空间是负的,则弹性项目将在两个方向上同时溢出)。
---w3c的解释

如下图所示:

center.png

]

4. space-between

弹性项目平均分布在该行上。如果剩余空间为负或者只有一个弹性项,则该值等同于flex-start。否则,第1个弹性项的外边距和行的main-start边线对齐,而最后1个弹性项的外边距和行的main-end边线对齐,然后剩余的弹性项分布在该行上,相邻项目的间隔相等。
---w3c的解释

如下图所示:

space-between.png

5. space-around

弹性项目平均分布在该行上,两边留有一半的间隔空间。如果剩余空间为负或者只有一个弹性项,则该值等同于center。否则,弹性项目沿该行分布,且彼此间隔相等(比如是20px),同时首尾两边和弹性容器之间留有一半的间隔(1/2*20px=10px)。

如下图:

space-around.png

所以最上面个那个综合的图片中,其实是为了看起来更加方便,是给每个子元素添加了外边距,外边距算作每个元素的样式。所以看起来是上面个综合图的样子。

参考

友情提示:初学者不易记全所有的属性和值的左右,鼠标放在属性名上即可显示相应属性的作用。

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

推荐阅读更多精彩内容