之前介绍CCNode的时候说过,动作是指在特定时间内完成移动、缩放、旋转等操作的行为,节点可以通过运行动作来实现动画效果,这里的动作就是指CCAction对象,它有很多的子类,每个子类都封装了不同的动作效果。

先来看看CCAction的继承结构图

这里我省略了瞬时动作(CCActionInstant)和间隔动作(CCActionInterval)的子类,因为它们又包含了非常多的子类,待会再介绍它们的子类。

CCAction和CCFiniteTimeAction都是抽象类,只定义了一些基本属性和方法,没有实际用途,我们需要使用它们的子类来初始化动作,然后运行到节点上。

间隔动作

间隔动作就是指需要经过一段时间才能完成的动作,所有的间隔动作都继承自CCActionInterval。比如CCRotateBy,可以在指定时间内旋转指定的角度

  1. // 1秒内顺时针旋转360°
  2. CCRotateBy *rotate = [CCRotateBy actionWithDuration:1 angle:360];
  3. [sprite runAction:rotate];

间隔动作的继承结构图:(只列出部分常用的)

由于CCActionInterval的子类太多了,这里就不一一介绍了,可以查阅下API文档

下面列举一下常见的CCActionInterval的子类:

1.CCBlink

闪烁效果

  1. // 5秒内闪烁20次
  2. CCBlink *blink = [CCBlink actionWithDuration:5 blinks:20];
  3. [sprite runAction:blink];

2.CCMoveBy和CCMoveTo

CCMoveBy是移动一段固定的距离,CCMoveTo是移动到指定的位置

  1. // 在1秒内,向右移动100单位,同时向上移动80单位
  2. CCMoveBy *moveBy = [CCMoveBy actionWithDuration:1 position:CGPointMake(100, 80)];
  1. // 在1秒内,从节点的当前位置移动到(100, 80)这个位置
  2. CCMoveTo *moveTo = [CCMoveTo actionWithDuration:1 position:CGPointMake(100, 80)];

3.CCRotateBy和CCRotateTo

CCRotateBy是在当前旋转角度的基础上再旋转固定的角度,CCRotateTo是从当前旋转角度旋转到指定的角度

假设精灵在初始化的时候已经顺时针旋转了45°

  1. sprite.rotation = 45;

如果使用了CCRotateBy

  1. CCRotateBy *rotateBy = [CCRotateBy actionWithDuration:1 angle:90];
  2. [sprite runAction:rotateBy];
  3. // 在1秒内,再顺时针旋转90°,那么sprite的最终旋转角度是45° + 90° = 135°

如果使用了CCRotateTo

  1. CCRotateTo *rotateTo = [CCRotateTo actionWithDuration:1 angle:90];
  2. [sprite runAction:rotateTo];
  3. // 在1秒内,顺时针旋转到90°,sprite的最终旋转角度就是90°

4.CCScaleBy和CCScaleTo

CCScaleBy是在当前缩放比例的基础上再缩放固定的比例,CCScaleTo是从当前缩放比例缩放到指定的比例

假设精灵在初始化的时候的缩放比例为0.8

  1. sprite.scale = 0.8;

如果使用了CCScaleBy

  1. CCScaleBy *scaleBy = [CCScaleBy actionWithDuration:1 scale:0.5];
  2. [sprite runAction:scaleBy];
  3. // 在1秒内,宽度和高度再缩小50%,那么sprite最终缩放比例是0.8 * 0.5 = 0.4

如果使用了CCScaleTo

  1. CCScaleTo *scaleTo = [CCScaleTo actionWithDuration:1 scale:0.5];
  2. [sprite runAction:scaleTo];
  3. // 在1秒内,宽度和高度缩小为0.5倍,那么sprite最终缩放比例是就0.5

5.CCFadeIn和CCFadeOut和CCFadeTo

CCFadeIn是淡入,即由暗转亮,从没有到有;CCFadeOut是淡出,即由亮转暗,从有到没有;CCFadeTo用来修改节点的不透明度

  1. // 在2秒内,从没有到有
  2. CCFadeIn *fadeIn = [CCFadeIn actionWithDuration:2];
  3. // 在2s内,从有到没有
  4. CCFadeOut *fadeOut = [CCFadeOut actionWithDuration:2];
  5. // 在2s内,不透明度变为120,即变为半透明 (不透明度取值范围是0-255)
  6. CCFadeTo *fadeTo = [CCFadeTo actionWithDuration:2 opacity:120];

