Motion Effects & Animation 和 Autolayout

Motion Effects

在iOS7中,当用户倾斜设备时,一个视图可以实时地响应。通常情况下,视图的响应将是稍微改变其位置。这被用于,例如,在该界面的各部分,让界面有种层叠感。当UIAlertView存在时,如果使用者倾斜装置,该UIAlertView会移动其位置;效果有点微妙,但足以表明UIAlertView稍微在屏幕的前面漂浮。

你自己的视图也可以用同样的方式来表现。一个视图如果有一个或多个motion effects动画效果(UIMotionEffect)。这个动画效果通过addMotionEffect:添加到视图上,motionEffects方法可以列举所有视图上的motion effect,removeMotionEffect:可以从视图上移除指定的motion effect。

该UIMotionEffect类是抽象的:它的工作是被继承。提供的主要子类是UIInterpolatingMotionEffect。UIInterpolatingMotionEffect有一个简单的键值路径,也就是使用键-值编码来指定它所影响的属性。它也有一个类型,指定受到设备倾斜的哪个方向(水平倾斜或垂直倾斜)影响。最后,它有一个最大和最小相对值,即认为受影响的属性对于设备倾斜的偏移值。相关的的motion effects应该合并成一个UIMotionEffectGroup(也是一个UIMotionEffect 子类),然后把它添加到视图中:

UIInterpolatingMotionEffect* m1 =
    [[UIInterpolatingMotionEffect alloc] initWithKeyPath:@"center.x"
    type:UIInterpolatingMotionEffectTypeTiltAlongHorizontalAxis];
m1.maximumRelativeValue = @5.0;
m1.minimumRelativeValue = @-5.0;
UIInterpolatingMotionEffect* m2 =
    [[UIInterpolatingMotionEffect alloc] initWithKeyPath:@"center.y"
    type:UIInterpolatingMotionEffectTypeTiltAlongVerticalAxis];
m2.maximumRelativeValue = @5.0;
m2.minimumRelativeValue = @-5.0;
UIMotionEffectGroup* g = [UIMotionEffectGroup new];
g.motionEffects = @[m1,m2];
[self.mars addMotionEffect:g];

你也可以实现自己的UIMotionEffect子类,通过实现一个简单的方法keyPathsAndRelativeValuesForViewerOffset:即可。当一般很少需要这么做。


Animation and Autolayout

动画和自动布局之间的相互影响可能会非常棘手。作为动画的一部分,你可能会改变一个视图的frame(bounds 或者center)。但当你使用自动布局时,是不应该改变frame的。结果是,动画可能不执行,或者动画执行正常,但是布局却没有发生;但是布局是完全有可能会发生的,这样就会伴随产生不良的影响。

正如我在前面的文章说过,当布局是自动布局管理的,那么视图的约束是关键。如果约束在布局时不能解决视图的大小和位置问题,那么视图就会跳出约束的限制。这是你不想要结果。

为了说明这将会是一个问题,仅仅以动画方式改变一个视图的位置,然后立刻调用layoutIfNeeded来布局:

CGPoint p = self.v.center;
p.x += 100;
[UIView animateWithDuration:1 animations:^{
    self.v.center = p;
} completion:^(BOOL b){
    [self.v layoutIfNeeded]; // this is what will happen at layout time
}];

如果我们使用自动布局,视图滑向右侧,然后跳回到左边。这是不好的。我们应该保持约束与实际同步,这样当布局发生时,我们的视图就不会跳进一个不确定的状态。

一种选择是修改被违反的约束,以符合新的实际情况。如果我们想得更远一点,我们可以提前拿到这些约束的引用;这种情况下我们的代码就可以移除和替换这些约束—-或者,如果唯一需要改变是一个约束的常量值,我们可以改变这个值(记得,常量是现有约束的唯一可写属性)。否则,当发现有什么违反了约束是,再获得对它们的引用,是很不容易的。

另一种方法,在需要改变的仅仅是一个约束常量的情况下,我们可以直接驱动这个约束的常量值的变化来执行动画,而不是改变视图的位置来执行动画。要做到这一点,我们需要设置这个约束常量一个新值,然后对布局行为执行动画。这是假定我们拿到了这个约束的引用的情况下。

// con is the constraint
con.constant += 100;
[UIView animateWithDuration:1 animations:^{
    [self.v layoutIfNeeded];
}];

