·您现在的位置: 云翼网络 >> 文章中心 >> 网站建设 >> app软件开发 >> IOS开发 >> 动画技术

动画技术

作者:佚名      IOS开发编辑:admin      更新时间:2022-07-23

每次看到别人的招聘信息,好多都是要求熟练使用Core Animation(CA)。木办法呀,那就学习吧。看到那些很长的名字的类还有很长的方法等等等等,就头疼,觉得太难了。你是不是也有这种情况?

我只想说:世上无难事,只怕有心人。只要我们能够静下心认真的看看,其实也不是那么难的。不罗嗦了。开始正题:

------------------------------------------------------------------------文章比较长,做好心理准备哈------------------------------------------------------------------------

这里主要介绍以下几块:(参考《iOS图形图像、动画和多媒体编程技术最佳实践》)

1、视图动画

2、iOS7自定的视图的过渡动画

3、iOS7 UIKit力学

4、iOS 7 运动效果(Motion Effects)

5、Core Animation

废话不多说,开始第一块:

视图动画

最简单的动画:

- (void)PResentViewController:(UIViewController *)viewControllerToPresent animated: (BOOL)flag completion:(void (^)(void))completion NS_AVAILABLE_IOS(5_0);

简单把,我们用代码进行视图切换的时候都会用到这个,其中的flag就是用来开启动画的。所以平常我们都会用到动画,只不过是没有注意罢了。

下面来看个比较简单的一个动画:

先看看要实现的效果:

怎么实现的呢?看看代码:

- (IBAction)buttonAction:(id)sender {
//    //方法1
//    _myButton.alpha = 0;
//    [UIView beginAnimations:@"animation" context:nil];
//
//    [UIView setAnimationDuration:1.5];
//    [UIView setAnimationDelegate:self];
//    [UIView setAnimationDidStopSelector:@selector(showButton)];
//        _ballLabel.frame = CGRectMake(_ballLabel.frame.origin.x, _ballLabel.frame.origin.y+200*flag, _ballLabel.frame.size.width, _ballLabel.frame.size.height);
//    flag = -flag;
//    [UIView commitAnimations];

//    //方法2:没控制按钮隐藏显示
//    [UIView animateWithDuration:1.5 animations:^{
//        CGRect frame = self.ballLabel.frame;
//        frame.origin.y += 200*flag;
//        self.ballLabel.frame = frame;
//        flag = -flag;
//    }];
    //动画3:当小球动时按钮消失,当小球不动时按钮出现
     _myButton.alpha = 0;
    [UIView animateWithDuration:1.5 animations:^{
       
        CGRect frame = self.ballLabel.frame;
        frame.origin.y +=200*flag;
        self.ballLabel.frame = frame;
        flag = -flag;
    } completion:^(BOOL finished) {
        _myButton.alpha = 1;
    }];
}
- (void)showButton {
    _myButton.alpha = 1;
}

上面的三种方法都能实现相同的效果。

在iOS4之前,如果我们想获取动画开始和结束事件,我们需要设置委托对象,然后通过下面两个方法来设置:

+ (void)setAnimationWillStartSelector:(SEL)selector;                // default = NULL. -animationWillStart:(NSString *)animationID context:(void *)context
+ (void)setAnimationDidStopSelector:(SEL)selector;  

第一个是动画刚开始要触发的动作,第二个市动画结束时要触发的动作。

后来iOS升级后可以在回调中进行处理了。就像动画3一样。

如果针对我们平时用到的简单动画,用UIView这种就可以了。

简单的动画就这样完了????里面的额函数什么意思呢???那就看一下最重要的两个吧。

+ (void)beginAnimations:(NSString *)animationID context:(void *)context;

标记动画开始(或执行)块的开头。

+ (void)commitAnimations;

标记动画开始(执行)块的结束,并且安排动画执行。

我们可以在中间写上一些自定义的东西。比如我们需要设置动画重复次数,开始时间,或者其他的。

------------------------------------------------应该能够接受吧,耐心看看------------------------------------------------

看看第三种方法:

+ (void)animateWithDuration:(NSTimeInterval)duration animations:(void (^)(void))animations completion:(void (^)(BOOL finished))completion NS_AVAILABLE_IOS(4_0);

