为什么UILabel边界上的动画仅在增大尺寸时起作用?

时间:2013-06-28 08:27:26

标签: ios objective-c uiviewanimation

我注意到当我在动画块中更改UILabel的边界时,它只有在我增加大小时才有效,当我减小大小时,UILabel只是改变了他的大小但没有动画。 用普通UIView替换UILabel可以按预期工作。

注意:将UILabel的contentMode属性更改为UIViewContentModeScaleToFill可以解决此问题,但我仍然不明白为什么在增加大小而不更改contentMode属性时它可以正常工作。

#import "FooView.h"

@interface FooView ()
@property (strong, nonatomic) UILabel   *label;
@property (assign, nonatomic) BOOL       resized;
@end

@implementation FooView

- (id)initWithFrame:(CGRect)frame
{
    self = [super initWithFrame:frame];
    if (self) {
        self.backgroundColor = [UIColor lightGrayColor];

        self.label = [[UILabel alloc] initWithFrame:(CGRect){0, 0, frame.size}];
        self.label.backgroundColor = [UIColor greenColor];
        [self addSubview:self.label];
        _resized = false;

        UITapGestureRecognizer *tapRecognizer = [[UITapGestureRecognizer alloc] initWithTarget:self action:@selector(changeSize)];
        tapRecognizer.numberOfTapsRequired = 1;
        [self addGestureRecognizer:tapRecognizer];
    }
    return self;
}

- (void)changeSize {
    [UIView animateWithDuration:0.8
                          delay:0.0
                        options:UIViewAnimationOptionCurveEaseIn
                     animations:^{
                         CGRect bounds = self.label.bounds;
                         if (self.resized) {
                             bounds.size.height *= 2;
                             bounds.size.width  *= 2;
                         } else {
                             bounds.size.width  /= 2;
                             bounds.size.height /= 2;
                         }
                         self.label.bounds = bounds;
                     }
                     completion:^(BOOL finished) {
                         self.resized = !self.resized;
                     }];
}

@end

1 个答案:

答案 0 :(得分:24)

这是因为UILabel将其图层的contentsGravity设置为正在呈现文本的方向,这恰好默认为UIViewContentModeLeft(或@"left")。因此,当告诉该层动画时,它首先看一眼其内容的重力并将随后的动画置于其上。因为它看到@"left"应该有@"resize",所以它假定缩放动画应该从左边开始,但它也必须尊重你给它的约束(边界变化),所以你的标签看起来会跳到它的最终尺寸,然后在它应该位于屏幕中央的位置。

如果您想单独留下contentMode,请使用CATransform3D并以此方式缩放标签图层,而不是更改边界。