大佬教程收集整理的这篇文章主要介绍了菜鸟学习Cocos2d-x 3.x——浅谈动作Action,大佬教程大佬觉得挺不错的,现在分享给大家,也给大家做个参考。
一款游戏,设计的再NB的游戏,如果都是一堆静态的图片,没有任何动作,那也只能“呵呵”了。动作体系对于一款游戏的成功与否,有着非常重要的影响。所以,这篇文章就对Cocos2d-x中的动作进行总结。先来看看Cocos2d-x中的与动作相关的类。
与动作相关的类图如下图所示:现在就对这些类进行简单的介绍,在后续的小节中还会进行详细的分析的。
下面就对上面说的这些类进行通过实际的代码进行总结。
以下是Action
类的主要成员函数:
/** * 返回一个新的Action对象,表示原动作的相反的动作 */
virtual Action* reverse() const = 0;
// 如果动作已经完成了,就返回true
virtual bool isDone() const;
// 在动作开始之前被调用,设置动作作用的对象
void startWithTarget(Node *target);
/** * 在动作完成以后会被调用,它会设置"target"对象为空 * 注:请永远不要手动调用该函数,而是调用对应的"target->stopAction(action);" */
stop();
/** * 每帧都会调用的方法,如果你需要在每帧控制动作,则需要重写,时间间隔为动作间隔时间 * 最好不要重写该函数,除非你真的知道怎么做 */
step(float dt);
/** * 每一帧都会调用一次该函数,参数time取值为0和1之间的任意值,例如: * 0表示动作刚刚开始的时候调用; * 0.5表示动作执行到一半的时候调用; * 1表示动作完成以后调用; */
update(float timE);
// 获得执行动作的对象
inline Node* getTarget() const { return _target; }
// 设置执行动作的对象
inline setTarget(Node *target) { _target = target; }
// 动作标签
int getTag() const { return _tag; }
setTag(int tag) { _tag = tag; }
ActionInstant
是瞬时动作类。瞬时动作表示瞬间完成动作,中间没有任何动画效果;由于ActionInstant
的子类那么多,这里就以FlipX
为例子,做一个简单的Demo。效果如下图所示:可以看到小狗走到最左边时,会有一个转向的过程,这就是FlipX
的动作效果。示例代码如下:
Size visibleSize = Director::geTinstance()->getVisibleSize();
// 创建移动动作
ActionInterval *moveto = MoveTo::create(5,Vec2(0,200));
// 创建X轴方向的翻转动作
ActionInstant *flipx = FlipX::create(true);
ActionInterval *moveBACk = MoveTo::create(200));
auto scene = Director::geTinstance()->getRunningScene();
auto layer = scene->getChildByTag(1);
auto dog = layer->getChildByTag(1);
// flipx->reverse()获得对应的逆向动作
auto action = Sequence::create(moveto,flipx,moveBACk,flipx->reverse(),null);
dog->runAction(action);
瞬时动作是只能够立刻完成的动作,这类动作是在下一帧立刻完成的动作,如设定位置、设定缩放等。把它们包装成动作后,可以与其他动作类组合为复杂动作。
动作会在指定的时间内完成,中间会有动画效果。延时动作通过属性值的逐渐变化来实现动画效果。需要注意的是XXTo和XXBy的区别在于XXTo是表示最终值,而XXBy则表示向量-改变值。比如@H_844_141@moveTo和@H_844_141@moveBy动作,如以下效果所示:示例代码如下:
200));
ActionInterval *moveby = MoveBy::create(number" style="color:rgb(140,Vec2(-200,211)">0));
auto dog1 = layer->getChildByTag(@H_544_197@auto dog2 = layer->getChildByTag(2);
dog1->runAction(moveto);
dog2->runAction(moveby);
上面的那只狗是使用的@H_844_141@moveBy动作,而下面这只狗使用的是@H_844_141@moveTo动作。@H_844_141@moveTo动作都在用,下面就来看看稍微复杂点的贝塞尔曲线动作。
使用贝塞尔曲线,可以使节点进行曲线运动。每条贝塞尔曲线都包含一个起点和一个终点。在一条曲线中,起点和终点各自包含一个控制点,而控制点到端点的连线称作控制线。控制点决定了曲线的形状,包含角度和长度两个参数。如下图:实现效果如下:ccBezierConfig bezier;
bezier.controlPoint_1 = Point(300);
bezier.controlPoint_2 = Point(400,211)">400);
bezier.endPosition = Point(50,211)">200);
auto bezierAction = BezierTo::create(2.0f,bezier);
dog1->runAction(bezierAction);
我们在实际开发中,主要的任务就是确定两个控制点,去协调精灵的移动弧度。 在游戏中,我们经常要实现一些加速度或者减速度的效果。Cocos2d-x已经为我们做好了。 ActionEase类可以实现动作的速度由快到慢、速度随时间改变的匀速运动。该类包含5类运动: 每类运动都包含3个不同时期的变换:In、Out和InOut。 上述5类运动分别对应以下的类: 通过图表和描述也不能形象的说明上述5中运动,下面就通过指数缓冲的实际运行效果来进行展示: 指数缓冲ActionInterval *moveto1 = MoveTo::create(100));
ActionInterval *moveto2 = MoveTo::create(300));
ActionInterval *moveto3 = MoveTo::create(500));
2);
auto dog3 = layer->getChildByTag(3);
auto action1 = EaseExponentialIn::create(moveto1);
auto action2 = EaseExponentialOut::create(moveto2);
auto action3 = EaseExponentialInOut::create(moveto3);
dog1->runAction(action1);
dog2->runAction(action2);
dog3->runAction(action3);
从下到上的三次小狗,分别对应的是 在游戏中,很多时候,一个对象并不是单纯的执行一个动作,而是依次执行一系列动作或者同时执行一系列动作,那么这又该如何去完成呢?现在就来看看Cocos2d-x中是如何去完成这些的吧。 在实际试验中,发现, 注意:Sequence动作不能嵌入其他复合动作内使用,DelayTime不属于复合动作,但是只能在复合动作内使用。 跟随动作 有的时候,我们需要在动作完成以后,回调一个我们自定义的函数,完成某些功能,比如攻击一个敌人,攻击动作完成以后,敌人需要减少血量,这个时候,就需要使用函数回调动作。 CallFunc系列动作包括 当某个对象执行 在 实际上, 在不同的情况下,我们可以根据不同的需求来选择不同的 动作管理类 动作管理单例通过定时刷新自己的 实际上是由动作管理单例驱动每个动作去更新自己的逻辑,而 后续分析具体的小游戏的时候,遇到了 终于总结完了,很显然,这部分的内容远不止这么点。我这里只是把比较重要的几个部分进行了详细的总结;后续的博文中,遇到了再做更详细的总结。希望这篇稍微有点长的博文对大家有帮助。 这篇文章稍微有点长,写的稍微有点累,写的真心不容易。各位,有钱的捧个钱场,有人的捧个人场。 2014年11月24日 于深圳。 原文地址:http://www.jellythink.com/archives/749 以上是大佬教程为你收集整理的菜鸟学习Cocos2d-x 3.x——浅谈动作Action全部内容,希望文章能够帮你解决菜鸟学习Cocos2d-x 3.x——浅谈动作Action所遇到的程序开发问题。 如果觉得大佬教程网站内容还不错,欢迎将大佬教程推荐给程序员好友。 缓冲动作
指数缓冲;
In表示开始的时候加速;
指数缓冲:
EaseExponentialIn
、EaseExponentialOut
和EaseExponentialInOut
;
EaseSineIn
、EaseSineOut
和EaseSineInOut
;EaseElasticIn
、EaseElasticOut
和EaseElasticInOut
;EaseBounceIn
、EaseBounceOut
和EaseBounceInOut
;EaseBackIn
、EaseBackOut
和EaseBackInOut
。EaseExponentialInOut
。EaseExponentialIn表现的效果为速度越来越快;
组合动作
Sequence 可以使用
Sequence
定义一个动作序列,应用实例可以参见瞬时动作这小节。
Spawn
也是定义一系列动作,但是定义的动作会同时执行,使用方法同Sequence
是一样的。下面就通过具体的效果和代码来看看。可以看到,移动和淡入的效果在同时进行。示例代码如下: 100));
ActionInterval *fadein = FadeIn::create(5);
1);
auto spawnAction = Spawn::create(moveto1,fadein,null);
dog1->runAction(spawnAction);
Repeat
和RepeatForever
两个类可以使动作重复执行,Repeat
可以在参数中指定重复次数;RepeatForever
则是一直重复执行。从左往右,第一只小狗只跳一次;第二只小狗使用的Repeat
类,定义的跳跃5次;第三只小狗使用的RepeatForever
,将会一直跳跃下去。实例代码如下: // 创建跳跃动作
ActionInterval *jumpTo1 = JumpBy::create(3,211)">0),211)">100,211)">1);
ActionInterval *jumpTo2 = JumpBy::create(1);
ActionInterval *jumpTo3 = JumpBy::create(3);
// 重复跳动5次
auto repeatJump = Repeat::create((FiniteTimeAction *)jumpTo2,211)">5);
// 一直重复跳动动作
auto repeatForeverJump = RepeatForever::create((ActionInterval *)jumpTo3);
dog1->runAction(jumpTo1);
dog2->runAction(repeatJump);
dog3->runAction(repeatForeverJump);
RepeatForever
对于JumpTo
,并不能得到我们预期的结果。后来在源代码中,发现JumpTo
没有重写Action
类的update
方法,导致了错误的结果,希望在下个版本中有更改吧。 通常在开发中我们需要将各种动作组合起来再让节点执行,复合动作的作用就是将各种动作组合在一起。而且,复合动作本身也是动作。因此可以作为一个普通动作嵌入到其他动作中。
跟随动作
Follow
是一个节点跟随另外一个节点的动作。我都不理解这个动作,此处不做过多的总结。以后明白了,再回来补上。 函数回调动作
CallFunc
、CallFuncN
、__CCCallFuncND
,以及__CCCallFuncO
四个动作,用来在动作中进行方法的调用(之所以不是函数调用,是因为它们只能调用某个类中的实例方法,而不能调用普通的C函数)。CallFunc
系列动作时,就会调用一个先前被设置好的方法,以完成某些特别的功能。CallFunc
系列动作的4个类中:CallFunc
调用的方法不包含参数;
Node*
类型的参数,表示执行动作的对象;Node*
与void*
);Ref*
类型的参数;CallFunc
系列动作的后缀”N”表示Node参数,指的是执行动作的对象, “D”表示Data参数,指的是用户自定义的数据,”O”表示对象,指的是一个用户自定义的Ref参数。(__CCCallFuncND
和__CCCallFuncO
的命名好奇葩,可能在后续的版本中又要变了。)CallFunc
动作。看看代码示例。// 使用Lambda表达式
auto callFunc1 = CallFunc::create([&]{log("cocos2d-x 1"); });
// 使用成员函数
auto callFunc2 = CallFuncN::create(this,callfuncN_SELEctor(HelloWorld::testFunC));
// 顺序执行所有动作
auto action = Sequence::create(moveTo,callFunc1,callFunc2,null);
dog1->runAction(action);
动作管理
ActionManager
是管理所有动作的一个单例类,当我们调用runAction
函数时,该函数会把动作通过动作管理类的addAction
函数将对象传递给ActionManager
的单例,该实例再把这个动作添加到自己的动作序列中。update
方法,在这个方法中去调用行为序列中每个动作的step
,这些step
方法再根据自身的完成进度去update
或是结束行为。runAction
方法只是将动作对象添加进ActionManager
的待执行动作队列中,由ActionManager
去管理所有动作。当节点被清除或是动作结束时,动作管理类会自动将动作从队列中删除,不需要人为的去管理。ActionManager
,再做详细的分析。 总结
大佬总结
本图文内容来源于网友网络收集整理提供,作为学习参考使用,版权属于原作者。
如您有任何意见或建议可联系处理。小编QQ:384754419,请注明来意。