0

0

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

P粉602998670

P粉602998670

发布时间:2025-09-18 18:01:01

|

853人浏览过

|

来源于php中文网

原创

核心思路是利用 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
。它的优点是:

  1. 无需精确计算高度: 极大地简化了逻辑,减少了 JavaScript 的介入。
  2. 性能相对友好: 结合
    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
的妙用:

GradPen论文
GradPen论文

GradPen是一款AI论文智能助手,深度融合DeepSeek,为您的学术之路保驾护航,祝您写作顺利!

下载

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 的优雅,但能带来最精确的动画效果。

  • 基本思路:

    1. 列表初始状态
      height: 0; overflow: hidden;
    2. 当需要展开时:
      • height
        暂时设置为
        auto
        ,让浏览器计算出实际的
        scrollHeight
      • 立即将
        height
        设置回
        0
      • 强制浏览器重绘(例如通过读取
        offsetHeight
        ),确保
        height: 0
        状态被渲染。
      • height
        设置为刚刚获取到的
        scrollHeight
        值,并应用
        transition
    3. 动画结束后,将
      height
      设置为
      auto
      ,以适应未来内容变化。
    4. 折叠时,则反向操作:获取当前
      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

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

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
硬盘接口类型介绍
硬盘接口类型介绍

硬盘接口类型有IDE、SATA、SCSI、Fibre Channel、USB、eSATA、mSATA、PCIe等等。详细介绍:1、IDE接口是一种并行接口,主要用于连接硬盘和光驱等设备,它主要有两种类型:ATA和ATAPI,IDE接口已经逐渐被SATA接口;2、SATA接口是一种串行接口,相较于IDE接口,它具有更高的传输速度、更低的功耗和更小的体积;3、SCSI接口等等。

1079

2023.10.19

PHP接口编写教程
PHP接口编写教程

本专题整合了PHP接口编写教程,阅读专题下面的文章了解更多详细内容。

169

2025.10.17

php8.4实现接口限流的教程
php8.4实现接口限流的教程

PHP8.4本身不内置限流功能,需借助Redis(令牌桶)或Swoole(漏桶)实现;文件锁因I/O瓶颈、无跨机共享、秒级精度等缺陷不适用高并发场景。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

1407

2025.12.29

java接口相关教程
java接口相关教程

本专题整合了java接口相关内容,阅读专题下面的文章了解更多详细内容。

17

2026.01.19

DOM是什么意思
DOM是什么意思

dom的英文全称是documentobjectmodel,表示文件对象模型,是w3c组织推荐的处理可扩展置标语言的标准编程接口;dom是html文档的内存中对象表示,它提供了使用javascript与网页交互的方式。想了解更多的相关内容,可以阅读本专题下面的文章。

3303

2024.08.14

overflow什么意思
overflow什么意思

overflow是一个用于控制元素溢出内容的属性,当元素的内容超出其指定的尺寸时,overflow属性可以决定如何处理这些溢出的内容。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

1755

2024.08.15

css中的padding属性作用
css中的padding属性作用

在CSS中,padding属性用于设置元素的内边距。想了解更多padding的相关内容,可以阅读本专题下面的文章。

133

2023.12.07

css3transition
css3transition

css3transition属性用于指定如何从一个CSS样式过渡到另一个CSS样式,本专题为大家提供transition相关的文章、相关下载和相关课程,大家可以免费体验。

231

2023.06.27

Python 自然语言处理(NLP)基础与实战
Python 自然语言处理(NLP)基础与实战

本专题系统讲解 Python 在自然语言处理(NLP)领域的基础方法与实战应用,涵盖文本预处理(分词、去停用词)、词性标注、命名实体识别、关键词提取、情感分析,以及常用 NLP 库(NLTK、spaCy)的核心用法。通过真实文本案例,帮助学习者掌握 使用 Python 进行文本分析与语言数据处理的完整流程,适用于内容分析、舆情监测与智能文本应用场景。

10

2026.01.27

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
国外Web开发全栈课程全集
国外Web开发全栈课程全集

共12课时 | 1.0万人学习

前端小白零基础入门HTML5+CSS3
前端小白零基础入门HTML5+CSS3

共361课时 | 33.6万人学习

JS轻松实现打地鼠游戏
JS轻松实现打地鼠游戏

共6课时 | 0.7万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号