如何判断一个DOM元素正在动画,一个CSS“阻塞”JS的例子

本文探讨了CSS动画与JS逻辑结合时可能出现的问题,尤其是在动画类丢失时如何确保动画正常执行并正确处理动画结束后的逻辑。通过实际案例,介绍了如何检测CSS动画属性的存在,以及在动画未生效时采取的备选方案。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

一般情况下CSS不会直接影响JS的程序逻辑,但是以CSS实现动画的话,这个便不太确定了,这个故事发生在与UED迁移全局样式的过程。

曾经我有一段实现弹出层隐藏动画的代码是这个样子的:

  1. if (this.needAnimat && typeof this.animateHideAction == 'function' && this.status != 'hide') {
  2.   this.animateHideAction.call(this, this.$el);
  3. } else
  4.   this.$el.hide();
复制代码
在所有组件中,如果设置了animatHideAction回调的,便会执行其中的动画逻辑,针对弹出层来说:

① alert

② loading

③ toast

④ 底部弹出层

等组件中动画效果各不相同:

① 动画显示时下沉,隐藏时上浮

② 动画渐隐渐显

③ 组件底部弹出

......

针对通用的动画,一般框架会提供一段CSS类做处理,不满足的情况,各个业务团队便需要自己封装:


  1. cm-fade-in, .cm-fade-out, .cm-down-in, .cm-down-out, .cm-up-in, .cm-up-out {
  2.   -webkit-animation-duration: 0.3s;
  3.           animation-duration: 0.3s;
  4.   -webkit-animation-fill-mode: both;
  5.           animation-fill-mode: both;
  6. }
  7. ......
  8. @keyframes fadeOut {
  9.   0% {
  10.     opacity: 1;
  11.     -webkit-transform: scale(1);
  12.             transform: scale(1);
  13.   }
  14.   100% {
  15.     opacity: 0;
  16.     -webkit-transform: scale(1.185);
  17.             transform: scale(1.185);
  18.   }
  19. }
  20. ......
复制代码


这个时候我们要实现一个居中弹出层渐隐的效果事实上只需要这样做:

  1. el.addClass('cm-fade-out');

  2. el.one($.fx.animationEnd, function () {
  3.   el.removeClass('cm-fade-out');
  4.   el.hide();
  5. });
复制代码


在动画结束后将对应的动画class移除,再执行真实的hide方法,隐藏dom结构。

其实,我记得是去年的时候我是这么处理这个代码的,当时被一个同事骂了不严谨,今年就使用了animationEnd接口:

  1. el.addClass('cm-fade-out');

  2. setTimeout(function () {
  3.   el.removeClass('cm-fade-out');
  4.   el.hide();
  5. }, 340);
复制代码


这里问题来了,使用animationEnd与setTimeout去除动画class,或者执行业务真实逻辑,到底哪家强,哪个合适?


第一反应都是认为animationEnd比较合理,于是我最近遇到了一个问题:


请求一个数据,loading一直在那里转,永远不消失了!而且执行了hideLoading的操作,与数据延迟毫无关系


于是我开始愉快的定位,当时搞了一会,发现loading的动画没有执行,仔细一定位,发现css中的动画相关的css丢了,于是造成的结果是:


  1. el.addClass('cm-fade-out');
复制代码


这个代码变成了单纯的class增加,并没有执行动画,也就是,animationEnd的事件没有触发,于是没有执行hide方法,所以loading框就一直在那里转


问题定位到了,解决方案就非常简单了,将css的动画加上即可;但是也说明了,这段代码中JS代码逻辑依赖了CSS相关,从而导致了CSS阻塞JS的假象


这里如果使用setTimeout的话虽然感觉没有animationEnd严谨,但是一定会保证这逻辑代码执行,从某种程度来说,似乎更好,这里的优化代码是:



  1. var isTrigger = false;

  2. el.addClass(scope.animateOutClass);

  3. el.one($.fx.animationEnd, function () {
  4.   isTrigger = true;
  5.   el.removeClass(scope.animateOutClass);
  6.   el.hide();
  7. });

  8. setTimeout(function () {
  9.   if (isTrigger) return;

  10.   el.removeClass(scope.animateOutClass);
  11.   el.off($.fx.animationEnd);
  12.   el.hide();
  13. }, 350);
