CABasicAnimation не оживляет мою собственность - PullRequest
8 голосов
/ 19 июля 2010

Я пытался понять, что не так с моей анимацией, и я до сих пор не понял этого. Я думаю, что это должно быть очень просто, но, возможно, я чего-то упускаю, даже после прочтения большого количества примеров и документации.

Моя проблема изначально возникла из-за того, что на iPhone вы не можете изменять размеры слоев автоматически (в режиме просмотра). В документации сказано иное, но в SDK нет слоя autoresizeMask для слоя. Поэтому я решил немного обойти и оживить слой самостоятельно.

У меня есть этот простой кусок кода, который должен выполнять простую анимацию изменения размера. Значения хорошие, и я даже установил делегат, чтобы отслеживать начало / конец анимации.

// I've got a property named layerImage (which is a CALayer)
- (void)animateTestWithFrame:(CGRect)value {
    CABasicAnimation *animation = [CABasicAnimation animationWithKeyPath:@"layerImage.frame"];
    animation.duration = 1;
    animation.fromValue = [NSValue valueWithCGRect:self.frame];
    animation.toValue = [NSValue valueWithCGRect:value];
    animation.removedOnCompletion = YES;
    animation.delegate = self;

    [self.layer addAnimation:animation forKey:@"layerImage.frame"];
}

Итак, есть идеи? (Это представление, содержащее код, является подпредставлением подпредставления окна, если это может иметь значение)

--- РЕДАКТИРОВАТЬ ---

Кажется, что кадр не может быть анимирован с помощью CABasicAnimation и именованного свойства frame. При использовании границ я получаю странный результат, но, по крайней мере, что-то получаю. Будем продолжать расследование по этому вопросу.

Ответы [ 3 ]

26 голосов
/ 20 июля 2010

Так что хорошо, что вы здесь все выяснили, но в вашем ответе на свой вопрос есть некоторые неточности. Позвольте мне исправить несколько вещей:

Свойство фрейма может быть анимированным - только без явной анимации. Если вы сделаете следующее для слоя, отличного от корневого слоя представления, рамка будет анимирована очень хорошо:

[CATransaction begin];
[CATransaction setAnimationDuration:2.0f];
[animationLayer setFrame:CGRectMake(100.0f, 100.0f, 100.0f, 100.0f)];
[CATransaction commit];

Помните, что установка свойства на слое оживляет изменение этого свойства по умолчанию . Фактически вы должны отключить анимацию при изменении свойства, если вы не хотите, чтобы оно анимировалось. (Конечно, это верно только в том случае, если вы анимируете слой, отличный от корневого слоя представления.) Вы правильно анимируете и положение, и границы, если вам нужно использовать явную анимацию.

Вы можете анимировать кадр в UIView, используя неявную анимацию:

[UIView beginAnimations:nil context:NULL];
[UIView setAnimationDuration:3.0f];
[[self view] setFrame:CGRectMake(45.0f, 45.0f, 100.0f, 100.0f)];
[UIView commitAnimations];

Это будет анимация от текущего кадра вида (границы и положение) до x = 45.0, y = 45.0, w = 100.0, h = 100.0.

Возможно, вы также неправильно понимаете разницу между анимацией и слоем. Вы добавляете анимацию к слоям, но добавление анимации к слою не устанавливает автоматически свойство, которое вы анимируете.

CALayers - это модельные объекты. Они содержат информацию о слое, который в итоге отображается на экране. Вы должны установить свойство слоя, если хотите, чтобы это свойство действительно имело это значение. Если вы просто анимируете свойство, оно будет только визуальным представлением, а не фактическим - то есть именно поэтому значение возвращается к исходному значению слоя, потому что вы фактически никогда не меняли его.

Что приводит меня к следующему пункту. Вы сказали:

Использовать "animation.removedOnCompletion = НЕТ; animation.fillMode = kCAFillModeForwards; "чтобы убедиться, что значения не сбрасываются в конце анимации

Это не совсем верно. Эти два значения просто приводят к тому, что анимация остается на своей конечной позиции визуально , однако фактические значения слоя не изменились. Они все те же значения, которые были при запуске анимации. В целом (за некоторыми исключениями) вы не хотите использовать эти два параметра, потому что они только визуальные . То, что вы хотите, это на самом деле установить значение слоя для свойства, которое вы анимируете.

Скажем, например, что вы хотите анимировать положение вашего слоя, используя явную анимацию. Вот код, который вы хотите:

CABasicAnimation *animation = [CABasicAnimation animationWithKeyPath:@"position"];
[animation setFromValue:[NSValue valueWithCGPoint:CGPointMake(70.0f, 70.0f)]];
[animation setToValue:[NSValue valueWithCGPoint:CGPointMake(150.0f, 150.0f)]];
[animation setDuration:2.0f];

