如何用css animation优化列表折叠展开动画

核心思路是利用 max-height 结合 opacity 和 transform 实现流畅折叠展开动画,避免直接动画 height 引发重排。通过设置足够大的 max-height 值、配合 overflow: hidden 与关键帧动画,在无需精确计算高度的前提下实现性能友好的视觉效果。使用 opacity 实现淡入淡出,transform 应用 scaleY 或 translateY 增强动态感,由 GPU 加速提升流畅度。为优化动态内容场景,可结合 will-change 提示、合理缓动函数与动画时长,并在必要时通过 JavaScript 获取 scrollHeight 实现精准 height 过渡,或仅对列表项单独执行 opacity 和 transform 动画以规避容器尺寸变化带来的性能问题。

如何用css animation优化列表折叠展开动画

CSS

animation

在优化列表折叠展开动画时,核心思路是巧妙地利用

max-height

结合

opacity

transform

属性,通过关键帧(

@keyframes

)来精细控制动画的每个阶段,从而实现比单纯

transition

更富有表现力和性能更佳的效果。这样做不仅能让视觉上更流畅自然,还能在一定程度上规避直接动画

height

属性可能带来的性能问题。

解决方案

要用 CSS

animation

优化列表折叠展开,我们通常会避免直接动画

height

,因为它很容易触发浏览器重排(reflow),导致动画卡顿。一个更稳妥的策略是动画

max-height

,并辅以

opacity

transform

来增强视觉效果和性能。

具体来说,当列表需要折叠时,我们将

max-height

从一个足够大的值(足以包含所有内容)动画到

0

。展开时则反向操作。同时,配合

opacity

1

0

(折叠)或

0

1

(展开),让内容有淡入淡出的效果。更进一步,可以加入

transform: scaleY()

translateY()

来模拟内容的收缩或滑动,因为

transform

属性的动画通常由 GPU 处理,性能更好。

以下是一个基础的 CSS

animation

示例:

立即学习“前端免费学习笔记(深入)”;

/* 列表容器 */.list-container {  overflow: hidden;  max-height: 1000px; /* 足够大的值,确保能容纳所有内容 */  opacity: 1;  transform-origin: top; /* 确保缩放从顶部开始 */  animation-fill-mode: forwards; /* 动画结束后保持最终状态 */  will-change: max-height, opacity, transform; /* 性能优化提示 */}/* 展开动画 */.list-container.is-expanded {  animation: expandList 0.4s ease-out forwards;}@keyframes expandList {  0% {    max-height: 0;    opacity: 0;    transform: scaleY(0.8) translateY(-10px);  }  100% {    max-height: 1000px; /* 或根据实际内容估算一个更大的值 */    opacity: 1;    transform: scaleY(1) translateY(0);  }}/* 折叠动画 */.list-container.is-collapsed {  animation: collapseList 0.4s ease-in forwards;}@keyframes collapseList {  0% {    max-height: 1000px;    opacity: 1;    transform: scaleY(1) translateY(0);  }  100% {    max-height: 0;    opacity: 0;    transform: scaleY(0.8) translateY(-10px);  }}/* 确保初始状态 */.list-container.hidden-by-default {  max-height: 0;  opacity: 0;  transform: scaleY(0.8) translateY(-10px);}

通过 JavaScript 切换

.is-expanded

.is-collapsed

类名来触发动画。这种方式提供了一个相对平滑且性能友好的解决方案。

在列表折叠动画中,

max-height

height

的选择有何不同,以及如何避免动画卡顿?

在实现列表折叠展开动画时,

max-height

height

的选择确实是个值得深思的问题,它直接关系到动画的流畅度和性能。我个人经验是,如果不是对内容高度有绝对的掌控,或者内容高度是动态变化的,那么

max-height

几乎是首选。

height

属性的动画,理论上能实现最精确的尺寸变化。如果你能准确知道列表展开后的最终高度,并且这个高度是固定的,那么直接动画

height

0

