cocos2d-x3.3 源码分析之-动作Action和ActionManager

写这篇文章,没有花我多少时间。但是由于我的笔记写在云笔记中,图片也多。csdn也不支持直接复制粘贴图片,可以说把笔记移植过来比我当初写还要费事,希望改进吧

从入口开始分析,我们来看看addaction函数,他接受一个动作,并绑定自己,是否暂停取决于节点的running属性。

我们来具体看看ActionManager这个类

class CC_DLLActionManager : public Ref

{

public:

ActionManager(void);

~ActionManager(void);

void addAction(Action *action, Node*target, bool paused);

void removeAllActions();

void removeAllActionsFromTarget(Node*target);

void removeAction(Action *action);

void removeActionByTag(int tag, Node*target);

void removeAllActionsByTag(int tag, Node*target);

Action* getActionByTag(int tag, const Node*target) const;

ssize_tgetNumberOfRunningActionsInTarget(const Node *target) const;

void pauseTarget(Node *target);

void resumeTarget(Node *target);

Vector<Node*>pauseAllRunningActions();

void resumeTargets(constVector<Node*>& targetsToResume);

void update(float dt);

protected:

void removeActionAtIndex(ssize_t index,struct _hashElement *element);

void deleteHashElement(struct _hashElement*element);

void actionAllocWithHashElement(struct_hashElement *element);

protected:

 struct _hashElement    *_targets;

struct_hashElement    *_currentTarget;

bool            _currentTargetSalvaged;

};

struct _hashElement   *_targets;

struct_hashElement    *_currentTarget;

如果是对定时器数据结构很熟悉的人没看到这个结构体应该不会陌生。的确,内部结构是相同的,只不过把之前的定时器对象改为现在的action.

类似的,每个hash节点一般由一个target标志,根据target找到节点。每个节点弱引用一个动作的array,也就是说,每个节点可以绑定多个动作,paused记录了当前动作的暂停状态。

下面来看看addAction函数:

voidActionManager::addAction(Action *action, Node *target, bool paused)

{

CCASSERT(action != nullptr, "");

CCASSERT(target != nullptr, "");

//查找hash表,看看是否有相同的target,如果没有新建一个插入表中。

tHashElement *element = nullptr;

// we should convert it to Ref*, because wesave it as Ref*

Ref *tmp = target;

HASH_FIND_PTR(_targets, &tmp, element);

if (! element)

{

element =(tHashElement*)calloc(sizeof(*element), 1);

element->paused = paused;

target->retain();

element->target = target;

HASH_ADD_PTR(_targets, target,element);

}

//为弱引用的数组分配内存

 actionAllocWithHashElement(element);

//同一个动作结束前,如果被同一个节点包含两次,触发断言

  CCASSERT(!ccArrayContainsObject(element->actions, action), "");

//将新的动作加入数组

 ccArrayAppendObject(element->actions,action);

action->startWithTarget(target);

}

现在我们不介绍最后一句,因为我们需要回过头去看看Action及其子类

Action:

class CC_DLL Action: public Ref, public Clonable

{

public:

  //动作的默认tag,一般不可以用默认tag索引动作

 static const int INVALID_TAG = -1;

/**

* @js NA

* @lua NA

*/

virtual std::string description() const;

/** returns a clone of action */

virtual Action* clone() const

{

CC_ASSERT(0);

return nullptr;

}

/** returns a new action that performs theexactly the reverse action */

virtual Action* reverse() const

{

CC_ASSERT(0);

return nullptr;

}

//! return true if the action has finished

//判断动作是否结束

 virtual bool isDone() const;

//! called before the action start. It willalso set the target.

//_taggetoriginTarget都赋值为新的绑定对象

  virtual void startWithTarget(Node *target);

/**

called after the action has finished. Itwill set the ‘target‘ to nil.

IMPORTANT: You should never call"[action stop]" manually. Instead, use:"target->stopAction(action);"

*/

//动作结束后,调用该函数,把_target设置为null

  virtual void stop();

//! called every frame with it‘s deltatime. DON‘T override unless you know what you are doing.

//类似定时器中的update函数

 virtual void step(float dt);

/**

called once per frame. time a value between0 and 1

For example:

- 0 means that the action just started

- 0.5 means that the action is in themiddle

- 1 means that the action is over

*/

//类似定时器中的trigger函数