这个方法高度封装,你可以不用上面说的第一种方法来标记开始和结束了。直接用这个方法。

duration:动画执行的时间。

animations:后面的block是动画执行的内容。

completion:后面的block是当动画结束时的操作。

个人建议还是使用下面那种方法,毕竟新版本的东西。

累么?不累看看下面的过渡动画。

过渡动画就是两个视图进行切换的时候的动画。先看看效果:

 

直接看看代码:

/**
 动画还有很多属性:例如动画曲线、过渡(界面跳转)动画、重复次数和自动反转等。
 */
- (IBAction)transitionAction1:(UIButton *)sender {
    [UIView beginAnimations:@"animagtionID" context:nil];
    [UIView setAnimationDuration:1.0];
    [UIView setAnimationCurve:UIViewAnimationCurveEaseIn];  //设置动画曲线:这里有四种UIViewAnimationCurveEaseInOut:缓入缓出。设置出场或者入场的速度。
    [UIView setAnimationRepeatAutoreverses:NO];   //如果设置成yes,那么会执行在返回。
    switch (sender.tag) {
        case 1:
            [UIView setAnimationTransition:UIViewAnimationTransitionCurlUp forView:self.view cache:YES];
            break;
        case 2:
            [UIView setAnimationTransition:UIViewAnimationTransitionFlipFromLeft forView:self.view cache:YES];
            break;
        case 3:
            [UIView setAnimationTransition:UIViewAnimationTransitionCurlDown forView:self.view cache:YES];
            break;
        case 4:
            [UIView setAnimationTransition:UIViewAnimationTransitionFlipFromRight forView:self.view cache:YES];
            break;
        default:
            break;
    }
    [UIView commitAnimations];
    //在ios4后添加了+transitionWithView:duration:options:animations:completion:指定的视图容器内创建动画过渡。+transitionFromView:toView:duration:options:completion:在指定的两个视图之间创建动画过渡。
    
}

其实也很简单,只不过在原来简单动画的基础上添加了一个:

+ (void)setAnimationTransition:(UIViewAnimationTransition)transition forView:(UIView *)view cache:(BOOL)cache;

typedef NS_ENUM(NSInteger, UIViewAnimationTransition) {
    UIViewAnimationTransitionNone,
    UIViewAnimationTransitionFlipFromLeft,
    UIViewAnimationTransitionFlipFromRight,
    UIViewAnimationTransitionCurlUp,
    UIViewAnimationTransitionCurlDown,
};

这就是UIViewAnimationTransition的枚举。里面就是这几种类型。感兴趣可以自己试试。

----------------------------------------------------------------------打断一下----------------------------------------------------------------------

transition:过渡;转变;转换;变调的意思。

----------------------------------------------------------------------打断结束----------------------------------------------------------------------

所以遇到animationTransition不要觉得它很难以理解:就是动画过渡的意思。

 

接下来的iOS7自定义动画只是简单介绍一下吧。因为。。。。。。。。。。。。。。

视图过渡有两种情况:树形结构导航和模态导航。

树形的就类似与表格点击cell进入另一个视图。是通过UINavigationController控制器堆栈实现的视图过渡。

模态导航:是通过UIViewController控制器实现的。

树形的自定义动画需要涉及两个协议:UIViewControllerAnimatedTransitioning和UINavigationAnimatedTransitioning协议。

@protocol UIViewControllerAnimatedTransitioning <NSObject>

// This is used for percent driven interactive transitions, as well as for container controllers that have companion animations that might need to
// synchronize with the main animation. 
- (NSTimeInterval)transitionDuration:(id <UIViewControllerContextTransitioning>)transitionContext;
// This method can only  be a nop if the transition is interactive and not a percentDriven interactive transition.
- (void)animateTransition:(id <UIViewControllerContextTransitioning>)transitionContext;


@optional

这是UIViewControllerAnimatedTransitioning的协议内容。要想去自定义动画就去实现这两个代理方法。

我只知道下面两句:

  UIViewController *fromVC = [transitionContext viewControllerForKey:UITransitionContextFromViewControllerKey];
    UIViewController *toVC = [transitionContext viewControllerForKey:UITransitionContextToViewControllerKey];

