UIViewPropertyAnimator在呈现另一个控制器后停止工作

时间:2018-04-09 16:09:50

标签: ios objective-c ios11 uiviewpropertyanimator

我使用UIViewPropertyAnimator缩小视图控制器的标题:

- (UIViewPropertyAnimator *)headerAnimator {
    if (!_headerAnimator) {
        _headerAnimator = [[UIViewPropertyAnimator alloc] initWithDuration:2.f curve:UIViewAnimationCurveLinear animations:^{
            self.profileImageBorderWidthConstraint.constant = 58.f;
            self.profileImageBorderHeightConstraint.constant = 58.f;
            self.profileImageBorder.layer.cornerRadius = 29.f;

            self.profileImageView.layer.cornerRadius = 25.f;

            self.nameLabelTopConstraint.constant = 16.f;

            [self.headerView layoutIfNeeded];
        }];
    }

    return _headerAnimator;
}

在滚动时,我设置了fragment

- (void)scrollViewDidScroll:(UIScrollView *)scrollView {
    CGFloat fraction = (scrollView.contentOffset.y / (HeaderHeight-CollapsedHeaderHeight));
    self.headerAnimator.fractionComplete = fraction;
}

这很好用。但是在我从此控制器中呈现另一个UIViewController并将其关闭后,动画师将处于非活动状态并停止响应分数设置。我该如何重新启动它?

我已经尝试过:

  • 将我的动画制作者属性设置为nil,让我们下次重新创建它(崩溃)
  • 暂停-viewWillAppear:以使其变为活动状态(变为活动状态,但不会设置动画)
  • -viewWillAppear:上启动(保持不活动状态)

1 个答案:

答案 0 :(得分:2)

到目前为止,我找到了iOS 11+的完美解决方案以及早期操作系统的解决方法。

iOS 11:您必须将pausesOnCompletion设置为YES。即使在控制器被解除后,这也会使动画处于活动状态。

对于早期版本,我不得不离开UIViewPropertyAnimator并创建一个自己的函数,我可以在其中更改动画分数并手动插值:

- (void)animateHeaderToScrollPoint {
    // Header
    CGFloat fraction = self.tableView.contentOffset.y / (HeaderHeight-CollapsedHeaderHeight);
    fraction = fminf(fraction, 1.f);
    fraction = fmaxf(fraction, 0.f);
    fraction = 1.f - fraction;

    self.profileImageBorderWidthConstraint.constant = 58.f + (86.f - 58.f) * fraction;
    self.profileImageBorderHeightConstraint.constant = 58.f + (86.f - 58.f) * fraction;
    self.profileImageBorder.layer.cornerRadius = 29.f + (43.f - 29.f) * fraction;

    self.profileImageView.layer.cornerRadius = 25.f + (39.f - 25.f) * fraction;

    self.nameLabelTopConstraint.constant = 16.f + (31.f - 16.f) * fraction;

    [self.headerView layoutIfNeeded];
}