FaceBook POP(PopAnimation)介绍与使用

掘金主页欢迎关注和提问,以后也会不定期回答简书上的提问

本文简单介绍了FaceBook开原动画框架POP的内容及基本使用
Demo下载地址:https://github.com/iOSAppleBea/PopAnimationDemo

推荐一份参考文献《iOS核心动画高级技巧》:https://zsisme.gitbooks.io/ios-/content/chapter1/layers-and-trees.html

使用POP可以创建的 4 类动画:

  • Spring (弹性)动效可以赋予物体愉悦的弹性效果;
  • Decay (衰减) 动效可以用来逐渐减慢物体的速度至停止;
  • Basic (基本)传统动画,可以在给定时间的运动中插入数值调整运动节奏,支持默认、线性、淡入、淡出、淡入淡出动画;
  • Custom (自定义)自定义动画,让设计值创建自定义动效,只需简单处理CADisplayLink,并联系时间-运动关系,使用难度略大于前三者;

它们的继承关系为

NSObject   
|- POPAnimation   
    |- POPPropertyAnimation   
        |- POPBasicAnimation   
        |- POPDecayAnimation   
        |- POPSpringAnimation   
    |- POPCustomAnimation   

所有的动画属性分为两大类:作用于View与作用于Layer

View系:
    不透明度    - kPOPViewAlpha
    颜色  - kPOPViewBackgroundColor
    大小  - kPOPViewBounds
    中心点 - kPOPViewCenter
    位置与尺寸   - kPOPViewFrame
    尺寸 - kPOPViewScaleXY
    尺寸(按比例变化) - kPOPViewSize

Layer系:
    颜色 - kPOPLayerBackgroundColor
    大小 - kPOPLayerBounds、kPOPLayerScaleXY、kPOPLayerSize
    不透明度 - kPOPLayerOpacity
    位置 - kPOPLayerPosition
    X 坐标 - kPOPLayerPositionX
    Y 坐标 - kPOPLayerPositionY
    旋转 - kPOPLayerRotation

POP动画常用属性

这里简单介绍几个使用过程中常用的属性

  • beginTime :动画开始时间,一般用animation.beginTime = CACurrentMediaTime() + 0.5;动画延迟0.5秒后执行;
  • duration :动画持续时长,多用于基础动画;
  • timingFunction :决定动画节奏;
  • fromValue :动画的起始状态;
  • toValue :动画的终止状态;
  • springBounciness :弹簧弹力(幅度) 取值范围为[0, 20],默认值为4;
  • springSpeed :弹簧速度,速度越快,动画时间越短 [0, 20],默认为12,和springBounciness一起决定着弹簧动画的效果;
  • velocity :速率,常用于衰减动画中,速率被用来计算运行的距离;
  • deceleration :负加速度,Default = 0.998,如果你开发给火星人用,那么这个值使用 0.376 会更合适。减速效果增强(缩小动画幅度)<< 0.998 << 减速效果降低(加大动画幅度)
  • completionBlock :动画完成后的回调,completionBlock 提供了一个 Callback,动画的执行过程会不断调用这个 block,finished 这个布尔变量可以用来做动画完成与否的判断;
    附:
        · dynamicsTension 弹簧的张力,影响回弹力度及速度;
        · dynamicsFriction 弹簧摩擦力,开启后,动画会不断重复,并且幅度逐渐削弱,直到停止;
        · dynamicsMass 质量,细微地影响动画的回弹力度和速度;
        这三者可以从更细的粒度上替代springBounciness和springSpeed控制弹簧动画的效果。

POP动画的介绍与使用

1. POPBasicAnimation基础动画

基本动画,接口方面和CABasicAniamtion很相似,使用可以提供初始值fromValue,这个 终点值toValue,动画时长duration以及决定动画节奏的timingFunction。timingFunction直接使用的CAMediaTimingFunction,是使用一个横向纵向都为一个单位的拥有两个控制点的贝赛尔曲线来描述的,横坐标为时间,纵坐标为动画进度。

    POPBasicAnimation *basicAnimation = [POPBasicAnimation  animationWithPropertyNamed:kPOPViewScaleXY];
    basicAnimation.toValue = [NSValue valueWithCGSize:CGSizeMake(3.0, 3.0)];
    basicAnimation.duration = 1;
    basicAnimation.timingFunction = [CAMediaTimingFunction functionWithName:kCAMediaTimingFunctionEaseInEaseOut];
    [self.ZView pop_addAnimation: basicAnimation forKey:@"basicAnimation"];

对timingFunction的设置也可以使用另外一种方法

    POPBasicAnimation *offscreenAnimation = [POPBasicAnimation easeInAnimation];
    offscreenAnimation.property = [POPAnimatableProperty propertyWithName:kPOPLayerPositionX];
    ...

此处需要说明的:p
- (void)pop_addAnimation:(POPAnimation *)anim forKey:(NSString *)key方法中Key的赋值,保证在动画载体中Key的唯一性即可;

2. PopSpringAnimation