通过transitionContext可以得到从哪个视图的那个视图和到哪个视图的那个视图。。。(有点绕,,,)

然后剩下的多查看一些资料学习把。

模态导航自定义过渡动画需要在视图控制器实现UIViewControllerTransitioningDelegate协议。

@protocol UIViewControllerTransitioningDelegate <NSObject>

@optional
- (id <UIViewControllerAnimatedTransitioning>)animationControllerForPresentedController:(UIViewController *)presented presentingController:(UIViewController *)presenting sourceController:(UIViewController *)source;

- (id <UIViewControllerAnimatedTransitioning>)animationControllerForDismissedController:(UIViewController *)dismissed;

- (id <UIViewControllerInteractiveTransitioning>)interactionControllerForPresentation:(id <UIViewControllerAnimatedTransitioning>)animator;

- (id <UIViewControllerInteractiveTransitioning>)interactionControllerForDismissal:(id <UIViewControllerAnimatedTransitioning>)animator;

- (UIPresentationController *)presentationControllerForPresentedViewController:(UIViewController *)presented presentingViewController:(UIViewController *)presenting sourceViewController:(UIViewController *)source NS_AVAILABLE_IOS(8_0);

@end

这个是协议内容。具体的实现自定义模态视图就是在这些代理方法中设置的。

------------------------------------------------------------------------------UIKit力学来了------------------------------------------------------------------------------

先复习一下英语:

dynamics:动力学,力学。

attachment:附件;依恋

collision:碰撞,冲突

gravity:重力

snap:

突然折断,拉断;猛咬;啪地关上。

开始力学学习。

UIKit力学可以使得视图对象具有真实的物理运动效果。在游戏中能够实现物理效果的技术叫物理引擎。

力学中用到的类:

UIDynamicAnimator:它可以理解为动态的动画师,它为他的动态的元素提供物理相关的能力和动画,并且提供动画环境。是用来存放要使用的力学行为(UIDynamicBehavior)。

UIDynamicBehavior有6个类:

UIAttachmentBehavior:吸附

UICollisionBehavior:碰撞

UIGravityBehavior:重力

UIPushBehavior:推

UISnapBehavior:甩

UIDynamicItemBehavior:

对于每个力学行为都有多个力学项目:(UIDynamicItem)。其中UIDynamicAnimator的ReferenceView属性指向了所要呈现的视图。

总体概括:一个UIDynamicAnimator可以包含多个力学行为,通过addBehavior方法添加力学行为。一个力学行为可以包括多个力学项目,通过addItem添加。还有UIDynamicAnimator与ReferenceView之间是1:1关系,即一个UIDynamicAnimator只有一个ReferenceView与之对应。

--------------------------------------------------下面的行为触发为了方便都是在视图加载出现后触发的--------------------------------------------------

重力行为:

- (void)viewDidLoad {
    [super viewDidLoad];
    _gravityButton.backgroundColor = [UIColor blackColor];
    _gravityButton.layer.cornerRadius = _gravityButton.frame.size.height/2;


    
}
- (void)viewDidAppear:(BOOL)animated {
    [super viewDidAppear:YES];
    animator = [[UIDynamicAnimator alloc] initWithReferenceView:self.view];   //存放要使用的力学行为
    gravity = [[UIGravityBehavior alloc] init];   //新建一个重力力学行为
    [animator addBehavior:gravity];   //将重力力学行为添加到力学行为里
    //设置重力方向
    CGVector gravityDirection = {0.0,0.5};//设置重力方向
    [gravity setGravityDirection:gravityDirection];
    [gravity addItem:_gravityButton];   //将_gravityButton对象添加到力学行为gravity对象中,其中_gravityButton是模拟重力行为的视图对象。
}

主要是我用了一个button,然后刚进入这个界面,button就会由于重力作用自由落体。效果如下:

简单介绍一下实现:

首先需要创建一个动画师(UIDynamicAnimator)用于存放要使用的重力行为。referenceView你可以理解为为动态动画师准备的视图。

然后是设置重力方向,其中的CGVector:

struct CGVector {
  CGFloat dx;
  CGFloat dy;
};

