百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 技术文章 > 正文

一行代码实现display"过渡动画"原理

zhezhongyun 2025-08-03 09:28 3 浏览

作者:Peter 谭老师

转发链接:
https://mp.weixin.qq.com/s/XhwPOv62gypzq5MhhP-5vg

写本文的起因

  • 上篇文章,提到如何让display出现过渡动画,却没有仔细介绍原理。
  • 为了更好的让想学习的人深入理解于是加班加点写下了这篇“短文”,我想以后还是以短文为主,不然大家看起来太累

正式开始

  • 初始化界面

<!DOCTYPE html>
<html lang="en">

<head>
    <meta charset="UTF-8">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <title>Document</title>
    <style>
          #app {
            width: 200px;
            height: 200px;
            background-color: red;
            display: none;
            transition: all 1s;
        }
    </style>
</head>

<body>
    <div id="app">
    </div>
    <button id="test">测试</button>
</body>

</html>
  • 此时我将app的display初始化为none,并且写入脚本文件
<style>
        #app {
            width: 200px;
            height: 200px;
            background-color: red;
            display: none;
        }
    </style>
    。。。
<script>
     test.onclick = function () {
        const app = document.querySelector('#app')
        console.log(app, 'app')
        app.style.transform = "translateX(200px)"
        app.style.display = "block"
    }
</script>
  • 初始化界面变成了这样:
  • 此时,我点击测试按钮
  • 并没有出现动画,非常生硬的出来了,有一些场景我又要性能,比如初始化不渲染,但是当它出现又要有动画的时候,就有可能使用这行代码

 test.onclick = function () {
        const app = document.querySelector('#app')
        console.log(app, 'app')
        app.style.display = "block"
        const height = app.offsetHeight
        app.style.transform = "translateX(200px)"
    }
  • 当我加入const height = app.offsetHeight这行代码的时候,再点击测试按钮,display切换就顺带出来了“动画”,有了过渡效果
  • 为什么会出现动画了呢?因为我读取dom的这些特殊属性时,浏览器就会强制清空渲染队列一次,让我拿到最新的值。也就是说读取的时候,其实已经是display为"block"了,因此。我们出现了过渡动画

效果如下所示:


出现“过渡动画”是什么情况?

  • 其实display是不能出现动画的,所以标题+了引号
  • 怎么才能有过渡?
    • 有数字的变化,例如透明度,从0-1.
    • 初始化有渲染展示的
    • transition里面包含的属性
    • ...等 大家可以补充

为什么加了一行代码后,就能出现动画了?

  • 大家在写现代前端框架,遇到最多的问题就是渲染的时期不确定的问题。
    • 例如vue里面的nextTick实现,有一个优雅降级的实现。它在mounted生命周期函数里面去获取dom节点时候,经常获取不到或者获取不到完整渲染的dom节点。(我很久没有使用vue了,有问题可以补充),为什么?
    • 像现在数据驱动的框架,只要数据改变了,对应逻辑绑定了数据的dom节点按道理应该更新,可是更新时机是我们无法确定的,因为这中间有中间层,比如存在diff算法计算过程,可能存在队列,因为当你频繁修改数据的时候,框架本身要做优化,合并一段时间的数据更新再去真正更新dom,等这些事情都做完了,才能去更新dom节点,然后我们才能看到最新数据对应的节点
    • 当我们真的要去更新dom节点的时候,也存在一个队列。这个就是浏览器的渲染队列
  • 如果你无法理解我上面说的,可以看我之前手写React系列文章中的setState异步队列实现

浏览器的渲染队列

  • 什么时候最能体现这个队列的作用?
    • 频繁直接操作dom时候,例如for循环里面频繁操作dom,这个时候浏览器就会优化我们的操作,合并一部分操作一次性执行
    • 渲染队列跟display的关联
<script>
    test.onclick = function () {
        const app = document.querySelector('#app')
        console.log(app, 'app')
        app.style.display = "block"
        const height = app.offsetHeight
        app.style.transform = "translateX(200px)"
    }
</script>
  • 当我们执行了app.style.display = "block"这行代码时候,dom节点此时并没有更新,js解析引擎是聪明的,它发现你后面马上有代码要修改dom节点,会先存入队列中集中一次性操作
  • 当我们执行了app.offsetHeight这行代码时候,发现我们需要读取dom节点的属性,浏览器害怕现在队列中没有执行的操作会让你读取到不正确的值引发BUG,于是就会清空渲染队列并且执行,让你拿到最精确/新的

  • 当你请求向浏览器请求一些 style信息的时候,就会让浏览器flush队列,比如:
    • offsetTop, offsetLeft, offsetWidth, offsetHeight
    • scrollTop/Left/Width/Height
    • clientTop/Left/Width/Height
    • width,height
  • 当你请求上面的一些属性的时候,浏览器为了给你最精确的值,需要flush队列,
  • 因为队列中可能会有影响到这些值的操作。即使你获取元素的布局和样式信息跟最近发生或改变的布局信息无关,
  • 浏览器都会强行刷新渲染队列。

