图层行为(隐式动画)

图层行为

今昔来做个实验,试着直接对UIView关联的图层做动画而不是一个单身的图层。清单7.4是对清单7.2代码的一些修改,移除了colorLayer,并且一贯设置layerView事关图层的背景观。

清单7.4 直接设置图层的性质

2018正版葡京赌侠诗 12018正版葡京赌侠诗 2

 1 @interface ViewController ()
 2 
 3 @property (nonatomic, weak) IBOutlet UIView *layerView;
 4 
 5 @end
 6 
 7 @implementation ViewController
 8 
 9 - (void)viewDidLoad
10 {
11     [super viewDidLoad];
12     //set the color of our layerView backing layer directly
13     self.layerView.layer.backgroundColor = [UIColor blueColor].CGColor;
14 }
15 
16 - (IBAction)changeColor
17 {
18     //begin a new transaction
19     [CATransaction begin];
20     //set the animation duration to 1 second
21     [CATransaction setAnimationDuration:1.0];
22     //randomize the layer background color
23     CGFloat red = arc4random() / (CGFloat)INT_MAX;
24     CGFloat green = arc4random() / (CGFloat)INT_MAX;
25     CGFloat blue = arc4random() / (CGFloat)INT_MAX;
26     self.layerView.layer.backgroundColor = [UIColor colorWithRed:red green:green blue:blue alpha:1.0].CGColor;
27     //commit the transaction
28     [CATransaction commit];
29 }

View Code

 

运作程序,你会发觉当按下按钮,图层颜色弹指间切换来新的值,而不是此前平滑过渡的卡通。暴发了如何吧?隐式动画好像被UIView关联图层给禁用了。

试想一下,如果UIView的性能都有动画特性的话,那么不论是在什么样时候修改它,我们都应有能注意到的。所以,假设说UIKit建立在Core
Animation(默许对所有东西都做动画)之上,那么隐式动画是什么样被UI基特禁用掉呢?

我们知道Core
Animation日常对CALayer的有所属性(可动画的性能)做动画,不过UIView把它事关的图层的这些特点关闭了。为了更好注解那或多或少,大家需要领悟隐式动画是怎样已毕的。

我们把改变属性时CALayer自动应用的动画称作行为,当CALayer的习性被涂改时候,它会调用-actionForKey:艺术,传递属性的称呼。剩下的操作都在CALayer的头文件中有详尽的证实,实质上是之类几步:

  • 图层首先检测它是还是不是有嘱托,并且是或不是贯彻CALayerDelegate协和指定的-actionForLayer:forKey办法。假若有,直接调用并回到结果。
  • 要是没有委托,或者委托没有落实-actionForLayer:forKey办法,图层接着检查包罗属性名称对应行为映射的actions字典。
  • 如果actions字典从没包蕴相应的属性,那么图层接着在它的style字典接着搜索属性名。
  • 最后,如果在style内部也找不到相应的行为,那么图层将会间接调用定义了每个属性的科班行事的-defaultActionForKey:方法。

就此一轮完整的追寻截至之后,-actionForKey:要么重临空(那种景色下将不会有动画爆发),要么是CAAction2018正版葡京赌侠诗,钻探对应的靶子,末了CALayer拿那几个结果去对原先和脚下的值做动画。

于是乎那就表达了UIKit是怎样禁用隐式动画的:每个UIView对它关系的图层都扮演了一个委托,并且提供了-actionForLayer:forKey的贯彻方式。当不在一个卡通块的落到实处中,UIView对负有图层行为再次回到nil,不过在动画block范围以内,它就回来了一个非空值。我们得以用一个demo做个几乎的尝试(清单7.5)

清单7.5 测试UIView的actionForLayer:forKey:实现

2018正版葡京赌侠诗 32018正版葡京赌侠诗 4

 1 @interface ViewController ()
 2 
 3 @property (nonatomic, weak) IBOutlet UIView *layerView;
 4 
 5 @end
 6 
 7 @implementation ViewController
 8 
 9 - (void)viewDidLoad