就是一个结构体而已。定义了重力的 方向。(感兴趣的可以自己调整查看效果)

然后就是把重力行为添加到animator中。

其中

 [gravity addItem:_gravityButton];   //将_gravityButton对象添加到力学行为gravity对象中,其中_gravityButton是模拟重力行为的视图对象。

添加元素你就可以理解成那个对象需要使用该力学行为,就把那个元素添加进去。

---------------------------------------------------------------重力行为结束---------------------------------------------------------------

碰撞行为:

- (void)viewDidLoad {
    [super viewDidLoad];
    _myButton.layer.cornerRadius = _myButton.frame.size.height/2;

    // Do any additional setup after loading the view.
}
//碰撞行为中特有的方法是碰撞检测,我们可以检测运动的物体和ReferenceView:[collsion setTranslatesReferenceBoundsIntoBoundary:YES];
//检测是否与其他物体边界发生碰撞的方法:
/*
 CGPoint p1 = barrier.frame.origin;
 CGPoint p2 = CGPointMake(p1*x+barrier.frame.size.width,p1.y);
 [collision addBoundaryWithdIdentifier:@"barrier" fromPoint:p1 toPoint:p2];
 */
- (void)viewDidAppear:(BOOL)animated {
    [super viewDidAppear:YES];
    animator = [[UIDynamicAnimator alloc] initWithReferenceView:self.view];
    garvity = [[UIGravityBehavior alloc] initWithItems:@[_myButton]];
    CGVector vector = {0.0,1.9};
    [garvity setGravityDirection:vector];
    [animator addBehavior:garvity];
    
    
    //碰撞行为
    collision = [[UICollisionBehavior alloc] initWithItems:@[_myButton]];
    collision.translatesReferenceBoundsIntoBoundary = YES;
    [animator addBehavior:collision];
}

效果图:

具体实现就不多说了,加了一个重力行为,然后落地后设置碰撞。

这里主要介绍亮点:

第一:检测是否与ReferenceView边界发生碰撞方法如下:

[collision setTranslatesReferenceBoundsIntoBoundary:YES];

第二:检测是否与其他物体边界发生碰撞方法如下:

 CGPoint p1 = barrier.frame.origin;
 CGPoint p2 = CGPointMake(p1*x+barrier.frame.size.width,p1.y);
 [collision addBoundaryWithdIdentifier:@"barrier" fromPoint:p1 toPoint:p2];

其中的barrier就是其他视图对象。具体可以理解一下慢慢。

---------------------------------------------------------------碰撞行为结束---------------------------------------------------------------

吸附行为:

- (void)viewDidLoad {
    [super viewDidLoad];
//    self.view.backgroundColor = []
}
- (void)viewDidAppear:(BOOL)animated {
    [super viewDidAppear:YES];
    animator = [[UIDynamicAnimator alloc] initWithReferenceView:self.view];
    
    //重力行为
    gravity = [[UIGravityBehavior alloc] initWithItems:@[_boxImageView]];
    [animator addBehavior:gravity];
    
    //碰撞行为
    collision = [[UICollisionBehavior alloc] initWithItems:@[_boxImageView]];
    [collision addBoundaryWithIdentifier:@"barrier" fromPoint:_barrier.frame.origin toPoint:CGPointMake(_barrier.frame.origin.x+_barrier.frame.size.width, _barrier.frame.origin.y)];
    
    collision.translatesReferenceBoundsIntoBoundary = YES;
    collision.collisionDelegate = self;
    [animator addBehavior:collision];
    /*
     UIDynamicItemBehavior是行为限制,
     */
    UIDynamicItemBehavior *itemBehaver = [[UIDynamicItemBehavior alloc] initWithItems:@[_boxImageView]];
    itemBehaver.elasticity = 0.5;
    [animator addBehavior:itemBehaver];
}
#pragma mark -
- (void)collisionBehavior:(UICollisionBehavior *)behavior beganContactForItem:(id<UIDynamicItem>)item withBoundaryIdentifier:(id<NSCopying>)identifier atPoint:(CGPoint)p {
    attach = [[UIAttachmentBehavior alloc] initWithItem:_attachPoint attachedToItem:_boxImageView];
    
    [animator addBehavior:attach];
}