清空渲染队列后

  • 当读取offsetHeight属性后,我们清空了渲染队列,那么此时dom重新渲染完成后,此时display已经是block了。而且展示在界面上面了,我们再操作dom属性就会出现过渡动画了。

最后

  • 纸上得来终觉浅,多实践、多思考是走向更高级别必经之路,想要看我之前手写源码文章的,我的gitHub源码地址是:https://github.com/JinJieTan/Peter-,记得Star

作者:Peter 谭老师

转发链接:
https://mp.weixin.qq.com/s/XhwPOv62gypzq5MhhP-5vg

相关推荐

3 分钟!AI 从零开发五子棋全过程曝光,网友:这效率我服了

<!DOCTYPEhtml><htmllang="zh-CN"><head><metacharset="UTF-8...

一行代码实现display&quot;过渡动画&quot;原理

作者:Peter谭老师转发链接:https://mp.weixin.qq.com/s/XhwPOv62gypzq5MhhP-5vg写本文的起因上篇文章,提到如何让display出现过渡动画,却没有仔...

脑洞:琼恩·雪诺、蝙蝠侠和魔形女的灵魂宠物了解一下

AlekseiVinogradovisaRussianfreelancedigitalartistwhoshareshisskillsandtalentwith120k...

浏览器的渲染机制、重绘、重排

1、什么是重排和重绘网页生成过程:HTML被HTML解析器解析成DOM树css则被css解析器解析成CSSOM树结合DOM树和CSSOM树,生成一棵渲染树(RenderTree)生成布局(flo...

托福写作高频考题写作思路&amp;词汇丨考虫独家

科技话题与媒体话题是托福写作的常考话题很多考生对这两类话题里的专有词汇表达也许很不了解所以今天就跟随考虫托福写作老师刘云龙老师一起来学习在这些话题的写作里你可以使用哪些有用的表达。希望大家有收获!记得...

在优麒麟上使用 Electron 开发桌面应用

使用Web标准来创建桌面GUI,上手快、成本低、跨平台、自适应分辨率,这些都是Electron的优势。作者/来源:优麒麟Electron是由Github开发,用HTML、CSS和...

php手把手教你做网站(三十八)jquery 转轮盘抽奖,开盲盒

抽奖和开盲盒性质一样的都是通过ajax读取后台的随机数据。1、转轮盘本来是想直接绘图实现轮盘,但是没有找到怎么填充文字,只好把轮盘弄成了背景图,通常用于游戏抽道具,商城积分抽奖,公司年末员工抽奖点击抽...

用 CSS 整活!3D 轮播图手把手教学,快乐代码敲出来

兄弟们,今天咱来搞点好玩的——用CSS整一个3D轮播图!咱野生程序员就是要在代码里找乐子,技术和快乐咱都得要!代码是写不完的,但咱能自己敲出快乐来,走起!一、先整个容器,搭个舞台咋先写一个...

实现一个超酷的 3D 立体卡片效 #前端开发

今天我们来实现一个超酷的3D立体卡片效果。正常情况下就是一个普通的图片展示卡片,鼠标悬停的时候图片会跳出卡片,并将影子投射到背景卡片上,在视觉上有一个3D立体感。html主要分成3个部分:容器→背景层...

Vue 3 Teleport与Suspense:解决UI难题的两个&quot;隐藏大招&quot;

模态框的"层级噩梦"与Teleport的救赎"这个模态框怎么又被父容器截断了?"团队协作开发后台系统时,小张第N次遇到这个问题。多层嵌套的组件结构里,弹窗被overfl...

让交互更加生动!有意思的鼠标跟随 3D 旋转动效

今天,群友问了这样一个问题,如下所示的鼠标跟随交互效果,如何实现:简单分析一下,这个交互效果主要有两个核心:借助了CSS3D的能力元素的旋转需要和鼠标的移动相结合本文,就将讲述如何使用纯CSS...

填坑:transform元素导致zindex失效终极方法

今天遇到了使用css3动画的元素层级被放大置顶的问题,ios浏览器上没问题,安卓原生浏览器和安卓微信上有问题。使用了css3动画的元素z-index失效,兄弟元素设置多高的z-index都盖不住解决办...

诡异的层级错乱:一个被transform隐藏的CSS陷阱

周五下午三点十七分,设计部突然发来紧急截图——原本应该悬浮在顶部的导航菜单,此刻正诡异地被下方的轮播图遮挡。我盯着屏幕上错乱的层级关系,手指下意识地敲下z-index:9999,心里清楚这不过是程序...

动画篇--碎片动画

本文授权转载,作者:Sindri的小巢(简书)前言从最开始动笔动画篇的博客,至今已经过去了四个多月。这段时间回头看了看自己之前的动画文章,发现用来讲解动画的例子确实不那么的赏心悦目。于是这段时间总是想...

Nature:大洋转换断层处的拉张构造与两阶段地壳增生

Nature:大洋转换断层处的拉张构造与两阶段地壳增生转换断层是三种基本的板块边界之一,全球总长度超过48000km(Bird,2003),它们的发现为板块构造理论的建立奠定了重要的基础(Wil...