10 {
11     [super viewDidLoad];
12     //test layer action when outside of animation block
13     NSLog(@"Outside: %@", [self.layerView actionForLayer:self.layerView.layer forKey:@"backgroundColor"]);
14     //begin animation block
15     [UIView beginAnimations:nil context:nil];
16     //test layer action when inside of animation block
17     NSLog(@"Inside: %@", [self.layerView actionForLayer:self.layerView.layer forKey:@"backgroundColor"]);
18     //end animation block
19     [UIView commitAnimations];
20 }
21 
22 @end

View Code

 

运行程序,控制台突显结果如下:

2018正版葡京赌侠诗 52018正版葡京赌侠诗 6

1 $ LayerTest[21215:c07] Outside: <null>
2 $ LayerTest[21215:c07] Inside: <CABasicAnimation: 0x757f090>

View Code

 

于是大家可以断言,当属性在动画块之外发生变更,UIView直白通过重回nil来剥夺隐式动画。但假设在动画块范围以内,按照动画具体品种再次来到相应的性质,在那些事例就是CABasicAnimation(第八章“显式动画”将会提到)。

当然重临nil并不是剥夺隐式动画唯一的方式,CATransacition有个措施叫做+setDisableActions:,可以用来对持有属性打开或者关闭隐式动画。即使在清单7.2的[CATransaction begin]未来添加上边的代码,同样也会堵住动画的暴发:

[CATransaction setDisableActions:YES];

小结一下,大家明白了如下几点

  • UIView涉及的图层禁用了隐式动画,对那种图层做动画的绝无仅有方式就是选拔UIView的动画片函数(而不是依靠CATransaction),或者一而再UIView,并覆盖-actionForLayer:forKey:办法,或者直接创建一个显式动画(具体细节见第八章)。
  • 对于单身存在的图层,大家得以因此落到实处图层的-actionForLayer:forKey:信托方法,或者提供一个actions字典来支配隐式动画。

我们来对颜色渐变的事例使用一个见仁见智的行为,通过给colorLayer安装一个自定义的actions字典。我们也得以选拔委托来促成,可是actions字典可以写更少的代码。那么究竟改什么创设一个恰当的表现目标啊?

作为平时是一个被Core
Animation隐式调用的显式卡通对象。那里大家利用的是一个完成了CATransaction的实例,叫做力促过渡

第八章元帅会详细表达过渡,然而对于当今,知道CATransition响应CAAction说道,并且可以当做一个图层行为就够用了。结果很赞,不论在怎么样时候改变背景颜色,新的色块都是从左侧滑入,而不是默许的渐变效果。

清单7.6 完成自定义行为

2018正版葡京赌侠诗 72018正版葡京赌侠诗 8

 1 @interface ViewController ()
 2 
 3 @property (nonatomic, weak) IBOutlet UIView *layerView;
 4 @property (nonatomic, weak) IBOutlet CALayer *colorLayer;/*热心人发现这里应该改为@property (nonatomic, strong)  CALayer *colorLayer;否则运行结果不正确。
 5 */
 6 
 7 @end
 8 
 9 @implementation ViewController
10 
11 - (void)viewDidLoad
12 {
13     [super viewDidLoad];
14 
15     //create sublayer
16     self.colorLayer = [CALayer layer];
17     self.colorLayer.frame = CGRectMake(50.0f, 50.0f, 100.0f, 100.0f);
18     self.colorLayer.backgroundColor = [UIColor blueColor].CGColor;
19     //add a custom action
20     CATransition *transition = [CATransition animation];
21     transition.type = kCATransitionPush;
22     transition.subtype = kCATransitionFromLeft;
23     self.colorLayer.actions = @{@"backgroundColor": transition};
24     //add it to our view
25     [self.layerView.layer addSublayer:self.colorLayer];
26 }
27 
28 - (IBAction)changeColor
29 {
30     //randomize the layer background color
31     CGFloat red = arc4random() / (CGFloat)INT_MAX;
32     CGFloat green = arc4random() / (CGFloat)INT_MAX;
33     CGFloat blue = arc4random() / (CGFloat)INT_MAX;
34     self.colorLayer.backgroundColor = [UIColor colorWithRed:red green:green blue:blue alpha:1.0].CGColor;
35 }
36 
37 @end
38  

View Code

2018正版葡京赌侠诗 9

 

发表评论

电子邮件地址不会被公开。 必填项已用*标注