6.CCRepeat

重复执行某个动作,可以指定重复的次数

  1. // 1秒中顺时针旋转360°
  2. CCRotateBy *rotateBy = [CCRotateBy actionWithDuration:1 angle:360];
  3. // 重复执行2次旋转动画
  4. CCRepeat *repeat = [CCRepeat actionWithAction:rotateBy times:2];
  5. [sprite runAction:repeat];

7.CCAnimate

按顺序地播放图片,可以实现帧动画。

例如有下面10张图片:(玩过街机的同学应该很熟悉,赵云的大鹏展翅)

不难发现,如果从1.png 到 10.png按顺序显示图片的话会形成一个动画

下面用CCAnimate实现动画效果

  1. // 用来存放所有的帧
  2. NSMutableArray *frames = [NSMutableArray array];
  3. // 加载所有的图片
  4. for (int i = 1; i<= 10; i++) {
  5. // 文件名
  6. NSString *name = [NSString stringWithFormat:@"zy.bundle/%i.png", i];
  7. // 根据图片名加载纹理,一个图片对应一个纹理对象
  8. CCTexture2D *texture = [[CCTextureCache sharedTextureCache] addImage:name];
  9. // 根据纹理初始化一个帧
  10. CGRect retct = CGRectMake(0, 0, texture.contentSize.width, texture.contentSize.height);
  11. CCSpriteFrame *frame = [[[CCSpriteFrame alloc] initWithTexture:texture rect:retct] autorelease];
  12. // 添加帧到数组中
  13. [frames addObject:frame];
  14. }
  15. // 根据帧数组初始化CCAnimation,每隔0.1秒播放下一张图片
  16. CCAnimation *animation = [CCAnimation animationWithFrames:frames delay:0.1];
  17. // 根据CCAnimation对象初始化动作
  18. CCAnimate *animate = [CCAnimate actionWithAnimation:animation];
  19. [sprite runAction:animate];

动画效果如下:

这里是将10帧分为10张不同的png图片。为了性能着想,其实最好将10帧打包成一个图片,到时从这张图片上面切割每一帧,这种图片我们可以称为"纹理相册",可以用TexturePacker制作纹理相册

8.CCSequence

一般情况下,如果给节点同时添加几个动作时,它们会同时运行。比如下面的代码效果是一边旋转一边缩放

  1. CCRotateBy *rotateBy = [CCRotateBy actionWithDuration:1 angle:360];
  2. CCScaleBy *scaleBy = [CCScaleBy actionWithDuration:1 scale:2];
  3. [sprite runAction:rotateBy];
  4. [sprite runAction:scaleBy];

但是有时候我们想让动作一个接着一个运行,那么就要用到CCSequence
下面演示的效果是,让精灵先变为红色,再从红色变为绿色,再从绿色变为蓝色

  1. CCTintTo *tintTo1 = [CCTintTo actionWithDuration:1 red:255 green:0 blue:0];
  2. CCTintTo *tintTo2 = [CCTintTo actionWithDuration:1 red:0 green:255 blue:0];
  3. CCTintTo *tintTo3 = [CCTintTo actionWithDuration:1 red:0 green:0 blue:255];
  4. // CCTintTo也是CCActionInterval的子类,可以用于更改精灵的颜色。
  5. CCSequence *sequence = [CCSequence actions:tintTo1, tintTo2, tintTo3, nil];
  6. [sprite runAction:sequence];

CCSequence会按顺序执行参数中传入的所有动作

9.CCActionEase

当对节点使用CCMoveTo动作时,它会匀速移动到目的地,如果使用CCActionEase就可以使节点由慢到快或者由快到慢地移向目的地。因此CCActionEase是用来改变动作的运行时速度的。

CCActionEase是个非常强大的类,子类非常多,这里只说明其中的几个:

CCEaseIn:由慢到快

CCEaseOut:由快到慢

CCEaseInOut:先由慢到快,再由快到慢

