探索 flex:1 常见问题与解决方案

flex: 1 有什么作用?

先来看一个简单的例子。

html 复制代码
<div class="container">
  <div class="menu"></div>
  <div class="main"></div>
</div>
css 复制代码
.container {
  display: flex;
  width: 300px;
  height: 200px;
  border: 1px solid red;
}

.menu {
  flex: 0 0 100px;
  background-color: gray;
}

.main {
  flex: 1;
  background-color: darkseagreen;
}

效果截图:

以上效果可以点击这里查看:传送门

从代码效果可以看出 .main 元素的宽占据了父元素除 .menu 元素外的剩余宽度。

flex: 1 的实际渲染属性

打开 Chrome 控制台可以看到:

flex: 1 实际上代表三个属性:

  • flex-grow: 1
  • flex-shrink: 1
  • flex-basis: 0%

MDN 上关于 flex: 1 这种 One value 的情况。

个人认为里面 "flex-basis is then equal to 0" 这段话的表述有点模糊,因为 flex-basis 也可以表示 0px

所以在这里再次强调下 flex: 1 的 flex-basis 是 0%

使用 flex: 1 可能遇到的问题

修改一下例子中的代码:

html 复制代码
<div class="container">
  <div class="menu"></div>
  <div class="main">
    <div class="content">
      Lorem ipsum dolor sit amet consectetur, adipisicing elit. Sequi provident
      molestias inventore quisquam tempore hic labore esse reprehenderit
      quibusdam, amet nisi nesciunt ab deleniti dolorem tenetur, animi, unde
      alias sint.
    </div>
  </div>
</div>
css 复制代码
.container {
  display: flex;
  width: 300px;
  height: 200px;
  border: 1px solid red;
}

.menu {
  flex: 0 0 100px;
  background-color: gray;
}

.main {
  flex: 1;
  background-color: darkseagreen;
}

.content {
  text-overflow: ellipsis;
  overflow: hidden;
  white-space: nowrap;
}

效果截图:

以上效果可以点击这里查看:传送门

发现 .main 元素的宽度超过了父元素的宽度。

为什么会出现这样的问题?

规范中解释了这个情况:

默认情况下,弹性项目不会缩小到其 minimum content size(即最长单词或固定尺寸元素的长度)

那什么是 minimum content size?

从中我们了解到 minimum content size 在不同的情况对应不同的值:

container value
scroll content-based minimum size
not scroll 0

结合本文例子,我们分析一下 .main 元素为什么会出现这个情况。

.main 元素是一个不可滚动容器,所以最小宽度值 = content-based minimum size

而规范中 content-based minimum size = Math.min(specified size suggestion, content size suggestion)。

.main 元素没有指定的 width 属性也没有 aspect-ratio 属性,所以决定 .main 元素最小宽度的是 min-width

嗅探一下 .main 元素的 min-width:

看一下规范

从中我们了解到,automin-width/min-height 来讲是一个 automatic minimum size 。除非相关布局定义,一般情况下 auto 会被解析为 0。

automatic minimum size规范中的定义是:

从中我们了解到 automatic minimum size 依赖于内容。

如何解决这个问题?

由于 .main 元素的 min-width 的值是 auto 导致了这个问题。那么只要将 min-width 的值设置为 0 就可以解决这个问题。

diff 复制代码
.main {
  flex: 1;
  background-color: darkseagreen;
+  min-width: 0;
}

效果截图:

以上效果可以点击这里查看:传送门

其实对于这个问题还有第二种解决方案。那就是将 .main 元素的 overflow 属性设置为 auto,目的是让 .main 元素变成一个可滚动容器。

diff 复制代码
.main {
  flex: 1;
  background-color: darkseagreen;
+  overflow: auto;
}

效果截图:

以上效果可以点击这里查看:传送门

从规范里我们了解到,不仅是 min-width 会受到 automatic minimum size 的影响,min-height 也会受到 automatic minimum size 的影响。

所以这里给大家留个例子:

html 复制代码
<div class="container">
  <div class="menu"></div>
  <div class="main">
    <div class="content">
      Lorem, ipsum dolor sit amet consectetur adipisicing elit. Sit, consequatur
      voluptate nam aut fugiat vero quibusdam et, quas iure, ut eveniet nulla
      obcaecati tempora! Labore, perspiciatis quo! Assumenda, id temporibus.
      Temporibus iste dolorem minima quas maiores sapiente minus at beatae harum
      quae asperiores rerum, similique earum ad non officia eaque repellendus
      omnis quod rem dicta quasi explicabo reiciendis cupiditate! Sed!
      Blanditiis dolor sapiente ab eaque a officia enim illum facilis? Eum
      consequatur velit in consectetur molestias temporibus, voluptatem quo iure
      similique totam minima minus ipsum tempora, pariatur doloribus, dicta
      vitae!
    </div>
  </div>
</div>
css 复制代码
.container {
  display: flex;
  flex-direction: column;
  width: 300px;
  height: 300px;
  border: 1px solid red;
}

.menu {
  flex: 0 0 64px;
  background-color: gray;
}

.main {
  flex: 1;
  background-color: darkseagreen;
}

效果截图:

相信大家已经知道怎么解决这个问题了。

总结

flex: 1 实际上代表三个属性:flex-grow: 1, flex-shrink: 1, flex-basis: 0%。设置 flex: 1 的元素可以占据父元素的剩余宽度/高度。但 flex: 1 的元素的宽度/高度可能会超过父元素的宽度/高度。这是因为 flex: 1 的元素的 min-width/min-height 的值是 auto,而 auto 对 min-width/min-height 来讲是一个 automatic minimum size。除非相关布局定义,一般情况下 auto 会被解析为 0。所以只要将 flex: 1 的元素的 min-width/min-height 的值设置为 0 就可以解决这个问题。也可以将 flex: 1 的元素的 overflow 属性设置为 auto,目的是让 flex: 1 的元素变成一个可滚动容器。

相关推荐
Cool----代购系统API16 分钟前
css设置盒子动画,CSS3 transition动画 animation动画
前端·css·css3
哟哟耶耶26 分钟前
css-设置元素的溢出行为为可见overflow: visible;
前端·css
sunly_29 分钟前
CSS:跑马灯
前端·css
2301_8187320637 分钟前
用layui表单,前端页面的样式正常显示,但是表格内无数据显示(数据库连接和获取数据无问题)——已经解决
java·前端·javascript·前端框架·layui·intellij idea
yqcoder38 分钟前
npm link 作用
前端·npm·node.js
林涧泣43 分钟前
【Uniapp-Vue3】页面和路由API-navigateTo及页面栈getCurrentPages
前端·vue.js·uni-app
Komorebi゛1 小时前
【uniapp】获取上传视频的md5,适用于APP和H5
前端·javascript·uni-app
林涧泣1 小时前
【Uniapp-Vue3】动态设置页面导航条的样式
前端·javascript·uni-app
杰九1 小时前
【全栈】SprintBoot+vue3迷你商城(10)
开发语言·前端·javascript·vue.js·spring boot
Hopebearer_2 小时前
入门 Canvas:Web 绘图的强大工具
前端·javascript·es6·canva可画