另一个问题是怎么处理视图的变换transforms。正如我前面文章说过的,提供给视图一个变换会立刻触发布局,约束就会负责定位这个视图。因此,在自动布局下对视图进行变换会发生一些意想不到的事情。

例如,你可能会觉得下面的动画在自动布局下会正常工作,但是你错了,即使这个简单的拉伸动画也会在自动布局下被打断—结果不是简单的拉伸动画,而是视图可能会不时跳到不同的位置:

[UIView animateWithDuration:0.3 delay:0
        options:UIViewAnimationOptionAutoreverse animations:^{
    self.v.transform = CGAffineTransformMakeScale(1.1, 1.1);
} completion:^(BOOL finished) {
    self.v.transform = CGAffineTransformIdentity;
}];

这种情况下,我们的解决方法是使用Core Animation;因为提供给图层一个变换动画,是不会触发布局的:

CABasicAnimation* ba =
    [CABasicAnimation animationWithKeyPath:@"transform"];
ba.autoreverses = YES;
ba.duration = 0.3;
ba.toValue =
    [NSValue valueWithCATransform3D:CATransform3DMakeScale(1.1, 1.1, 1)];
[self.v.layer addAnimation:ba forKey:nil];

另一种可能的做法是使用原始视图的快照,把这个快照临时添加到界面上—不使用自动布局,然后把原来的视图隐藏,最后驱动这个快照指定动画:

UIView* snap = [self.v snapshotViewAfterScreenUpdates:YES];
snap.frame = self.v.frame;
[self.v.superview addSubview:snap];
self.v.hidden = YES;
[UIView animateWithDuration:0.3 delay:0
         options:UIViewAnimationOptionAutoreverse animations:^{
    snap.transform = CGAffineTransformMakeScale(1.1, 1.1);
} completion:^(BOOL finished) {
    snap.transform = CGAffineTransformIdentity;
    self.v.hidden = NO;
    [snap removeFromSuperview];
}];

但是,如果动画的意图是,真正的视图最终需要被转移到新的永久位置上,那么它的约束仍然要进行修改。

另一个有用的技巧是认清一个事实,就是“动画电影”只是实际的一个面具。下面的例子是我缩小视图(english)到没有:

CABasicAnimation* ba = [CABasicAnimation animationWithKeyPath:@"opacity"];
self.english.layer.opacity = 0;
ba.duration = 0.2;
[self.english.layer addAnimation:ba forKey:nil];
CABasicAnimation* ba2 = [CABasicAnimation animationWithKeyPath:@"bounds"];
ba2.duration = 0.2;
ba2.toValue = [NSValue valueWithCGRect:self.english.layer.bounds];
[self.english.layer addAnimation:ba2 forKey:nil];

这在自动布局下并不会被中断,因为我从来没有做过任何违反现有约束的事,我没有改变视图的bounds!我让这个图层不可见,通过改变它的opacity为0。另一方面,动画仍然驱动这个视图缩小到没有,这是一个假象,用户不会发现其实它仍处于原来的尺寸。

自动布局从iOS6开始出现,那时候就是与动画不兼容的,这个不兼容是iOS的一个严重缺陷,而苹果却刻意掩饰它,避谈这两者的不兼容。

最近的文章

Touches 触摸 & Gesture Recognizer 手势识别

每个UITouch对象对应一个手指。反过来说,每一个手指触摸屏幕时是由在UIEvent里的一个UITouch对象表示的。 对于给定的UITouch对象(请记住,一个特定的手指),只有五件事情可能发生。这些被称为接触阶段,并通过一个UITouch实例的相位特性进行了说明: UITouchPhaseBegan 手指第一次触摸屏幕,这个UITouch对象刚刚被创建。这总是第一个阶段,而且只会进入一次。 UITouchPhaseMoved 手指在屏...…

iOS继续阅读
更早的文章

CIFilter Transitions & UIKit Dynamics

CIFilter Transitions : CIFilter 转换Core Image 过滤器包含了转换。你提供两张图片和一个介于0到1的帧时间;过滤器提供一秒内从第一张图片转换到第二张图片的相应帧。例如下图显示了在0.75秒时的帧画面,正在从纯红色的图片通过星光发射转换动画转换到我自己的图片。(你看不到这张我的图片,因为这个转换,默认会先把第一张图片“爆炸”成纯白色,然后迅速消失,出现第二张图片)驱动一个Core Image 转换过滤器执行动画是由你决定。因此,我们需要反复快速调用相同...…

iOS继续阅读