举个代码例子:

  1. CCMoveTo *moveTo = [CCMoveTo actionWithDuration:4 position:ccp(300, 200)];
  2. CCEaseInOut *easeInOut = [CCEaseInOut actionWithAction:moveTo rate:5];
  3. [sprite runAction:easeInOut];

你会看到精灵先由慢到快,再由快到慢。rate参数决定了速率变化的明显程度,当它大于1时才有效

10.CCGridAction

使用CCGridAction的具体子类可以实现三维效果,例如翻页效果(CCPageTurn3D)、波浪效果(CCWaves)、流体效果(CCLiquid)。虽然能实现很好看的3D效果,但是它有很大的缺点,如果不启用深度缓冲,3D效果会有些失真,如果启用了深度缓冲,会特别耗内存。

如果想开启深度缓冲的话,就要修改EAGLView的初始化参数depthFormat:

  1. EAGLView *glView = [EAGLView viewWithFrame:[window bounds]
  2. pixelFormat:kEAGLColorFormatRGB565
  3. depthFormat:GL_DEPTH_COMPONENT24_OES];

可以改为GL_DEPTH_COMPONENT16_OES(16位深度缓冲)或者GL_DEPTH_COMPONENT24_OES(24位深度缓冲),16位深度缓冲占用的内存较少,但是仍然会有些失真

瞬时动作

瞬时动作(CCActionInstant)是指能够瞬间完成的动作,可用于改变节点位置、翻转节点形成镜像、设置节点的可视性等。

下面大致看下瞬时动作的继承结构图:

看完这个图,你可能觉得CCActionInstant好像没有一点使用价值,因为它的好多动作都可以通过修改节点属性来完成。比如可以通过设置节点的visible属性来代替使用CCShow\CCHide\CCToggleVisibility、可以通过修改节点的position属性来代替使用CCPlace。其实当它们与CCSequence结合使用时才有价值。比如,我们想先让节点运行一个CCMoveTo移动到某个位置,移动完毕后再隐藏节点,这时候我们就可以将CCMoveTo、CCHide两个动作按顺序放进CCSequence中达到想要的效果。

  1. // 移动到(300, 200)
  2. CCMoveTo *moveTo = [CCMoveTo actionWithDuration:2 position:ccp(300, 200)];
  3. // 隐藏节点
  4. CCHide *hide = [CCHide action];
  5. CCSequence *sequence = [CCSequence actions:moveTo, hide, nil];
  6. [sprite runAction:sequence];

有时候,在一个动作序列(CCSequence)里面,我们需要在一个动作运行完毕后,调用某个方法执行一些操作,然后再执行下一个动作。那我们就可以结合CCCallFunc和CCSequence完成这个功能。

比如,我们让精灵先变为红色,再从红色变为绿色,再从绿色变为蓝色,而且在每次变换颜色后都调用某个方法执行一些操作:

  1. // 变为红色
  2. CCTintTo *tintTo1 = [CCTintTo actionWithDuration:2 red:255 green:0 blue:0];
  3. // 变为红色后调用self的turnRed方法
  4. CCCallFunc *fn1 = [CCCallFunc actionWithTarget:self selector:@selector(turnRed)];
  5. // 变为绿色
  6. CCTintTo *tintTo2 = [CCTintTo actionWithDuration:2 red:0 green:255 blue:0];
  7. // 变为绿色后调用self的turnGreen:方法,参数是运行当前动作的节点
  8. CCCallFuncN *fn2 = [CCCallFuncN actionWithTarget:self selector:@selector(turnGreen:)];
  9. // 变为蓝色
  10. CCTintTo *tintTo3 = [CCTintTo actionWithDuration:2 red:0 green:0 blue:255];
  11. // 变为蓝色后调用self的turnBlue:data:方法,第一个参数是运行当前动作的节点,第二个参数是data的值
  12. CCCallFuncND *fn3 = [CCCallFuncND actionWithTarget:self selector:@selector(turnBlue:data:) data:@"blue"];
  13. // 最后调用turnDone:方法,传递了一个@"done"字符串作为参数
  14. CCCallFuncO *fn4 = [CCCallFuncO actionWithTarget:self selector:@selector(turnDone:) object:@"done"];
  15. CCSequence *sequence = [CCSequence actions:tintTo1, fn1, tintTo2, fn2, tintTo3, fn3, fn4, nil];
  16. [sprite runAction:sequence];