到那个固定值,效果会非常完美。但问题在于,实际项目中列表内容往往是动态的,比如从后端接口加载数据,或者用户自定义内容,导致最终高度不确定。每次动画前都要用 JavaScript 计算

scrollHeight

,然后把这个值赋给

height

,这本身就增加了复杂性,而且频繁的

scrollHeight

读取和

height

赋值可能会导致浏览器强制重排,反而带来性能负担,甚至在动画开始前出现一帧的跳动。

相比之下,

max-height

的策略就显得“粗暴”而有效。我们给

max-height

设置一个足够大的值(比如

1000px

甚至

9999px

),只要这个值大于任何可能出现的实际内容高度,列表就能完全展开。动画时,从

0

动画到这个大值,或者从大值动画到

0

。它的优点是:

无需精确计算高度: 极大地简化了逻辑,减少了 JavaScript 的介入。性能相对友好: 结合

overflow: hidden

,浏览器在动画

max-height

时,通常不需要像动画

height

那样频繁地进行布局计算,尤其是在内容实际高度远小于

max-height

的情况下,动画过程中的布局变化相对较少。

然而,

max-height

也有其缺点。如果实际内容高度远小于你设置的

max-height

,动画会显得有些“空洞”,比如内容只有

50px

高,但动画

max-height

0

1000px

却要持续

0.4s

,用户会觉得动画时间过长,内容出现后还有一段“空白”时间。

为了避免动画卡顿,无论选择

height

还是

max-height

,都需要注意以下几点:

利用

overflow: hidden

这是

max-height

动画的关键。它能剪裁超出容器的内容,防止在动画过程中出现滚动条或内容溢出。动画非布局属性: 优先动画

opacity

transform

(如

scaleY

translateY

)。这些属性的变化不会触发布局或绘制,而是直接在合成层(compositor layer)上进行,由 GPU 加速,性能极佳。即使

max-height

动画本身会引起一些布局变化,通过

opacity

transform

的辅助,也能在视觉上掩盖一部分卡顿感。使用

will-change

这是一个性能优化提示。在动画元素上设置

will-change: max-height, opacity, transform;

可以告诉浏览器,这些属性即将发生变化,让浏览器提前进行优化准备。但要谨慎使用,过度使用反而可能消耗更多内存。合理的动画时长和缓动函数: 通常

200ms

400ms

是一个比较舒适的动画时长。选择合适的

ease-in

ease-out

ease-in-out

缓动函数,能让动画感觉更自然,避免突兀。避免在动画进行中修改 DOM: 在动画过程中,尽量避免对列表项进行添加、删除或修改操作,这会强制浏览器重新计算布局,导致动画中断或卡顿。

总的来说,对于大多数动态列表折叠场景,

max-height

配合

overflow: hidden

opacity

transform

是一个更实用且性能表现良好的方案。如果对动画效果有极致要求,且内容高度固定或可预测,才考虑用 JavaScript 精确控制

height

如何结合

opacity

transform

属性,创造更平滑自然的折叠展开效果?

仅仅动画

max-height

可能会让折叠展开显得有些生硬,或者出现前面提到的“空洞感”。结合

opacity

transform

属性,就能为动画注入更多生命力,使其看起来更平滑、更自然,甚至带有一些微小的“物理”反馈。这不仅仅是美观问题,也是一种用户体验的优化,因为更自然的动画能减少用户的认知负荷。

我个人在做这类动画时,几乎都会同时考虑这三个属性的组合。

1.

opacity

的运用:

opacity

的作用是让内容在折叠时逐渐淡出,展开时逐渐淡入。这能很好地掩盖

max-height

动画过程中可能出现的生硬感,尤其是当

max-height

的值远大于实际内容高度时,

opacity

的渐变能让用户觉得内容是“消失”或“出现”的,而不是简单地被“切掉”或“显示”。

展开动画 (

expandList

):

opacity

0

渐变到

1

。在

max-height

开始增长的同时,内容也开始变得可见。折叠动画 (

collapseList

):

opacity

1

渐变到

0

。在