效果如下:

其中的图片是_boxImageView。方块是_attachPoint。

这里我们需要实现UICollisionBehaviorDelegate代理方法。上面有个UIDynamicItemBehavior类。可以暂时把它理解为行为限制类。一会再说:

---------------------------------------------------------------吸附行为结束---------------------------------------------------------------

推行为:

/*
 推行为可以让视图对象朝着某个方向运行,这个推力有瞬间和持续两种方式。
 设置推行为需要考虑方向和力的大小。
 */
- (void)viewDidAppear:(BOOL)animated {
    [super viewDidAppear:YES];
    animator = [[UIDynamicAnimator alloc] initWithReferenceView:self.view];
    push = [[UIPushBehavior alloc] initWithItems:@[_box] mode:UIPushBehaviorModeContinuous];
    CGVector pushDirection = {0.5,0};
    [push setPushDirection:pushDirection];
    [push setMagnitude:5.0f];
    [animator addBehavior:push];
}

效果如下:

这里说一下:推行为有瞬间和持续两种方式。

瞬间行为:UIPushBehaviorModeInstantaneous:运动比较快。

持续行为:UIPushBehaviorModeContinuous:由慢变快

还有setMagnitude:方法是用来设置推的力度。

---------------------------------------------------------------推行为结束---------------------------------------------------------------

甩行为:

/*
 能够时物体朝着某个目标点抛出甩出。并且有瞬间加速度,由慢及快,再由快及慢,最后停止在目标点。
 */
- (void)viewDidAppear:(BOOL)animated {
    CGPoint p = [[self view] center];
    animator = [[UIDynamicAnimator alloc] initWithReferenceView:self.view];
    snap = [[UISnapBehavior alloc] initWithItem:_testButton snapToPoint:p];
    [animator addBehavior:snap];
}

效果如下:

这里没什么可说的,看一下代码就行了。大概甩的动作就是:

能够时物体朝着某个目标点抛出甩出。并且有瞬间加速度,由慢及快,再由快及慢,最后停止在目标点。

---------------------------------------------------------------甩行为结束---------------------------------------------------------------

UIDynamicItemBehavior:

它是用来设置力学行为参数的。参数包括:弹性系数、摩擦系数、阻力和密度等物理属性

density:密度:

elasticity:弹力系数0-1.0表示没有反弹,1表示完全弹性碰撞

friction:摩擦系数

resistance:阻力,CGFLOAT_MAX表示最大阻力

allowRotation:是否允许旋转。

angularResistance:角阻力,物理旋转时候,旋转方向的阻力。

---------------------------------------------------------------力学结束---------------------------------------------------------------

现在看一下:iOS7运动效果:(Motion Effects)

运动效果主要提供的API是UIInterpolatingMotionEffect类。

UIView可以调用addMotionEffect:方法来添加运动效果。看例子:

- (void)viewDidLoad {
    [super viewDidLoad];
//    [[UIapplication sharedApplication] setStatusBarHidden:YES];
    //设置图片的偏移量
    UIInterpolatingMotionEffect * imageEffe = [[UIInterpolatingMotionEffect alloc] initWithKeyPath:@"center.x" type:UIInterpolatingMotionEffectTypeTiltAlongHorizontalAxis];
    imageEffe.maximumRelativeValue = @50.0;
    imageEffe.minimumRelativeValue = @-50.0;
    [self.imageView addMotionEffect:imageEffe];
    
    
    UIInterpolatingMotionEffect *nameEffe = [[UIInterpolatingMotionEffect alloc] initWithKeyPath:@"center.x" type:UIInterpolatingMotionEffectTypeTiltAlongHorizontalAxis];
    nameEffe.maximumRelativeValue = @100.0;
    nameEffe.minimumRelativeValue = @-100.0;
    [self.nameButton addMotionEffect:nameEffe];
}

这个在模拟器无法查看,只能真机查看,当左右晃动设备的时候,里面的图片也在移动。

 

---------------------------------------------------------------------------不行了---------------------------------------------------------------------------

Core Animation下一个博客在介绍吧。因为它很重要。很重要。很重要。

添加个源码把:http://pan.baidu.com/s/1hq2HN4g