下面是回调方法的实现:

  1. - (void)turnRed {
  2. NSLog(@"变为红色");
  3. }
  4. // node是运行当前动作的节点
  5. - (void)turnGreen:(id)node {
  6. NSLog(@"变为绿色:%@", node);
  7. }
  8. // node是运行当前动作的节点
  9. - (void)turnBlue:(id)node data:(void *)data {
  10. NSLog(@"变为蓝色,%@,%@", node, data);
  11. }
  12. - (void)turnDone:(id)param {
  13. NSLog(@"变换完毕:%@", param);
  14. }

你会发现,精灵变为红色后就会调用turnRed方法,变为绿色后会调用turnGreen:方法,变为蓝色后会先调用turnBlue:data:方法,最后调用turnDone:方法

最后做一个总结:

下面这几个类都会在运行动作时调用一个方法

CCCallFunc :调用方法时不传递参数

CCCallFuncN :调用方法时,可以传递1个参数,参数值是运行当前动作的节点

CCCallFuncND :调用方法时,可以传递2个参数,第1个参数是运行当前动作的节点,第2个参数是actionWithTarget:selector:data:方法中的data值

CCCallFuncO :调用方法时,可以传递1个参数,参数值是actionWithTarget:selector:object:方法中的object值

CCRepeatForever

CCRepeatForever直接继承自CCAction,可以不停地运行某个间隔动作(CCActionInterval)。

如果你想让精灵不停地旋转,可以这样写:

  1. // 1秒内顺时针旋转360°
  2. CCRotateBy *rotate = [CCRotateBy actionWithDuration:1 angle:360];
  3. // 使用CCRepeatForever重复CCRotateBy动作
  4. CCRepeatForever *repeat = [CCRepeatForever actionWithAction:rotate];
  5. [sprite runAction:repeat];

也能够利用CCRepeatForever重复一个动作序列(CCSequence)

  1. // 变为红色
  2. CCTintTo *tintTo1 = [CCTintTo actionWithDuration:1 red:255 green:0 blue:0];
  3. // 变为绿色
  4. CCTintTo *tintTo2 = [CCTintTo actionWithDuration:1 red:0 green:255 blue:0];
  5. // 变为蓝色
  6. CCTintTo *tintTo3 = [CCTintTo actionWithDuration:1 red:0 green:0 blue:255];
  7. CCSequence *sequence = [CCSequence actions:tintTo1, tintTo2, tintTo3, nil];
  8. CCRepeatForever *repeat = [CCRepeatForever actionWithAction:sequence];
  9. [sprite runAction:repeat];

你会发现精灵的颜色状态是:红 -> 绿 -> 蓝 -> 红 -> 绿 -> 蓝 -> 红 ...,一直在红绿蓝3种颜色之间按顺序切换

CCSpeed

CCSpeed也是直接继承自CCAction,可以影响间隔动作(继承自CCActionInterval的动作)的运行速度。

  1. // 1秒内顺时针旋转360°
  2. CCRotateBy *rotate = [CCRotateBy actionWithDuration:1 angle:360];
  3. // 速度变为原来的一半
  4. CCSpeed *speed = [CCSpeed actionWithAction:rotate speed:0.5];
  5. [sprite runAction:speed];

本来1秒就完成旋转的,设置speed为0.5后,就需要2秒才能完成旋转

