不定高元素动画实现方案(上)

前情

最近接了一个需求,需要实现一个列表,列表可展开收起,展开收起需要有一个动画效果,而列表个数不定且每项内容高度也不固定,所以是一个不定高的收起展开效果,我尝试了如下几中方式,最后选择了我觉得最佳的

通过max-height来实现

其实把高度设为auto,可以让元素高度撑开,但是auto是不会有动画效果的,此时我们可以把max-height设为一个较大的值,这样就实现动画效果了

代码如下:

html 复制代码
<!DOCTYPE html>
<html>
<head>
  <meta charset="utf-8">
  <meta name="viewport" content="width=device-width">
  <title>JS Bin</title>
</head>
<body>
  <div class="container">
    <div class="inner">
      <div class="header">header</div>
      <ul class="list">
        <li>111111111</li>
        <li>2222222222</li>
        <li>333333333</li>
      </ul>
    </div>
  </div>
</body>
</html>
html 复制代码
*{
  margin: 0;
  padding: 0;
}
.container{
  width: 100%;
  max-height: 48px;
  overflow: hidden;
  transition:max-height .4s;
}
.container:hover{
  max-height: 111px;
}
.header{
  width: 100%;
  height: 48px;
  background-color: #ccc;
  display: flex;
  align-items: center;
  justify-content: center;
}
.list{
  background-color: green;
}

演示地址:https://jsbin.com/cedifocuci/edit?html,css,output

注意:

max-height设的太高动画效果会太快,设的太小又无法适配所有高度,此方案只能说是能解决问题,但是不是最理想的方案

通过Grid布局来实现

通过网络布局,一开始设置grid-template-rows为0fr,这里有一些技巧,如果子元素有内容,0fr是不生效的,要给子元素设置min-height为0;如果初始要有基础高度,可以设置min-height为指定高度来实现基础高度,此处为48px

关键代码如下:

html 复制代码
<!DOCTYPE html>
<html>
<head>
  <meta charset="utf-8">
  <meta name="viewport" content="width=device-width">
  <title>JS Bin</title>
</head>
<body>
  <div class="container">
    <div class="inner">
      <div class="header">header</div>
      <ul class="list">
        <li>Grid111111111</li>
        <li>Grid2222222222</li>
        <li>Grid333333333</li>
        <li>Grid444444444</li>
      </ul>
    </div>
  </div>
</body>
</html>
css 复制代码
*{
  margin: 0;
  padding: 0;
}
.container{
  width: 100%;
  display: grid;
  grid-template-rows: 0fr;
  min-height: 48px;
  overflow: hidden;
  transition:all .4s;
}
.inner{
  min-height: 0;
}
.container:hover{
  grid-template-rows: 1fr;
}
.header{
  width: 100%;
  height: 48px;
  background-color: #ccc;
  display: flex;
  align-items: center;
  justify-content: center;
}
.list{
  background-color: green;
}

演示地址:https://jsbin.com/xasuviputo/edit?html,css,output

注意:

主流浏览器都支持,但是部分国内浏览器还是有很大兼容问题,如果对兼容性有很高要求的话此方案就不合适了,同时如果你有基础高度,那基础高度的那一段是没有动画效果的

通过clip-path来实现

clip-path 又叫CSS裁剪路径,clip-path 属性用于创建一个元素的可视区域,区域内的内容会显示,而区域外的内容则会被隐藏。这个属性可以用来对元素进行形状裁剪,实现各种视觉效果

此处使用clip-path:inset()来实现,可以传入四个数字inset(1 2 3 4),它们的规则如下图

关键代码如下:

html 复制代码
<!DOCTYPE html>
<html>
<head>
  <meta charset="utf-8">
  <meta name="viewport" content="width=device-width">
  <title>JS Bin</title>
</head>
<body>
  <div class="container">
    <div class="inner">
      <div class="header">header</div>
      <ul class="list">
        <li>clip-path111111111</li>
        <li>clip-path2222222222</li>
        <li>clip-path333333333</li>
        <li>clip-path444444444</li>
      </ul>
    </div>
  </div>
</body>
</html>
css 复制代码
*{
  margin: 0;
  padding: 0;
}
.container{
  width: 100%;
  clip-path: inset(0 0 calc(100% - 48px) 0);
  overflow: hidden;
  transition:all .4s;
}
.container:hover{
  clip-path: inset(0 0 0 0);
}
.header{
  width: 100%;
  height: 48px;
  background-color: #ccc;
  display: flex;
  align-items: center;
  justify-content: center;
}
.list{
  background-color: green;
}

演示地址:https://jsbin.com/zelopuyuxi/edit?html,css,output

注意:

主流浏览器都支持,但是部分国内浏览器还是有很大兼容问题,如果对兼容性有很高要求的话此方案就不合适了,相比grid此方案不会有基础高度无过渡效果的问题

小结

对于做技术的我们,每天都是提出问题解决问题的一个过程,过程中会尝试各种方案,因为解决问题的方案千千万,此篇文章记录了实现不定高内容过渡效果的三种实现方式,怕篇幅太长,此文暂时写这么多,下篇继续......

相关推荐
layman052817 小时前
webpack5 css-loader:从基础到原理
前端·css·webpack
半桔17 小时前
【前端小站】CSS 样式美学:从基础语法到界面精筑的实战宝典
前端·css·html
AI老李17 小时前
PostCSS完全指南:功能/配置/插件/SourceMap/AST/插件开发/自定义语法
前端·javascript·postcss
_OP_CHEN17 小时前
【前端开发之CSS】(一)初识 CSS:网页化妆术的终极指南,新手也能轻松拿捏页面美化!
前端·css·html·网页开发·样式表·界面美化
啊哈一半醒17 小时前
CSS 主流布局
前端·css·css布局·标准流 浮动 定位·flex grid 响应式布局
PHP武器库17 小时前
ULUI:不止于按钮和菜单,一个专注于“业务组件”的纯 CSS 框架
前端·css
电商API_1800790524717 小时前
第三方淘宝商品详情 API 全维度调用指南:从技术对接到生产落地
java·大数据·前端·数据库·人工智能·网络爬虫
晓晓莺歌17 小时前
vue3某一个路由切换,导致所有路由页面均变成空白页
前端·vue.js
Up九五小庞17 小时前
开源埋点分析平台 ClkLog 本地部署 + Web JS 埋点测试实战--九五小庞
前端·javascript·开源
qq_1777673718 小时前
React Native鸿蒙跨平台数据使用监控应用技术,通过setInterval每5秒更新一次数据使用情况和套餐使用情况,模拟了真实应用中的数据监控场景
开发语言·前端·javascript·react native·react.js·ecmascript·harmonyos