max-height

开始减小的同时,内容也逐渐变得透明直至消失。

2.

transform

的妙用:

绘蛙AI修图 绘蛙AI修图

绘蛙平台AI修图工具,支持手脚修复、商品重绘、AI扩图、AI换色

绘蛙AI修图 285 查看详情 绘蛙AI修图

transform

属性是 CSS 动画的利器,因为它能直接作用于元素的合成层,不触发布局和绘制,性能极高。在折叠展开动画中,

scaleY

translateY

是常用的两个

transform

函数。

transform: scaleY()

(垂直缩放):

可以模拟内容从顶部或底部被“挤压”或“拉伸”的感觉。展开动画:

scaleY

可以从

0.8

(略微压缩)或

0

(完全扁平)渐变到

1

。配合

transform-origin: top;

(或

bottom

),能让缩放效果从指定方向开始。例如,从

scaleY(0.8)

渐变到

scaleY(1)

,能给内容一种“弹入”的轻微弹性感。折叠动画:

scaleY

1

渐变到

0.8

0

注意: 直接对文本或图片进行

scaleY

可能会导致内容变形,所以通常我们会对包裹列表项的容器进行

scaleY

transform: translateY()

(垂直位移):

可以模拟内容在折叠时向上“滑出”视线,展开时向下“滑入”视线。展开动画:

translateY

可以从

10px

(略低于最终位置)或

-10px

(略高于最终位置)渐变到

0

。例如,从

translateY(-10px)

渐变到

translateY(0)

,能让内容有一种从上方“落入”的感觉。折叠动画:

translateY

0

渐变到

10px

-10px

组合示例:

在解决方案部分给出的

keyframes

示例中,我已经结合了这三个属性:

@keyframes expandList {  0% {    max-height: 0;    opacity: 0;    transform: scaleY(0.8) translateY(-10px); /* 略微压缩并向上偏移 */  }  100% {    max-height: 1000px;    opacity: 1;    transform: scaleY(1) translateY(0); /* 完全展开并回到原位 */  }}@keyframes collapseList {  0% {    max-height: 1000px;    opacity: 1;    transform: scaleY(1) translateY(0);  }  100% {    max-height: 0;    opacity: 0;    transform: scaleY(0.8) translateY(-10px); /* 略微压缩并向上偏移 */  }}

这里,

max-height

负责主要的高度变化,

opacity

负责淡入淡出,而

transform: scaleY(0.8) translateY(-10px)

则为展开动画的起始和折叠动画的结束提供了一个“微小”的动态效果。内容在出现时,会先略微压缩并从上方一点点“滑入”;在消失时,则会略微压缩并向上方“滑出”。这种细微的调整能让整个动画过程看起来更富有弹性,更符合我们对物理世界的直觉,从而带来更平滑、更自然的视觉体验。

在实际项目中,如何处理动态内容或列表项数量不确定的情况下的动画优化?

实际项目中,动态内容和不确定数量的列表项是常态,这确实给 CSS 动画带来了一些挑战。纯 CSS

max-height

方案虽然简洁,但在这种情况下可能会遇到一些“不完美”的地方。在我看来,处理这类问题,往往需要结合 CSS 和 JavaScript,或者更巧妙地利用 CSS 本身的特性。

1.

max-height

的“估值”与“妥协”:

前面提到,

max-height

的一个问题是如果设置过大,动画时间会感觉“空洞”。对于动态内容,我们无法预知确切高度,所以设置一个足够大的

max-height

值是必须的。

估算最大值: 如果能大致预估列表的最大可能高度(比如最多

N

个列表项,每个项平均高度

X

,那么

max-height

可以设为

N * X + padding

),这能让动画在大多数情况下看起来比较合理。但总会有超出预期的情况。接受“空洞”: 有时,为了纯 CSS 方案的简洁和性能,我们需要接受

max-height

动画可能带来的轻微“空洞”感。通过

opacity

transform

的配合,可以很好地掩盖这种不足,让用户感知到的流畅度更高。