九、CCAction的更多相关文章

  1. cocos2D(九)---- CCAction

    之前介绍CCNode的时候说过,动作是指在特定时间内完毕移动.缩放.旋转等操作的行为,节点能够通过执行动作来实现动画效果,这里的动作就是指CCAction对象,它有非常多的子类,每一个子类都封装了不同 ...

  2. 如何一步一步用DDD设计一个电商网站(九)—— 小心陷入值对象持久化的坑

    阅读目录 前言 场景1的思考 场景2的思考 避坑方式 实践 结语 一.前言 在上一篇中(如何一步一步用DDD设计一个电商网站(八)—— 会员价的集成),有一行注释的代码: public interfa ...

  3. 谈谈一些有趣的CSS题目(九)-- 巧妙的实现 CSS 斜线

    开本系列,谈谈一些有趣的 CSS 题目,题目类型天马行空,想到什么说什么,不仅为了拓宽一下解决问题的思路,更涉及一些容易忽视的 CSS 细节. 解题不考虑兼容性,题目天马行空,想到什么说什么,如果解题 ...

  4. CRL快速开发框架系列教程九(导入/导出数据)

    本系列目录 CRL快速开发框架系列教程一(Code First数据表不需再关心) CRL快速开发框架系列教程二(基于Lambda表达式查询) CRL快速开发框架系列教程三(更新数据) CRL快速开发框 ...

  5. Python(九)Tornado web 框架

    一.简介 Tornado 是 FriendFeed 使用的可扩展的非阻塞式 web 服务器及其相关工具的开源版本.这个 Web 框架看起来有些像web.py 或者 Google 的 webapp,不过 ...

  6. 我的MYSQL学习心得(九) 索引

    我的MYSQL学习心得(九) 索引 我的MYSQL学习心得(一) 简单语法 我的MYSQL学习心得(二) 数据类型宽度 我的MYSQL学习心得(三) 查看字段长度 我的MYSQL学习心得(四) 数据类 ...

  7. 【Oracle 集群】Linux下Oracle RAC集群搭建之基本测试与使用(九)

    Oracle 11G RAC数据库安装(九) 概述:写下本文档的初衷和动力,来源于上篇的<oracle基本操作手册>.oracle基本操作手册是作者研一假期对oracle基础知识学习的汇总 ...

  8. Jsp的九大对象,七大动作,三大指令

    jsp九大内置对象:1>out 向客户端输出数据,字节流.如out.print(" dgaweyr"); 2>request 接收客户端的http请求.String g ...

  9. 今天我们来认识一下JSP的九大内置对象

    虽然现在基本上我们都是使用SpringMVC+AJAX进行开发了Java Web了,但是还是很有必要了解一下JSP的九大内置对象的.像request.response.session这些对象,即便使用 ...

随机推荐

  1. struts2视频学习笔记 11-12(动态方法调用,接收请求参数)

    课时11 动态方法调用 如果Action中存在多个方法时,可以使用!+方法名调用指定方法.(不推荐使用) public String execute(){ setMsg("execute&q ...

  2. <context:annotation-config> 和 <context:component-scan>的差别

    <context:annotation-config> is used to activate annotations in beans already registered in the ...

  3. string字符串类型

    一次设置一个key-value 使用set命令可以一次设置一个key-value,使用get命令可以查询key所关联的字符串值.如下图所示. 一次设置多个key-value 使用mset命令可以设置多 ...

  4. spring+springmvc+mybatis整合

    1.web.xml配置 <?xml version="1.0" encoding="UTF-8"?> <web-app version=&qu ...

  5. win7下python安装pyquery

    安装pyquery之前首先要明确一点,easyinstall 是一款python包管理器,类似于node的npm,用于安装python的扩展包,它安装的包是以*.egg的方式. 要安装pq需要经历以下 ...

  6. 一模 (5) day2

    第一题: 题目大意:使得 x^x 达到或超过 n 位数字的最小正整数 x 是多少? n<=2*10^9 解题过程: 1.以前看到过这题了,一个数x的位数=(int)lg(x)+1  换一下底就是 ...

  7. 解决JavaScript中使用$.ajax方式提交数组参数

    一般的,可能有些人在一个参数有多个值的情况下,可能以某个字符分隔的形式传递,比如页面上有多个checkbox: $.ajax{ url:"xxxx", data:{ p: &quo ...

  8. MySql插入记录时判断

    我们在开发数据库相关的逻辑过程中, 经常检查表中是否已经存在这样的一条记录, 如果存在则更新或者不做操作, 如果没有存在记录,则需要插入一条新的记录. 这样的逻辑固然可以通过两条sql语句完成. SE ...

  9. js用正则表达式控制价格输入

    <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN"> <html> <hea ...

  10. 使用Myeclipse创建自定义签名debug keystore

    1.在已经创建后的android项目上右击鼠标,如图所示 2.选择next下一步 3.选择create new keystore 注意  这里密码要输入android 4.点击next,录入基本信息 ...