  virtual void update(float time);

inline Node* getTarget() const { return_target; }

/** The action will modify the targetproperties. */

inline void setTarget(Node *target) {_target = target; }

inline Node* getOriginalTarget() const {return _originalTarget; }

/** Set the original target, since targetcan be nil.

Is the target that were used to run theaction. Unless you are doing something complex, like ActionManager, you shouldNOT call this method.

The target is ‘assigned‘, it is not‘retained‘.

@since v0.8.2

*/

inline void setOriginalTarget(Node*originalTarget) { _originalTarget = originalTarget; }

inline int getTag() const { return _tag; }

inline void setTag(int tag) { _tag = tag; }

CC_CONSTRUCTOR_ACCESS:

Action();

virtual ~Action();

protected:

 Node   *_originalTarget;

/** The "target".

The target will be set with the‘startWithTarget‘ method.

When the ‘stop‘ method is called, targetwill be set to nil.

The target is ‘assigned‘, it is not‘retained‘.

*/

Node    *_target;

/** The action tag. An identifier of theaction */

int    _tag;

private:

CC_DISALLOW_COPY_AND_ASSIGN(Action);

};

isDownupdatestep这三个函数是虚函数,action并没有做什么实际的事情,所以我们现在还不知道他干了些什么,我们之后去看看具体的动作类就会清楚了。

_originalTarget_target这两者有什么区分现在也不清楚

我们来看看最重要的一个子类FiniteTimeAction:

class CC_DLLFiniteTimeAction : public Action

{

public:

//! get duration in seconds of the action

inline float getDuration() const { return_duration; }

//! set duration in seconds of the action

inline void setDuration(float duration) {_duration = duration; }

//

// Overrides

//

virtual FiniteTimeAction* reverse() constoverride

{

CC_ASSERT(0);

return nullptr;

}

virtual FiniteTimeAction* clone() constoverride

{

CC_ASSERT(0);

return nullptr;

}

CC_CONSTRUCTOR_ACCESS:

FiniteTimeAction()

: _duration(0)

{}

virtual ~FiniteTimeAction(){}

protected:

//! duration in seconds

  float _duration;

private:

CC_DISALLOW_COPY_AND_ASSIGN(FiniteTimeAction);

};

只是增加了一个_duration属性,也就是动作的持续时间,应该是为了他的延时动作子类准备的。

我么接着看ActionInstant这个瞬时动作:

class CC_DLLActionInstant : public FiniteTimeAction //<NSCopying>

{

public:

//

// Overrides

//

virtual ActionInstant* clone() constoverride

{

CC_ASSERT(0);

return nullptr;

}

virtual ActionInstant * reverse() constoverride

{

CC_ASSERT(0);

return nullptr;

}

 virtual bool isDone() const override;

 virtual void step(float dt) override;

virtual void update(float time) override;

};

瞬时动作类重写了step函数和isDown函数,其他的isDone和update函数并没有做出变化

step:

调用了update函数,并传入参数1。我们可以看看update函数的注释

called once perframe. time a value between 0 and 1

For example:

0 means that the action just started

0.5 means that the action is in the middle

1 means that the action is over

*/

1表示动作结束了

isDown:

isDown也返回true。我们或许会疑惑,不过接下来看了ActionManager中的update函数之后你就会明白了:

该函数在导演类的init函数中被注册到定时器:

该函数有最高的定时触发优先级,每一帧动作的运行都会触发该函数。

voidActionManager::update(float dt)

{

for (tHashElement *elt = _targets; elt !=nullptr; )

{

_currentTarget = elt;

_currentTargetSalvaged = false;

if (! _currentTarget->paused)

{

// The ‘actions‘ MutableArray maychange while inside this loop.

for (_currentTarget->actionIndex= 0; _currentTarget->actionIndex < _currentTarget->actions->num;

_currentTarget->actionIndex++)

{

_currentTarget->currentAction =(Action*)_currentTarget->actions->arr[_currentTarget->actionIndex];

if(_currentTarget->currentAction == nullptr)

{

continue;

}

_currentTarget->currentActionSalvaged = false;

  _currentTarget->currentAction->step(dt);

if(_currentTarget->currentActionSalvaged)

{

// The currentAction toldthe node to remove it. To prevent the action from

// accidentallydeallocating itself before finishing its step, we retained

// it. Now that step isdone, it‘s safe to release it.

_currentTarget->currentAction->release();

} else

if (_currentTarget->currentAction->isDone())

{

    _currentTarget->currentAction->stop();

Action *action =_currentTarget->currentAction;

// Make currentAction nilto prevent removeAction from salvaging it.

_currentTarget->currentAction = nullptr;

  removeAction(action);

}

_currentTarget->currentAction = nullptr;

}

}

// elt, at this moment, is still valid

// so it is safe to ask this here(issue #490)

elt = (tHashElement*)(elt->hh.next);

// only delete currentTarget if noactions were scheduled during the cycle (issue #481)

if (_currentTargetSalvaged &&_currentTarget->actions->num == 0)

{

deleteHashElement(_currentTarget);

}

}

// issue #635

_currentTarget = nullptr;

}