2. JavaScript 辅助获取精确高度(“混合式”方案):

这是解决动态内容高度不确定性最有效的方法,虽然牺牲了一部分纯 CSS 的优雅,但能带来最精确的动画效果。

基本思路:

列表初始状态

height: 0; overflow: hidden;

当需要展开时:将

height

暂时设置为

auto

,让浏览器计算出实际的

scrollHeight

。立即将

height

设置回

0

。强制浏览器重绘(例如通过读取

offsetHeight

),确保

height: 0

状态被渲染。将

height

设置为刚刚获取到的

scrollHeight

值,并应用

transition

。动画结束后,将

height

设置为

auto

,以适应未来内容变化。折叠时,则反向操作:获取当前

scrollHeight

,设置

height

为该值,然后

transition

height: 0

代码示例(简化版):

function toggleList(element) {  if (element.style.height && element.style.height !== '0px') {    // 正在展开或已展开,准备折叠    element.style.height = element.scrollHeight + 'px'; // 确保从当前高度开始折叠    requestAnimationFrame(() => {      element.style.height = '0';    });  } else {    // 正在折叠或已折叠,准备展开    element.style.height = 'auto'; // 临时设置为auto获取实际高度    const scrollHeight = element.scrollHeight;    element.style.height = '0'; // 立即设回0    requestAnimationFrame(() => { // 确保浏览器已经渲染了height:0      element.style.height = scrollHeight + 'px';    });  }  // 动画结束后移除height属性,让其自适应  element.addEventListener('transitionend', () => {    if (element.style.height !== '0px') {      element.style.height = 'auto';    }  }, { once: true });}// CSS// .list-container {//   overflow: hidden;//   transition: height 0.4s ease-in-out;// }// .list-container[style="height: 0px;"] {//   height: 0;// }

这种方案虽然需要 JavaScript 介入,但它能提供最精确、最流畅的动画,尤其适合内容高度差异大且不可预测的场景。

3. 仅动画列表项,而非容器高度:

对于列表项数量不确定,但每个列表项高度相对固定的情况,可以考虑不动画列表容器的

height

max-height

,而是动画列表项自身的

opacity

transform

思路: 容器的高度会瞬间变化,但内部的列表项会优雅地淡入淡出或滑动进入/离开。实现:列表

以上就是如何用css animation优化列表折叠展开动画的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/1071030.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
迅雷网盘下载失败怎么办_迅雷网盘下载失败原因与解决措施
上一篇 2025年12月2日 07:14:18
谷歌浏览器打不开扩展商店怎么办 Google Chrome扩展无法访问解决方案
下一篇 2025年12月2日 07:14:22

相关推荐

  • 修复Django电商项目中AJAX过滤产品列表图片不显示问题

    在Django电商项目中,当使用AJAX动态加载过滤后的产品列表时,常遇到图片无法正常显示的问题。这通常是由于前端模板中图片加载方式(如data-setbg属性结合JavaScript库)与AJAX动态内容更新机制不兼容所致。解决方案是直接在AJAX返回的HTML中使用标准的标签来渲染图片,确保浏览…

    2026年5月10日
    000
  • Golang JSON序列化:控制敏感字段暴露的最佳实践

    本教程探讨golang中如何高效控制结构体字段在json序列化时的可见性。当需要将包含敏感信息的结构体数组转换为json响应时,通过利用`encoding/json`包提供的结构体标签,特别是`json:”-“`,可以轻松实现对特定字段的忽略,从而避免敏感数据泄露,确保api…

    2026年5月10日
    000
  • 比特币新手教程 比特币交易平台有哪些

    比特币是一种去中心化的数字货币,基于区块链技术实现点对点交易,具有匿名性、有限发行和不可篡改等特点;新手可通过交易所购买,P2P交易获得比特币,常用平台包括Binance、OKX和Huobi;交易流程包括注册账户、实名认证、绑定支付方式、充值法币并下单购买,可选择市价单或限价单;比特币存储方式有交易…

    2026年5月10日
    000
  • c++中的SFINAE技术是什么_c++模板编程中的SFINAE原理与应用

    SFINAE 是“替换失败不是错误”的原则,指模板实例化时若参数替换导致错误,只要存在其他合法候选,编译器不报错而是继续重载决议。它用于条件启用模板、类型检测等场景,如通过 decltype 或 enable_if 控制函数重载,实现类型特征判断。尽管 C++20 引入 Concepts 简化了部分…

    2026年5月10日
    000
  • Go语言mgo查询构建:深入理解bson.M与日期范围查询的正确实践

    本文旨在解决go语言mgo库中构建复杂查询时,特别是涉及嵌套`bson.m`和日期范围筛选的常见错误。我们将深入剖析`bson.m`的类型特性,解释为何直接索引`interface{}`会导致“invalid operation”错误,并提供一种推荐的、结构清晰的代码重构方案,以确保查询条件能够正确…

    2026年5月10日
    100
  • css max-height属性怎么用

    max-height 属性设置元素的最大高度。 说明 该属性值会对元素的高度设置一个最高限制。因此,元素可以比指定值矮,但不能比其高。不允许指定负值。 注意:max-height 属性不包括外边距、边框和内边距。 立即学习“前端免费学习笔记(深入)”; 值描述none 默认。定义对元素被允许的最大高…

    2026年5月10日
    100
  • vscode上怎么运行html_vscode上运行html步骤【指南】

    首先保存文件为.html格式,再通过浏览器或Live Server插件打开预览;推荐安装Live Server实现本地服务器运行与实时刷新,提升开发体验。 在 VS Code 上运行 HTML 文件并不需要复杂的配置,只需几个简单步骤即可预览页面效果。VS Code 本身是一个代码编辑器,不直接运行…

    2026年5月10日
    100
  • 修复点击时按钮抖动:CSS垂直对齐实践

    本文探讨了在Web开发中,交互式按钮(如播放/暂停按钮)在点击时发生意外垂直位移的问题。通过分析CSS样式变化对元素布局的影响,我们发现这是由于按钮不同状态下的边框样式和内边距改变,以及默认的垂直对齐行为共同作用所致。核心解决方案是利用CSS的vertical-align属性,将其设置为middle…

    2026年5月10日
    000
  • Golang goroutine与channel调试技巧

    使用go run -race检测数据竞争,结合runtime.NumGoroutine监控协程数量,通过pprof分析阻塞调用栈,利用select超时避免永久阻塞,有效排查goroutine泄漏、死锁和数据竞争问题。 Go语言的goroutine和channel是并发编程的核心,但它们也带来了调试上…

    2026年5月10日
    000
  • 《魔兽世界》将于6月11日开启国服回归技术测试

    《魔兽世界》将于6月11日开启国服回归技术测试《魔兽世界》将于6月11日开启国服回归技术测试《魔兽世界》将于6月11日开启国服回归技术测试《魔兽世界》将于6月11日开启国服回归技术测试

    《%ign%ignore_a_1%re_a_1%》官方宣布,将于6月11日开启国服回归技术测试,时间为7天,并称可以在6月内正式开服,玩家们可以访问官网下载战网客户端并预下载“巫妖王之怒”客户端,技术测试详情见下图。 WordAi WordAI是一个AI驱动的内容重写平台 53 查看详情 以上就是《…

    2026年5月10日 用户投稿
    200
  • 使用 Jupyter Notebook 进行探索性数据分析

    Jupyter Notebook通过单元格实现代码与Markdown结合,支持数据导入(pandas)、清洗(fillna)、探索(matplotlib/seaborn可视化)、统计分析(describe/corr)和特征工程,便于记录与分享分析过程。 Jupyter Notebook 是进行探索性…

    2026年5月10日
    000
  • 如何在HTML中插入表单元素_HTML表单控件与输入类型使用指南

    HTML表单通过标签构建,包含action和method属性定义数据提交目标与方式,常用input类型如text、password、email等适配不同输入需求,配合label、required、placeholder提升可用性,结合textarea、select、button等控件实现完整交互,是…

    2026年5月10日
    000
  • 前端缓存策略与JavaScript存储管理

    根据数据特性选择合适的存储方式并制定清晰的读写与清理逻辑,能显著提升前端性能;合理运用Cookie、localStorage、sessionStorage、IndexedDB及Cache API,结合缓存策略与定期清理机制,可在保证用户体验的同时避免安全与性能隐患。 前端缓存和JavaScript存…

    2026年5月10日
    100
  • HTML5网页如何实现手势操作 HTML5网页移动端交互的处理技巧

    首先利用原生touch事件实现滑动判断,再通过preventDefault解决滚动冲突,接着引入Hammer.js处理复杂手势,最后通过优化点击区域、避免事件冲突和增加视觉反馈提升体验。 在移动端浏览器中,HTML5网页可以通过触摸事件实现手势操作,提升用户体验。虽然原生JavaScript提供了基…

    2026年5月10日
    000
  • 创建指定大小并填充特定数据的Golang文件教程

    本文将介绍如何使用Golang创建一个指定大小的文件,并用特定数据填充它。我们将使用 `os` 包提供的函数来创建和截断文件,从而实现快速生成大文件的目的。示例代码展示了如何创建一个10MB的文件,并将其填充为全零数据。掌握这些方法,可以方便地在例如日志系统或磁盘队列等场景中,预先创建测试文件或初始…

    2026年5月10日
    000
  • Python命令怎样使用profile分析脚本性能 Python命令性能分析的基础教程

    使用Python的cProfile模块分析脚本性能最直接的方式是通过命令行执行python -m cProfile your_script.py,它会输出每个函数的调用次数、总耗时、累积耗时等关键指标,帮助定位性能瓶颈;为进一步分析,可将结果保存为文件python -m cProfile -o ou…

    2026年5月10日
    000
  • 如何插入查询结果数据_SQL插入Select查询结果方法

    如何插入查询结果数据_SQL插入Select查询结果方法如何插入查询结果数据_SQL插入Select查询结果方法如何插入查询结果数据_SQL插入Select查询结果方法如何插入查询结果数据_SQL插入Select查询结果方法

    使用INSERT INTO…SELECT语句可高效插入数据,通过NOT EXISTS、LEFT JOIN、MERGE语句或唯一约束避免重复;表结构不一致时可通过别名、类型转换、默认值或计算字段处理;结合存储过程可提升可维护性,支持参数化与动态SQL。 将查询结果数据插入到另一个表中,可以…

    2026年5月10日 用户投稿
    000
  • 使用 WebCodecs VideoDecoder 实现精确逐帧回退

    本文档旨在解决在使用 WebCodecs VideoDecoder 进行视频解码时,实现精确逐帧回退的问题。通过比较帧的时间戳与目标帧的时间戳,可以避免渲染中间帧,从而提高用户体验。本文将提供详细的解决方案和示例代码,帮助开发者实现精确的视频帧控制。 在使用 WebCodecs VideoDecod…

    2026年5月10日
    000
  • PHP动态生成表单输入与POST数据获取实践指南

    本教程详细阐述了如何在php中根据动态数据源(如数据库值)生成多个表单输入框,并演示了如何通过post方法准确无误地获取这些动态生成的输入值。文章强调了正确的输入框命名策略,避免了常见的命名误区,并提供了完整的代码示例,确保开发者能够高效处理动态表单数据。 动态生成表单输入 在Web开发中,我们经常…

    2026年5月10日
    000
  • Discord.py 交互按钮超时与持久化解决方案

    本教程旨在解决Discord.py中交互按钮在一段时间后出现“This Interaction Failed”错误的问题。我们将深入探讨视图(View)的超时机制,并提供通过正确设置timeout参数以及利用bot.add_view()方法实现按钮持久化的具体方案,确保您的机器人交互功能稳定可靠,即…

    2026年5月10日
    000

发表回复

登录后才能评论
关注微信