// Actually set the position on the *layer* that you want it to be
// when the animation finishes.
[animationLayer setPosition:CGPointMake(150.0f, 150.0f)];

// Add the animation for the position key to ensure that you
// override the animation for position changes with your animation.
[animationLayer addAnimation:animation forKey:@"position"];

Вы также можете рассмотреть группировку анимации. С группой анимации вы можете сгруппировать несколько анимаций вместе, а затем контролировать, как они связаны друг с другом. В вашем случае продолжительность анимации границ и позиции одинакова, и поэтому то, что вы пытаетесь сделать, будет работать без группы, но если вы хотите сместить начало анимации, например, вам не нужна позиция анимация запускается до секунды или двух в кадре анимации, вы можете пошатнуть их, установив beginTime значение анимации позиции.

Наконец, мне было бы любопытно узнать, почему вы не можете использовать неявные анимации, доступные в UIView. Я использую их в подавляющем большинстве кода Core Animation, который я пишу, и не понимаю, почему это не сработает в вашей ситуации.

С уважением.

3 голосов
/ 19 июля 2010

Ключевым путем должен быть только путь к ключу свойства, а не имя объекта.

Используйте это

[CABasicAnimation animationWithKeyPath:@"frame"]

вместо этого

[CABasicAnimation animationWithKeyPath:@"layerImage.frame"]

И, кстати, просто, когда вы добавляете анимацию в слой, ключ не означает ключевое свойство animate. Просто ключ (имя), который вы хотите, чтобы эта анимация имела (это относится к последней строке вашего кода)

1 голос
/ 20 июля 2010

Таким образом, решение состояло в том, чтобы анимировать @ "границы" и @ "положение" слоя, потому что кадр не может быть анимирован на iPhone.Мне потребовалось некоторое время, чтобы понять, что положение было центром слоя, а изменение границ выходило за пределы центра, но это была легкая часть.

Итак, в резюме я сделал следующее:

  1. В setFrame создайте 2 анимации со свойством bounds и position.
  2. Используйте "animation.removedOnCompletion = NO; animation.fillMode = kCAFillModeForwards;"чтобы гарантировать, что значения не будут сброшены в конце анимации
  3. Зарегистрируйте делегата для себя, чтобы реализовать "animationDidStop: закончено:".Кажется, вам все еще нужно установить значения: "layerImage.bounds = [animation.toValue CGRectValue]; layerImage.position = [animation.toValue CGPointValue];".

Я не смогиспользовать анимационную систему UIView напрямую, потому что она не делала то, что я хотел на слоях.

Спасибо tadej5553 за то, что указал мне на проблему со слоями, которую я имел с "addAnimation".Так вот код для тех, кто хотел бы увидеть, как это выглядит.

- (void)animationDidStop:(CAAnimation *)anim finished:(BOOL)flag {
    CABasicAnimation *animation = (CABasicAnimation*)anim;

    if ([animation.keyPath isEqualToString:@"bounds"]) {
        layerImage.bounds = [animation.toValue CGRectValue];
    } else if ([animation.keyPath isEqualToString:@"position"]) {
        layerImage.position = [animation.toValue CGPointValue];
    }
}

- (void)setFrame:(CGRect)value {
    CGRect bounds = CGRectMake(0, 0, value.size.width, value.size.height);

    if ([UIView isAnimationStarted]) {
        // animate the bounds
        CABasicAnimation *animation = [CABasicAnimation animationWithKeyPath:@"bounds"];
        animation.duration = [UIView animationDuration];
        animation.fromValue = [NSValue valueWithCGRect:layerImage.bounds];
        animation.toValue = [NSValue valueWithCGRect:bounds];
        animation.removedOnCompletion = NO;
        animation.fillMode = kCAFillModeForwards;
        animation.timingFunction = [UIView animationFunction];
        animation.delegate = self;
        [layerImage addAnimation:animation forKey:@"BoundsAnimation"];

        // animate the position so it stays at 0, 0 of the frame.
        animation = [CABasicAnimation animationWithKeyPath:@"position"];
        animation.duration = [UIView animationDuration];
        animation.fromValue = [NSValue valueWithCGPoint:layerImage.position];
        animation.toValue = [NSValue valueWithCGPoint:CGPointMake(bounds.size.width / 2, bounds.size.height / 2)];
        animation.removedOnCompletion = NO;
        animation.fillMode = kCAFillModeForwards;
        animation.timingFunction = [UIView animationFunction];
        animation.delegate = self;
        [layerImage addAnimation:animation forKey:@"PositionAnimation"];
    } else {
        layerImage.frame = bounds;
    }

    [super setFrame:value];
}
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...