弹簧动画是Bezier曲线无法表述的,所以无法使用PopBasicAniamtion来实现。PopSpringAnimation便是专门用来实现弹簧动画的。

    POPSpringAnimation *springAnimation = [POPSpringAnimation animationWithPropertyNamed:kPOPLayerScaleXY];
    springAnimation.beginTime = CACurrentMediaTime() + 2;
    springAnimation.toValue = [NSValue valueWithCGSize:CGSizeMake(2.0, 2.0)];
    springAnimation.springBounciness = 5.f;
    springAnimation.springSpeed = 10;
    springAnimation.completionBlock = ^(POPAnimation *anim, BOOL finished) {
        POPSpringAnimation *resetAnimation = [POPSpringAnimation animationWithPropertyNamed:kPOPLayerScaleXY];
        resetAnimation.toValue = [NSValue valueWithCGSize:CGSizeMake(1.0, 1.0)];
        resetAnimation.springBounciness = 20.f;
        resetAnimation.springSpeed = 10;
        [self.ZView.layer pop_addAnimation:scaleAnimation forKey:@"resetAnimation"];
    };
    [self.ZView.layer pop_addAnimation: springAnimation forKey:@"springAnimation"];

如果你想让视图在原地抖动,产生吸引用户的效果,那么给velocity属性一个初值,并设置其弹簧洗漱 类似下面这样:

POPSpringAnimation *positionAnimation = [POPSpringAnimation animationWithPropertyNamed:kPOPLayerPositionX];
    positionAnimation.beginTime = CACurrentMediaTime() + 0.1;
    positionAnimation.velocity = @2000;
    positionAnimation.springBounciness = 20;

视图就会奇妙的一阵“抽搐”,随后这个动画将变得索然无味😎哈哈哈

3. PopDecayAnimation

基于Bezier曲线的timingFuntion同样无法表述Decay Aniamtion,所以Pop就单独实现了一个 PopDecayAnimation,用于衰减动画。

    POPDecayAnimation *decay_1 = [POPDecayAnimation animationWithPropertyNamed:kPOPLayerBounds];
    decay_1.velocity = [NSValue valueWithCGRect:CGRectMake(0, 0, 50.0, 50.0)];
    [_ZButton.layer pop_addAnimation:decay_1 forKey:@"lalalal"];
    
    POPDecayAnimation *decay_2 = [POPDecayAnimation animationWithPropertyNamed:kPOPLayerPosition];
    decay_2.velocity = [NSValue valueWithCGPoint:CGPointMake(100, 500)];
    [_Z2Button.layer pop_addAnimation:decay_2 forKey:@"lalalal"];

衰减动画没有 toValue 只有 fromValue,然后按照 velocity 来做衰减操作,如果不对fromValue赋值,那么动画会按照载体的当前状态执行。这里非常值得一提的是,velocity 也是必须和你操作的属性有相同的结构,如果你操作的是 bounds,想实现一个水滴滴到桌面的扩散效果,那么应该是 [NSValue valueWithCGRect:CGRectMake(0, 0,20.0, 20.0)]。

动画委托

POPAnimatorDelegate是可选的,拥有如下方法

- (void)pop_animationDidStart:(POPAnimation *)anim;
- (void)pop_animationDidStop:(POPAnimation *)anim finished:(BOOL)finished;
- (void)pop_animationDidReachToValue:(POPAnimation *)anim;

值得说的几点

  • 最后我们使用 pop_addAnimation 来让动画开始生效,如果你想删除动画的话,那么你需要调用 pop_removeAllAnimations。 与 iOS 自带的动画不同,如果你在动画的执行过程中删除了物体的动画,那么物体会停在动画状态的最后一个瞬间,而不是闪回开始前的状态;
  • Pop Animation应用于CALayer时,在动画运行的任何时刻,layer和其presentationLayer的相关属性值始终保持一致,而Core Animation做不到
  • 如果期望试图始终保持动画后的试图状态,那么请不要对该试图或其父仕途进行任何试图元素上的增删,此类操作会导致使用过PopAnimation的试图无法保持动画的终了状态。
  • 位置类的Animation,类似X/Y的toValue赋值,是相对于父试图的坐标,而且,其坐标是被操作试图的锚点。
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,547评论 6 477
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,399评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,428评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,599评论 1 274
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,612评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,577评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,941评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,603评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,852评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,605评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,693评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,375评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,955评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,936评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,172评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 43,970评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,414评论 2 342

推荐阅读更多精彩内容

  • 当听闻Facebook要开源自己的Animation框架的时候,我还以为是基于Core Animation进行的封...
    韩七夏阅读 1,069评论 0 0
  • pop支持4种动画类型:弹簧动画效果、衰减动画效果、基本动画效果和自定义动画效果。 弹簧动画效果 1.效果图如下:...
    happyte阅读 3,283评论 0 13
  • 目录 ** UIView 动画 ** ** Core Animation ** ** FaceBook POP动画...
    方向_4d0d阅读 1,580评论 0 3
  • Workflow Workflow是iOS上一款优秀的的工作流应用,过去¥18的价格让我望而却步,原因是我不确定能...
    LukeInside阅读 991评论 0 1
  • 8天愉快的寒假国学班,让溪溪对中国传统文化有了兴趣。每天上完后回来就叽叽喳喳现学现卖了。感谢程老师,让孩子...
    笑青阅读 251评论 0 0