复制代码


如果animationEnd执行了便不理睬setTimeout,否则便走setTimeout逻辑,也不至于影响业务逻辑,但是这个似乎不是最优解决方案。


因为我没有办法,因为这里得有350ms的延迟,在不存在css动画的时候,似乎整个弹出层消失逻辑都变得2B了起来,比较好的方式是,我在执行动画前检测是否具有该css比较靠谱


所以,javascript检测CSS的某一个className是否存在,似乎变成了关键,但是就算就算能找到具有某class,这个class也未必具有动画属性,或者该属性被篡改


况且使用document.styleSheets方式去判断某个样式class是否存在,经过之前的经验,本身就是大坑,还会有跨域什么的场景,坑死人,比如这个代码:


  1. function getAllSelectors() {
  2.   var ret = [];
  3.   for (var i = 0; i < document.styleSheets.length; i++) {
  4.     var rules = document.styleSheets[i].rules || document.styleSheets[i].cssRules;
  5.     for (var x in rules) {
  6.       if (typeof rules[x].selectorText == 'string') ret.push(rules[x].selectorText);
  7.     }
  8.   }
  9.   return ret;
  10. }

  11. function selectorExists(selector) {
  12.   var selectors = getAllSelectors();
  13.   for (var i = 0; i < selectors.length; i++) {
  14.     if (selectors[i] == selector) return true;
  15.   }
  16.   return false;
  17. }

  18. //调用方式
  19. selectorExists('.class');
  20. selectorExists('#id');
复制代码


上面的代码,本身比较完善了,但是如果某一个css文件跨域的话就完蛋,所以这个方案不靠谱:


① class检测方案本身不靠谱


② 就算class靠谱,也不能保证class就具有动画相关属性,所以也不靠谱!


最终我想到的方案还是对动画属性做检测,检测点主要在动画属性的检测,比如关键属性:


① animation-name