我们可以看到,step就相当于一个触发函数,而动作中的update函数就是触发函数中调用的函数,出发完成后,判断动作是否结束,如果结束调用stop函数,并移除动作。

我们现在可以回到之前的瞬时动作,update函数被传值为1isDown返回true,就很好理解了。

瞬时动作的基类没有实现update函数,我们以一个具体的瞬时动作来做例子:

class CC_DLL Show : public ActionInstant

{

public:

/** Allocates and initializes the action */

static Show * create();

//

// Overrides

//

   virtual void update(float time) override;

virtual ActionInstant* reverse() constoverride;

virtual Show* clone() const override;

CC_CONSTRUCTOR_ACCESS:

Show(){}

virtual ~Show(){}

private:

CC_DISALLOW_COPY_AND_ASSIGN(Show);

};

重写了update函数:

接着看

void ActionManager::update(float dt)中的那个removeAction函数:

好了一目了然,之前的_originTarget_targrt的区别也就知道了,_target是为了实现节点动作用的,二_target置为空之后,_originTarget便可以用来删除这个动作,动作删除后被release

好了,我们可以接着看延时动作了:

class CC_DLLActionInterval : public FiniteTimeAction

{

public:

/** how many seconds had elapsed since theactions started to run. */

inline float getElapsed(void) { return_elapsed; }

//extension in GridAction

void setAmplitudeRate(float amp);

float getAmplitudeRate(void);

//

// Overrides

//

 virtual bool isDone(void) const override;

virtual voidstep(float dt) override;

virtual void startWithTarget(Node *target)override;

virtual ActionInterval* reverse() constoverride

{

CC_ASSERT(0);

return nullptr;

}

virtual ActionInterval *clone() constoverride

{

CC_ASSERT(0);

return nullptr;

}

CC_CONSTRUCTOR_ACCESS:

/** initializes the action */

bool initWithDuration(float d);

protected:

float_elapsed;

bool  _firstTick;

};

_elapsed表示延时动作度过的时间

isDone:

很简单,只要度过的时间大于等于动作的生命周期就表示动作结束。

step:

很简单的定时逻辑,ActionManager中的update函数每一次出发后调用step函数,dt一般是每一帧的时间(大多数是1
/60
)。每一次调用后,把dt加到_elapsed就可以做到计时了。update的参数被限制在0-1之间,具体的实现得看一个具体的例子:

这里的initWithDuration就是我们创建动作时候传递的动作生命周期。

update:

很简单吧,每次变化后的属性重新设置给Node就达到了动画的目的。

我们接下来看看移除动作:

void removeAction(Action *action):

移除很简单,找到绑定对象,从对象的动作数组中移除即可。

接下来看看Node中的stopAction函数:

函数原理都是一样的,拿其中一个举例:

间接调用了移除动作的函数。

注意事项:不要用action中的stop函数来停止动作,因为判断动作是否结束的标志是isDown函数,并且判断过程还是在动作执行之后。stop函数会将_target置为null,这样运行动作就会试图去修改_target的属性,导致程序奔溃。还有一点就是动作运行结束后会被删除,如果想多次运行动作,请retain。

总结:

动作最重要的几个函数

isDown

step

stop

update

以及ActionManager::update

还有动作的重要属性:

_target

_origintarget

_elapsed

时间: 2024-10-09 11:47:52

cocos2d-x3.3 源码分析之-动作Action和ActionManager的相关文章

【Cocos2d-x】源码分析之 2d/ui/Widget

从今天开始 咱也模仿 红孩儿这些大牛分析源码 ,由于水平有限 不对之处欢迎狂喷.哈哈. #ifndef __UIWIDGET_H__ #define __UIWIDGET_H__ #include "ui/CCProtectedNode.h" #include "ui/UILayoutDefine.h" #include "ui/UILayoutParameter.h" #include "ui/GUIDefine.h" NS

