【编者按】本文主要通过生动的实例,介绍为 Angular 应用添加动画的原理与过程。文章系国内ITOM 管理平台OneAPM 编译呈现。
我们知道,Angular 应用在更新 DOM 时,会直接将元素转储为视图而没有过渡,其默认的用户体验并不和谐。
不过,好消息是,Angular 附带了对动画的大力支持;当然,坏消息是它可能和预期效果有所出入。Angular 并不能制作动画,但是为用户的自定义动画提供了许多组件。
在非 AngularJavascript 应用中更新 DOM 时,程序员会无意识地在动画中加入自定义成分;但是,在 Angular 应用中,经常会使用内置指令,而不是在DOM上直接更改。
如果不使用 Angular,怎样将动画添加到Web应用中呢?你需要:
定义动画开始和结束的风格;
添加或更改某个元素,并将其设置为起始风格;
设置动画的结束风格;
通常,你会使用Javascript或CSS来完成以上步骤。
当往 Angular 应用添加动画时,当然也要遵循这个模式,但是却以 Angular 特有的方式——动画代码完全从指令代码分离出来。
Angular 的内置指令是预先为动画设定的。这就意味着,你可以使用许多通过 CSS 类或 Javascript 代码就能调用的动画“事件”。这些事件与元素或类的添加/删除相对应。
这可能听起来有点怪,但其好处是你可以创建自定义指令,然后让这些指令的终端用户自定义他们的动画。
这正是 Angular 设计指令的特有方式。这样一来,由于 Angular 没有预定义动画,开发和设计人员就可以选择自己喜欢的方式来创建动画,比如利用CSS过渡/动画或JavaScript库。
如果自己写一个简单的自定义指令并做成动画,更有助于理解各个部分如何协同工作;然后再回过头来,更容易理解内置指令的工作模式。
下面是一个简单的指令,旨在无动画支持时隐藏元素:
app.controller("example", function($scope){ $scope.awesome = false; }); app.directive("myHide", function(){ return { restrict: 'A', link: function(scope, elem, attrs){ scope.$watch(attrs.myHide, function(value){ if (value) { elem.addClass("hide"); } else { elem.removeClass("hide"); } }); } }; });
myHide 指令关注着一个表达式的取值(本例中 ‘awesome’ 的值),当表达式判定结果为真时,在元素中添加类;若为假,则移除类。因为类集显示设为 none,所以当表达式为真时 myHide 元素为隐藏状态。
<div class="myHideExample" ng-controller="example"> <div class="message"> <p my-hide="awesome">Hide this text if awesome</p> </div> <button class="button" ng-click="awesome = !awesome">Toggle awesomeness</button> </div>
这有动画效果,但没有过渡,只是弹出进出。
既然 Angular 动画只是在关键事件元素中添加CSS类(或通过触发Javascript回调函数,我们稍后将会介绍),再加上Javascript 只能添加或删除 CSS 类的约束条件,我们可以为指令添加一个简单的渐淡动画。因为Javascript 并不了解动画过程,所以若不定义CSS 类,指令虽然可以执行,但不会产生动画效果。
myHide 动画能使元素的不透明度从1淡化到0(当状态切换时则反之)。在动画结束时,显示应该设置为none。
这就有一个有趣的问题,因为只有动画结束时才能将显示设为none——否则整个动画运行时,该元素不可见。因此,需要一个CSS类代表过渡/动画,还需要另一个CSS类,方便在所有事情完成后将显示设为none。
//the final state .hide { display: none; } //the animation .hide-add-start { transition: opacity 1s; opacity: 0; }
接着,再在适当的时候,把指令中的几行 Javascript 语句加入到类中。
/ add this first to start the animation elem.addClass("hide-add-start"); setTimeout(function() { // add the hide class after animation is finished elem.addClass("hide"); // clean up elem.removeClass("hide-add-start"); }, 1000);
所以 .hide-add-start 类添加了过渡效果和最终值,过渡完成之后再添加 .hide 类以便将显示设为 none。
. hide-remove { transition: opacity 1s; opacity: 0; } .hide-remove-active { opacity: 1; }
在实现移除 .hide 类的动画效果时,第一步是将不透明度设为0;否则,该元素会直接弹出,不存在任何动画效果。
为了创建不透明度从0到1过渡效果,需要另一个类来定义结束状态。因此,需要一个类来定义起始状态和过渡/动画,另一个类来定义结束状态的动画。
Javascript 代码与添加.hide类的过程几乎一样,但是现在需要两个类。
elem.addClass("hide-remove"); elem.removeClass("hide"); // cause a reflow elem[0].offsetHeight; elem.addClass("hide-remove-active"); setTimeout(function(){ elem.removeClass("hide-remove"); elem.removeClass("hide-remove-active"); }, 1000);
移除.hide类和添加.hide-remove-active类之间的那一行代码会引起回流。如果没有那行,浏览器就不能应用过渡效果,造成元素直接弹出。
为指令添加动画并不像添加和删除一个类那样简单。你需要知道动画什么时候开始、什么时候结束,开始和结束的状态,知道后需要 JavaScript 协调这一切,这也正是 $animate 的作用内容。
$Animate 服务有添加/删除类和元素的方法。当在指令中使用这些方法时,针对制作动画的元素,Angular 会自动添加和删除类。
它还能在正确的时间添加或删除类,因此你可以自定义开始和结束状态。不仅如此,Angular还能从CSS中读取时间,以便在同一位置定义时间。
$animate 服务有几种用于添加/删除/移动元素或添加/删除类的方法。其理念是使用这些方法而不是直接操作DOM,并用 Angular 触发 Javascript 动画,或添加/删除额外需要的CSS类。
你无需加载 ngAnimate 就可以注入 $animate 服务,而且在不触发动画的情况下各个部分都能正常工作。这就太好了,因为即使未定义或使用动画,你也可以创建正常工作的自定义指令。
如果希望动画被激活,就必须下载 ng-animate 模块 Javascript,并把ng-animate 模块列入你的应用程序,如下所示:
var app = angular.module('animations', ['ngAnimate']);
有了 $animate,myHide 指令的新版本如下所示:
app.directive("myHide", function($animate){ return { restrict: 'A', link: function(scope, elem, attrs){ scope.$watch(attrs.myHide, function(value){ if (value) { $animate.addClass(elem, "hide"); } else { $animate.removeClass(elem, "hide"); } }); } }; });
CSS将略有不同。除了要添加到元素中的实际的类,addClass 和 removeClass 语句还添加了两个附加的类:其中一个用于动画和起始风格,另一个用于结束风格。这两个附加类在结束时都会被删除。
添加CSS类需遵循命名约定。因此,在本例中,你添加的类是 “hide” ,则 $animate 会在应定义动画和起始风格的位置再添加一个 “hide-add” 类,同时在任意结束风格的位置添加一个 “hide-add-active” 类。
以下是一个 说明文档 的截图,其中说明了需要创建哪些额外的类,命名约定和每个类的添加时间。
根据以上规则,CSS 可如下所示:
. .hide-add { display: block; transition: opacity 1s; opacity: 1; } .hide-add-active { opacity: 0; } .hide-remove { transition: opacity 1s; display: block; opacity: 0; } .hide-remove-active { opacity: 1; }
“hide-add” 类将显示值设为 “block”,因为 “hide” 类在同一时间加入,并设置显示为 “none”,而这不是我们想要的。
即使 $animate 指令只能添加一个类,但是它同样支持 DOM 上用于添加/删除CSS类的其他操作方法,因此你可以在 Angular 应用上实现几乎所有动画。
大多数内置指令都使用 $animate
进行DOM操作,这意味着你同样可以为它们实现动画。若想了解使用 $animate
的内置指令列表,可点击 此处 。
你也可以使用 Javascript 动画而不是 CSS 动画/过渡。下面的实例使用了TweenMax 库,不过你也可以使用其他自己喜欢的库。
除了添加 CSS 类,$animate 服务也能触发你在 APP 中定义的任何JavaScript动画。
app.directive("myHideJs", function($animate){ return { restrict: 'A', link: function(scope, elem, attrs){ scope.$watch(attrs.myHideJs, function(value){ if (value) { $animate.addClass(elem, "hide-js"); } else { $animate.removeClass(elem, "hide-js"); } }); } }; }); app.animation('.hide-js-animated', function(){ return { addClass: function(element, className){ TweenMax.to(element, 1, { 'opacity': 0 }); }, removeClass: function(element, className){ TweenMax.to(element, 1, { 'opacity': 1 }); } } });
可以看到,在该指令使用 $animate 服务和用其进行 CSS 动画的方式一样,并无区别。
指令下面是动画,使用简单、单一的 CSS 类选择器来命名。使用该动画的元素必须包括这个类,否则将无法进行动画操作。
由动画调用返回的对象定义了两个属性,addClass 和 removeClass。定义这两个属性则是因为指令中用到了addClass 和 removeClass。如果元素从指令中移除或添加,则定义为 ‘leave’ 和 ‘enter’ 属性。
你可以在”由JavaScript定义的动画”部分 查看完整的事件列表 。
下面是使用JavaScript动画的Angular模板。请注意,最终要作动画的元素中的类,要与Angular应用所定义的动画名称匹配。
<div class="myHideExample" ng-controller="example"> <div class="message"> <p my-hide-js="awesome" class="hide-js-animated">Hide this text if awesome</p> </div> <button class="button" ng-click="awesome = !awesome">Toggle awesomeness</button> </div>
大多数内置指令都使用 $animate,正如 myHide指令。下面为ngHide代码:
var ngHideDirective = ['$animate', function($animate) { return { restrict: 'A', multiElement: true, link: function(scope, element, attr) { scope.$watch(attr.ngHide, function ngHideWatchAction(value) { // The comment inside of the ngShowDirective explains why we add and // remove a temporary class for the show/hide animation $animate[value ? 'addClass' : 'removeClass'](element,NG_HIDE_CLASS, { tempClasses: NG_HIDE_IN_PROGRESS_CLASS }); }); } }; }];
是不是很眼熟?这是因为它几乎和你这段时间一直在看的 myHide 指令完全一样。不过也有少许不同,主要是ngHide使用三元运算符来代替 if / else,从而确定调用 addClass还是removeClass。
再看看其他内置指令,就会看到对 $animate的调用。每个指令的说明文档记录了可以在动画中使用的事件列表。之后,就只是创建CSS动画还是JavaScript动画,以及将所有名称都与命名约定相匹配的问题。
Angular的学习曲线虽然并不简单,但归根结底还是值得我们学习的。不过, Angular 充满了奇怪的新概念,而且最终的结果有时看起来简直不可思议。
所有的框架都坚持己见,Angular 也不例外。问题在于,通过 Angular可以创建运行简单的应用程序;但是,在了解它之前,你可能会遇到许多难以检测和调试的问题。这时候,借助OneAPM 提供的检测工具,就能轻松解决这些难题。
OneAPM Browser Insight 是一个基于真实用户的 Web前端性能监控平台,能帮助大家定位网站性能瓶颈,实现网站加速效果可视化;支持浏览器、微信、App 浏览 HTML 和 HTML5 页面。想阅读更多技术文章,请访问OneAPM 官方技术博客。
原文链接: http://www.planningforaliens.com/angular/animate-your-angular-application/