② transition的检测



  1. <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
  2. <html xmlns="http://www.w3.org/1999/xhtml">
  3. <head>
  4.   <title></title>
  5.   <script id="others_zepto_10rc1" type="text/javascript" class="library" src="http://sandbox.runjs.cn/js/sandbox/other/zepto.min.js"></script>
  6.   <style>
  7.    
  8. .cm-fade-in {
  9.   -webkit-animation-name: fadeIn;
  10.           animation-name: fadeIn;
  11. }

  12. .cm-fade-out {
  13.   -webkit-animation-name: fadeOut;
  14.           animation-name: fadeOut;
  15. }

  16. @-webkit-keyframes fadeIn {
  17.   0% {
  18.     opacity: 0;
  19.     -webkit-transform: scale(0.815);
  20.             transform: scale(0.815);
  21.   }
  22.   100% {
  23.     opacity: 1;
  24.     -webkit-transform: scale(1);
  25.             transform: scale(1);
  26.   }
  27. }

  28. @keyframes fadeIn {
  29.   0% {
  30.     opacity: 0;
  31.     -webkit-transform: scale(0.815);
  32.             transform: scale(0.815);
  33.   }
  34.   100% {
  35.     opacity: 1;
  36.     -webkit-transform: scale(1);
  37.             transform: scale(1);
  38.   }
  39. }
  40. @-webkit-keyframes fadeOut {
  41.   0% {
  42.     opacity: 1;
  43.     -webkit-transform: scale(1);
  44.             transform: scale(1);
  45.   }
  46.   100% {
  47.     opacity: 0;
  48.     -webkit-transform: scale(1.185);
  49.             transform: scale(1.185);
  50.   }
  51. }
  52. @keyframes fadeOut {
  53.   0% {
  54.     opacity: 1;
  55.     -webkit-transform: scale(1);
  56.             transform: scale(1);
  57.   }
  58.   100% {
  59.     opacity: 0;
  60.     -webkit-transform: scale(1.185);
  61.             transform: scale(1.185);
  62.   }
  63. }
  64. .cm-down-in {
  65.   -webkit-animation-name: downIn;
  66.           animation-name: downIn;
  67. }

  68. .cm-down-out {
  69.   -webkit-animation-name: downOut;
  70.           animation-name: downOut;
  71. }

  72. @-webkit-keyframes downIn {
  73.   0% {
  74.     opacity: 0;
  75.     -webkit-transform: translate3d(0, 100%, 0);
  76.             transform: translate3d(0, 100%, 0);
  77.   }
  78.   100% {
  79.     opacity: 1;
  80.     -webkit-transform: translate3d(0, 0, 0);
  81.             transform: translate3d(0, 0, 0);
  82.   }
  83. }

  84. @keyframes downIn {
  85.   0% {
  86.     opacity: 0;
  87.     -webkit-transform: translate3d(0, 100%, 0);
  88.             transform: translate3d(0, 100%, 0);
  89.   }
  90.   100% {
  91.     opacity: 1;
  92.     -webkit-transform: translate3d(0, 0, 0);
  93.             transform: translate3d(0, 0, 0);
  94.   }
  95. }
  96. @-webkit-keyframes downOut {
  97.   0% {
  98.     opacity: 1;
  99.     -webkit-transform: translate3d(0, 0, 0);
  100.             transform: translate3d(0, 0, 0);
  101.   }
  102.   100% {
  103.     opacity: 0;
  104.     -webkit-transform: translate3d(0, 100%, 0);
  105.             transform: translate3d(0, 100%, 0);
  106.   }
  107. }
  108. @keyframes downOut {
  109.   0% {
  110.     opacity: 1;
  111.     -webkit-transform: translate3d(0, 0, 0);
  112.             transform: translate3d(0, 0, 0);
  113.   }
  114.   100% {
  115.     opacity: 0;
  116.     -webkit-transform: translate3d(0, 100%, 0);
  117.             transform: translate3d(0, 100%, 0);
  118.   }
  119. }
  120. .cm-up-in {
  121.   -webkit-animation-name: upIn;
  122.           animation-name: upIn;
  123. }

  124. .cm-up-out {
  125.   -webkit-animation-name: upOut;
  126.           animation-name: upOut;
  127. }
  128.   </style>
  129. </head>
  130. <body>
  131.   <script type="text/javascript">
  132.     var hasAnimationProperty = function (className) {
  133.       var animateProprtys = [
  134.       //有什么判断的便新增,暂时只判断animation,不同的动画特性,判断方式不一致
  135.       //        $.fx.cssPrefix + 'transition',
  136.         $.fx.cssPrefix + 'animation-name'
  137.       ];
  138.       var el = $('<div></div>');
  139.       $('body').append(el);

  140.       var i, len;

  141.       //赋予其class
  142.       el.attr('class', className);

  143.       for (i = 0, len = animateProprtys.length; i < len; i++) {
  144.         if (el.css(animateProprtys[i]) != 'none') return true;
  145.       }
  146.       s = '';
  147.       return false;
  148.     };

  149.     //false
  150.     console.log(hasAnimationProperty('test'));
  151.     //true
  152.     console.log(hasAnimationProperty('cm-up-out'));
  153.     //true
  154.     console.log(hasAnimationProperty('cm-up-in'));

  155.   </script>
  156. </body>
  157. </html>
复制代码


核心代码:


  1. var hasAnimationProperty = function (className) {
  2.   var animateProprtys = [
  3.   //有什么判断的便新增,暂时只判断animation,不同的动画特性,判断方式不一致
  4.   //        $.fx.cssPrefix + 'transition',
  5.     $.fx.cssPrefix + 'animation-name'
  6.   ];
  7.   var el = $('<div></div>');
  8.   $('body').append(el);

  9.   var i, len;

  10.   //赋予其class
  11.   el.attr('class', className);

  12.   for (i = 0, len = animateProprtys.length; i < len; i++) {
  13.     if (el.css(animateProprtys[i]) != 'none') return true;
  14.   }
  15.   s = '';
  16.   return false;
  17. };

  18. //false
  19. console.log(hasAnimationProperty('test'));
  20. //true
  21. console.log(hasAnimationProperty('cm-up-out'));
  22. //true
  23. console.log(hasAnimationProperty('cm-up-in'));
复制代码


如此一来,便能判断该class是否具有样式属性了,但是这个代码还需要扩展,而且这么也有性能损害,其中涉及到dom操作了,但是想想动画造成到gpu负担,好像也没什么问题
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值