【Cocos2d-x】源码分析之 2d/ui/UILayout

#ifndef __LAYOUT_H__ #define __LAYOUT_H__ #include "ui/UIWidget.h" NS_CC_BEGIN namespace ui { typedef enum { LAYOUT_COLOR_NONE,//空 LAYOUT_COLOR_SOLID,//单一固定颜色的 LAYOUT_COLOR_GRADIENT//有梯度变化的 }LayoutBackGroundColorType;//容器背景颜色类型 typedef enum { LA

传奇源码分析-客户端(游戏逻辑处理源分析四)

现在假设玩家开始操作游戏:传奇的客户端源代码工程WindHorn一.CWHApp派生CWHWindow和CWHDXGraphicWindow.二.CWHDefProcess派生出CloginProcess.CcharacterProcess.CgameProcess客户端WinMain调用CWHDXGraphicWindow g_xMainWnd;创建一个窗口.客户端CWHDXGraphicWindow在自己的Create函数中调用了CWHWindow的Create来创建窗口,然后再调用自己的C

redis源码分析3---结构体---字典

redis源码分析3---结构体---字典 字典,简单来说就是一种用于保存键值对的抽象数据结构: 注意,字典中每个键都是独一无二的:在redis中,内部的redis的数据库就是使用字典作为底层实现的: 1 字典的实现 在redis中,字典是使用哈希表作为底层实现的,一个hash表里面可以有多个hash表节点,而每个hash表节点就保存了字典中的一个键值对: hash表定义 table属性是一个数组,数组中的每个元素都是一个指向dictEntry结构的指针,每个dictEntry结构保存着一个键值

ABP源码分析三十三:ABP.Web

ABP.Web模块并不复杂,主要完成ABP系统的初始化和一些基础功能的实现. AbpWebApplication : 继承自ASP.Net的HttpApplication类,主要完成下面三件事一,在Application_Start完成AbpBootstrapper的初始化.整个ABP系统的初始化就是通过AbpBootstrapper完成初始化的.二,在Application_BeginRequest设置根据request或cookie中的Culture信息,完成当前工作线程的CurrentCu

JVM源码分析之堆外内存完全解读

概述 广义的堆外内存 说到堆外内存,那大家肯定想到堆内内存,这也是我们大家接触最多的,我们在jvm参数里通常设置-Xmx来指定我们的堆的最大值,不过这还不是我们理解的Java堆,-Xmx的值是新生代和老生代的和的最大值,我们在jvm参数里通常还会加一个参数-XX:MaxPermSize来指定持久代的最大值,那么我们认识的Java堆的最大值其实是-Xmx和-XX:MaxPermSize的总和,在分代算法下,新生代,老生代和持久代是连续的虚拟地址,因为它们是一起分配的,那么剩下的都可以认为是堆外内存

Java集合系列之LinkedList源码分析

一.LinkedList简介 LinkedList是一种可以在任何位置进行高效地插入和移除操作的有序序列,它是基于双向链表实现的. ps:这里有一个问题,就是关于实现LinkedList的数据结构是否为循环的双向链表,上网搜了有很多文章都说是循环的,并且有的文章中但是我看了源代码觉得应该不是循环的? 例如在删除列表尾部节点的代码: private E unlinkLast(Node<E> l) { final E element = l.item; final Node<E> pr

YII 的源码分析(二)

上一篇简单分析了一下yii的流程,从创建一个应用,到屏幕上输出结果.这一次我来一个稍复杂一点的,重点在输出上,不再是简单的一行"hello world",而是要经过view(视图)层的处理. 依然是demos目录,这次我们选择hangman,一个简单的猜字游戏.老规则,还是从入口处开始看. index.php: <?php // change the following paths if necessary $yii=dirname(__FILE__).'/../../frame

cocos2d-x 源码分析 : control 源码分析 ( 控制类组件 controlButton)

源码版本来自3.1rc 转载请注明 cocos2d-x源码分析总目录 http://blog.csdn.net/u011225840/article/details/31743129 1.继承结构 control的设计整体感觉挺美的,在父类control定义了整个控制事件的基础以及管理,虽然其继承了Layer,但其本身和UI组件的实现并没有关联.在子类(controlButton,controlSwitch,controlStepper等中实现不同的UI组件).下面通过源码来分析control与