50

UIKeyboardAnimationCurveUserInfoKey一个UIViewAnimationCurve价值。如何将其转换为与参数 ofUIViewAnimationOptions一起使用的相应值?options+[UIView animateWithDuration:delay:options:animations:completion:]

// UIView.h

typedef enum {
    UIViewAnimationCurveEaseInOut,         // slow at beginning and end
    UIViewAnimationCurveEaseIn,            // slow at beginning
    UIViewAnimationCurveEaseOut,           // slow at end
    UIViewAnimationCurveLinear
} UIViewAnimationCurve;

// ...

enum {
    // ...
    UIViewAnimationOptionCurveEaseInOut            = 0 << 16, // default
    UIViewAnimationOptionCurveEaseIn               = 1 << 16,
    UIViewAnimationOptionCurveEaseOut              = 2 << 16,
    UIViewAnimationOptionCurveLinear               = 3 << 16,
    // ...
};
typedef NSUInteger UIViewAnimationOptions;

switch显然,我可以使用语句创建一个简单的类别方法,如下所示:

// UIView+AnimationOptionsWithCurve.h

@interface UIView (AnimationOptionsWithCurve)
@end

// UIView+AnimationOptionsWithCurve.m

@implementation UIView (AnimationOptionsWithCurve)

+ (UIViewAnimationOptions)animationOptionsWithCurve:(UIViewAnimationCurve)curve {
    switch (curve) {
        case UIViewAnimationCurveEaseInOut:
            return UIViewAnimationOptionCurveEaseInOut;
        case UIViewAnimationCurveEaseIn:
            return UIViewAnimationOptionCurveEaseIn;
        case UIViewAnimationCurveEaseOut:
            return UIViewAnimationOptionCurveEaseOut;
        case UIViewAnimationCurveLinear:
            return UIViewAnimationOptionCurveLinear;
    }
}

@end

但是,有没有更简单/更好的方法?

4

5 回答 5

44

您建议的类别方法是执行此操作的“正确”方法——您不一定能保证这些常量保持其值。但是,从查看它们的定义方式来看,您似乎可以这样做

animationOption = animationCurve << 16;

...如果编译器想抱怨的话,可能会先转换为 NSUInteger,然后再转换为 UIViewAnimationOptions。

于 2011-09-06T23:39:37.777 回答
36

可以说,您可以采用您的第一个解决方案并将其设置为内联函数以节省堆栈推送。这是一个非常严格的条件(常量绑定等),它应该编译成一个非常小的程序集。

编辑:每@matt,给你(Objective-C):

static inline UIViewAnimationOptions animationOptionsWithCurve(UIViewAnimationCurve curve)
{
  switch (curve) {
    case UIViewAnimationCurveEaseInOut:
        return UIViewAnimationOptionCurveEaseInOut;
    case UIViewAnimationCurveEaseIn:
        return UIViewAnimationOptionCurveEaseIn;
    case UIViewAnimationCurveEaseOut:
        return UIViewAnimationOptionCurveEaseOut;
    case UIViewAnimationCurveLinear:
        return UIViewAnimationOptionCurveLinear;
  }
}

斯威夫特 3:

extension UIViewAnimationOptions {
    init(curve: UIViewAnimationCurve) {
        switch curve {
            case .easeIn:
                self = .curveEaseIn
            case .easeOut:
                self = .curveEaseOut
            case .easeInOut:
                self = .curveEaseInOut
            case .linear:
                self = .curveLinear
        }
    }
}
于 2011-11-16T20:53:55.237 回答
19

在斯威夫特你可以做

extension UIViewAnimationCurve {
    func toOptions() -> UIViewAnimationOptions {
        return UIViewAnimationOptions(rawValue: UInt(rawValue << 16))
    }
}
于 2015-03-11T10:33:27.057 回答
11

基于开关的解决方案的一个问题是它假设永远不会传入任何选项组合。但实践表明,可能存在假设不成立的情况。我发现的一个实例是(至少在 iOS 7 上)当您获得键盘动画以动画您的内容以及键盘的出现/消失时。

如果您收听keyboardWillShow:keyboardWillHide:通知,然后获得键盘宣布它将使用的曲线,例如:

UIViewAnimationCurve curve = [userInfo[UIKeyboardAnimationCurveUserInfoKey] integerValue];

您可能会获得值 7。如果将其传递给 switch 函数/方法,您将无法获得该值的正确转换,从而导致不正确的动画行为。

Noah Witherspoon 的回答将返回正确的值。结合这两种解决方案,您可能会编写如下内容:

static inline UIViewAnimationOptions animationOptionsWithCurve(UIViewAnimationCurve curve)
{
    UIViewAnimationOptions opt = (UIViewAnimationOptions)curve;
    return opt << 16;
}

这里需要注意的是,正如 Noah 也指出的那样,如果 Apple 更改了两种类型不再对应的枚举,那么这个函数就会中断。无论如何都要使用它的原因是,基于开关的选项并非在您今天可能遇到的所有情况下都有效,而这确实有效。

于 2013-11-25T09:52:17.627 回答
5

iOS 10+
斯威夫特 5

转换UIView.AnimationCurve为的 Swift 替代方法UIView.AnimationOptions是使用UIViewPropertyAnimator(iOS 10+),它接受UIView.AnimationCurve并且比UIView.animate.

您很可能会使用UIResponder.keyboardAnimationCurveUserInfoKey,它返回一个NSNumber. 此密钥的文档是(Apple 自己的符号,不是我的):

public class let keyboardAnimationCurveUserInfoKey: String // NSNumber of NSUInteger (UIViewAnimationCurve)

使用这种方法,我们可以消除任何猜测:

if let kbTiming = notification.userInfo?[UIResponder.keyboardAnimationCurveUserInfoKey] as? NSNumber, // doc says to unwrap as NSNumber
    let timing = UIView.AnimationCurve.RawValue(exactly: kbTiming), // takes an NSNumber
    let curve = UIView.AnimationCurve(rawValue: timing) { // takes a raw value
    let animator = UIViewPropertyAnimator(duration: duration, curve: curve) {
        // add animations
    }
    animator.startAnimation()
}
于 2020-06-18T16:36:53.113 回答