axmol/cocos/2d/CCActionInterval.cpp

2231 lines
47 KiB
C++

/****************************************************************************
Copyright (c) 2010-2012 cocos2d-x.org
Copyright (c) 2008-2010 Ricardo Quesada
Copyright (c) 2011 Zynga Inc.
http://www.cocos2d-x.org
Permission is hereby granted, free of charge, to any person obtaining a copy
of this software and associated documentation files (the "Software"), to deal
in the Software without restriction, including without limitation the rights
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
copies of the Software, and to permit persons to whom the Software is
furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in
all copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
THE SOFTWARE.
****************************************************************************/
#include "CCActionInterval.h"
#include "CCSprite.h"
#include "CCNode.h"
#include "CCStdC.h"
#include "CCActionInstant.h"
#include <stdarg.h>
NS_CC_BEGIN
// Extra action for making a Sequence or Spawn when only adding one action to it.
class ExtraAction : public FiniteTimeAction
{
public:
static ExtraAction* create();
virtual ExtraAction* clone() const;
virtual ExtraAction* reverse(void) const;
virtual void update(float time);
virtual void step(float dt);
};
ExtraAction* ExtraAction::create()
{
ExtraAction* pRet = new ExtraAction();
if (pRet)
{
pRet->autorelease();
}
return pRet;
}
ExtraAction* ExtraAction::clone(void) const
{
// no copy constructor
auto a = new ExtraAction();
a->autorelease();
return a;
}
ExtraAction* ExtraAction::reverse(void) const
{
return ExtraAction::create();
}
void ExtraAction::update(float time)
{
CC_UNUSED_PARAM(time);
}
void ExtraAction::step(float dt)
{
CC_UNUSED_PARAM(dt);
}
//
// IntervalAction
//
bool ActionInterval::initWithDuration(float d)
{
_duration = d;
// prevent division by 0
// This comparison could be in step:, but it might decrease the performance
// by 3% in heavy based action games.
if (_duration == 0)
{
_duration = FLT_EPSILON;
}
_elapsed = 0;
_firstTick = true;
return true;
}
bool ActionInterval::isDone(void) const
{
return _elapsed >= _duration;
}
void ActionInterval::step(float dt)
{
if (_firstTick)
{
_firstTick = false;
_elapsed = 0;
}
else
{
_elapsed += dt;
}
this->update(MAX (0, // needed for rewind. elapsed could be negative
MIN(1, _elapsed /
MAX(_duration, FLT_EPSILON) // division by 0
)
)
);
}
void ActionInterval::setAmplitudeRate(float amp)
{
CC_UNUSED_PARAM(amp);
// Abstract class needs implementation
CCASSERT(0, "");
}
float ActionInterval::getAmplitudeRate(void)
{
// Abstract class needs implementation
CCASSERT(0, "");
return 0;
}
void ActionInterval::startWithTarget(Node *target)
{
FiniteTimeAction::startWithTarget(target);
_elapsed = 0.0f;
_firstTick = true;
}
//
// Sequence
//
Sequence* Sequence::createWithTwoActions(FiniteTimeAction *actionOne, FiniteTimeAction *actionTwo)
{
Sequence *sequence = new Sequence();
sequence->initWithTwoActions(actionOne, actionTwo);
sequence->autorelease();
return sequence;
}
Sequence* Sequence::create(FiniteTimeAction *pAction1, ...)
{
va_list params;
va_start(params, pAction1);
Sequence *pRet = Sequence::createWithVariableList(pAction1, params);
va_end(params);
return pRet;
}
Sequence* Sequence::createWithVariableList(FiniteTimeAction *pAction1, va_list args)
{
FiniteTimeAction *pNow;
FiniteTimeAction *pPrev = pAction1;
bool bOneAction = true;
while (pAction1)
{
pNow = va_arg(args, FiniteTimeAction*);
if (pNow)
{
pPrev = createWithTwoActions(pPrev, pNow);
bOneAction = false;
}
else
{
// If only one action is added to Sequence, make up a Sequence by adding a simplest finite time action.
if (bOneAction)
{
pPrev = createWithTwoActions(pPrev, ExtraAction::create());
}
break;
}
}
return ((Sequence*)pPrev);
}
Sequence* Sequence::create(const Vector<FiniteTimeAction*>& arrayOfActions)
{
Sequence* pRet = NULL;
do
{
long count = arrayOfActions.count();
CC_BREAK_IF(count == 0);
auto prev = arrayOfActions[0];
if (count > 1)
{
for (long i = 1; i < count; ++i)
{
prev = createWithTwoActions(prev, arrayOfActions[i]);
}
}
else
{
// If only one action is added to Sequence, make up a Sequence by adding a simplest finite time action.
prev = createWithTwoActions(prev, ExtraAction::create());
}
pRet = static_cast<Sequence*>(prev);
}while (0);
return pRet;
}
bool Sequence::initWithTwoActions(FiniteTimeAction *pActionOne, FiniteTimeAction *pActionTwo)
{
CCASSERT(pActionOne != NULL, "");
CCASSERT(pActionTwo != NULL, "");
float d = pActionOne->getDuration() + pActionTwo->getDuration();
ActionInterval::initWithDuration(d);
_actions[0] = pActionOne;
pActionOne->retain();
_actions[1] = pActionTwo;
pActionTwo->retain();
return true;
}
Sequence* Sequence::clone(void) const
{
// no copy constructor
auto a = new Sequence();
a->initWithTwoActions(_actions[0]->clone(), _actions[1]->clone() );
a->autorelease();
return a;
}
Sequence::~Sequence(void)
{
CC_SAFE_RELEASE(_actions[0]);
CC_SAFE_RELEASE(_actions[1]);
}
void Sequence::startWithTarget(Node *target)
{
ActionInterval::startWithTarget(target);
_split = _actions[0]->getDuration() / _duration;
_last = -1;
}
void Sequence::stop(void)
{
// Issue #1305
if( _last != - 1)
{
_actions[_last]->stop();
}
ActionInterval::stop();
}
void Sequence::update(float t)
{
int found = 0;
float new_t = 0.0f;
if( t < _split ) {
// action[0]
found = 0;
if( _split != 0 )
new_t = t / _split;
else
new_t = 1;
} else {
// action[1]
found = 1;
if ( _split == 1 )
new_t = 1;
else
new_t = (t-_split) / (1 - _split );
}
if ( found==1 ) {
if( _last == -1 ) {
// action[0] was skipped, execute it.
_actions[0]->startWithTarget(_target);
_actions[0]->update(1.0f);
_actions[0]->stop();
}
else if( _last == 0 )
{
// switching to action 1. stop action 0.
_actions[0]->update(1.0f);
_actions[0]->stop();
}
}
else if(found==0 && _last==1 )
{
// Reverse mode ?
// XXX: Bug. this case doesn't contemplate when _last==-1, found=0 and in "reverse mode"
// since it will require a hack to know if an action is on reverse mode or not.
// "step" should be overriden, and the "reverseMode" value propagated to inner Sequences.
_actions[1]->update(0);
_actions[1]->stop();
}
// Last action found and it is done.
if( found == _last && _actions[found]->isDone() )
{
return;
}
// Last action found and it is done
if( found != _last )
{
_actions[found]->startWithTarget(_target);
}
_actions[found]->update(new_t);
_last = found;
}
Sequence* Sequence::reverse() const
{
return Sequence::createWithTwoActions(_actions[1]->reverse(), _actions[0]->reverse());
}
//
// Repeat
//
Repeat* Repeat::create(FiniteTimeAction *pAction, unsigned int times)
{
Repeat* pRepeat = new Repeat();
pRepeat->initWithAction(pAction, times);
pRepeat->autorelease();
return pRepeat;
}
bool Repeat::initWithAction(FiniteTimeAction *pAction, unsigned int times)
{
float d = pAction->getDuration() * times;
if (ActionInterval::initWithDuration(d))
{
_times = times;
_innerAction = pAction;
pAction->retain();
_actionInstant = dynamic_cast<ActionInstant*>(pAction) ? true : false;
//an instant action needs to be executed one time less in the update method since it uses startWithTarget to execute the action
if (_actionInstant)
{
_times -=1;
}
_total = 0;
return true;
}
return false;
}
Repeat* Repeat::clone(void) const
{
// no copy constructor
auto a = new Repeat();
a->initWithAction( _innerAction->clone(), _times );
a->autorelease();
return a;
}
Repeat::~Repeat(void)
{
CC_SAFE_RELEASE(_innerAction);
}
void Repeat::startWithTarget(Node *target)
{
_total = 0;
_nextDt = _innerAction->getDuration()/_duration;
ActionInterval::startWithTarget(target);
_innerAction->startWithTarget(target);
}
void Repeat::stop(void)
{
_innerAction->stop();
ActionInterval::stop();
}
// issue #80. Instead of hooking step:, hook update: since it can be called by any
// container action like Repeat, Sequence, Ease, etc..
void Repeat::update(float dt)
{
if (dt >= _nextDt)
{
while (dt > _nextDt && _total < _times)
{
_innerAction->update(1.0f);
_total++;
_innerAction->stop();
_innerAction->startWithTarget(_target);
_nextDt += _innerAction->getDuration()/_duration;
}
// fix for issue #1288, incorrect end value of repeat
if(dt >= 1.0f && _total < _times)
{
_total++;
}
// don't set an instant action back or update it, it has no use because it has no duration
if (!_actionInstant)
{
if (_total == _times)
{
_innerAction->update(1);
_innerAction->stop();
}
else
{
// issue #390 prevent jerk, use right update
_innerAction->update(dt - (_nextDt - _innerAction->getDuration()/_duration));
}
}
}
else
{
_innerAction->update(fmodf(dt * _times,1.0f));
}
}
bool Repeat::isDone(void) const
{
return _total == _times;
}
Repeat* Repeat::reverse() const
{
return Repeat::create(_innerAction->reverse(), _times);
}
//
// RepeatForever
//
RepeatForever::~RepeatForever()
{
CC_SAFE_RELEASE(_innerAction);
}
RepeatForever *RepeatForever::create(ActionInterval *pAction)
{
RepeatForever *pRet = new RepeatForever();
if (pRet && pRet->initWithAction(pAction))
{
pRet->autorelease();
return pRet;
}
CC_SAFE_DELETE(pRet);
return NULL;
}
bool RepeatForever::initWithAction(ActionInterval *pAction)
{
CCASSERT(pAction != NULL, "");
pAction->retain();
_innerAction = pAction;
return true;
}
RepeatForever *RepeatForever::clone(void) const
{
// no copy constructor
auto a = new RepeatForever();
a->initWithAction(_innerAction->clone());
a->autorelease();
return a;
}
void RepeatForever::startWithTarget(Node* target)
{
ActionInterval::startWithTarget(target);
_innerAction->startWithTarget(target);
}
void RepeatForever::step(float dt)
{
_innerAction->step(dt);
if (_innerAction->isDone())
{
float diff = _innerAction->getElapsed() - _innerAction->getDuration();
_innerAction->startWithTarget(_target);
// to prevent jerk. issue #390, 1247
_innerAction->step(0.0f);
_innerAction->step(diff);
}
}
bool RepeatForever::isDone() const
{
return false;
}
RepeatForever *RepeatForever::reverse() const
{
return RepeatForever::create(_innerAction->reverse());
}
//
// Spawn
//
Spawn* Spawn::create(FiniteTimeAction *pAction1, ...)
{
va_list params;
va_start(params, pAction1);
Spawn *pRet = Spawn::createWithVariableList(pAction1, params);
va_end(params);
return pRet;
}
Spawn* Spawn::createWithVariableList(FiniteTimeAction *pAction1, va_list args)
{
FiniteTimeAction *pNow;
FiniteTimeAction *pPrev = pAction1;
bool bOneAction = true;
while (pAction1)
{
pNow = va_arg(args, FiniteTimeAction*);
if (pNow)
{
pPrev = createWithTwoActions(pPrev, pNow);
bOneAction = false;
}
else
{
// If only one action is added to Spawn, make up a Spawn by adding a simplest finite time action.
if (bOneAction)
{
pPrev = createWithTwoActions(pPrev, ExtraAction::create());
}
break;
}
}
return ((Spawn*)pPrev);
}
Spawn* Spawn::create(const Vector<FiniteTimeAction*>& arrayOfActions)
{
Spawn* pRet = NULL;
do
{
long count = arrayOfActions.count();
CC_BREAK_IF(count == 0);
auto prev = arrayOfActions[0];
if (count > 1)
{
for (int i = 1; i < arrayOfActions.count(); ++i)
{
prev = createWithTwoActions(prev, arrayOfActions[i]);
}
}
else
{
// If only one action is added to Spawn, make up a Spawn by adding a simplest finite time action.
prev = createWithTwoActions(prev, ExtraAction::create());
}
pRet = static_cast<Spawn*>(prev);
}while (0);
return pRet;
}
Spawn* Spawn::createWithTwoActions(FiniteTimeAction *action1, FiniteTimeAction *action2)
{
Spawn *spawn = new Spawn();
spawn->initWithTwoActions(action1, action2);
spawn->autorelease();
return spawn;
}
bool Spawn:: initWithTwoActions(FiniteTimeAction *pAction1, FiniteTimeAction *pAction2)
{
CCASSERT(pAction1 != NULL, "");
CCASSERT(pAction2 != NULL, "");
bool bRet = false;
float d1 = pAction1->getDuration();
float d2 = pAction2->getDuration();
if (ActionInterval::initWithDuration(MAX(d1, d2)))
{
_one = pAction1;
_two = pAction2;
if (d1 > d2)
{
_two = Sequence::createWithTwoActions(pAction2, DelayTime::create(d1 - d2));
}
else if (d1 < d2)
{
_one = Sequence::createWithTwoActions(pAction1, DelayTime::create(d2 - d1));
}
_one->retain();
_two->retain();
bRet = true;
}
return bRet;
}
Spawn* Spawn::clone(void) const
{
// no copy constructor
auto a = new Spawn();
a->initWithTwoActions(_one->clone(), _two->clone());
a->autorelease();
return a;
}
Spawn::~Spawn(void)
{
CC_SAFE_RELEASE(_one);
CC_SAFE_RELEASE(_two);
}
void Spawn::startWithTarget(Node *target)
{
ActionInterval::startWithTarget(target);
_one->startWithTarget(target);
_two->startWithTarget(target);
}
void Spawn::stop(void)
{
_one->stop();
_two->stop();
ActionInterval::stop();
}
void Spawn::update(float time)
{
if (_one)
{
_one->update(time);
}
if (_two)
{
_two->update(time);
}
}
Spawn* Spawn::reverse() const
{
return Spawn::createWithTwoActions(_one->reverse(), _two->reverse());
}
//
// RotateTo
//
RotateTo* RotateTo::create(float fDuration, float fDeltaAngle)
{
RotateTo* pRotateTo = new RotateTo();
pRotateTo->initWithDuration(fDuration, fDeltaAngle);
pRotateTo->autorelease();
return pRotateTo;
}
bool RotateTo::initWithDuration(float fDuration, float fDeltaAngle)
{
if (ActionInterval::initWithDuration(fDuration))
{
_dstAngleX = _dstAngleY = fDeltaAngle;
return true;
}
return false;
}
RotateTo* RotateTo::create(float fDuration, float fDeltaAngleX, float fDeltaAngleY)
{
RotateTo* pRotateTo = new RotateTo();
pRotateTo->initWithDuration(fDuration, fDeltaAngleX, fDeltaAngleY);
pRotateTo->autorelease();
return pRotateTo;
}
bool RotateTo::initWithDuration(float fDuration, float fDeltaAngleX, float fDeltaAngleY)
{
if (ActionInterval::initWithDuration(fDuration))
{
_dstAngleX = fDeltaAngleX;
_dstAngleY = fDeltaAngleY;
return true;
}
return false;
}
RotateTo* RotateTo::clone(void) const
{
// no copy constructor
auto a = new RotateTo();
a->initWithDuration(_duration, _dstAngleX, _dstAngleY);
a->autorelease();
return a;
}
void RotateTo::startWithTarget(Node *target)
{
ActionInterval::startWithTarget(target);
// Calculate X
_startAngleX = target->getRotationX();
if (_startAngleX > 0)
{
_startAngleX = fmodf(_startAngleX, 360.0f);
}
else
{
_startAngleX = fmodf(_startAngleX, -360.0f);
}
_diffAngleX = _dstAngleX - _startAngleX;
if (_diffAngleX > 180)
{
_diffAngleX -= 360;
}
if (_diffAngleX < -180)
{
_diffAngleX += 360;
}
//Calculate Y: It's duplicated from calculating X since the rotation wrap should be the same
_startAngleY = _target->getRotationY();
if (_startAngleY > 0)
{
_startAngleY = fmodf(_startAngleY, 360.0f);
}
else
{
_startAngleY = fmodf(_startAngleY, -360.0f);
}
_diffAngleY = _dstAngleY - _startAngleY;
if (_diffAngleY > 180)
{
_diffAngleY -= 360;
}
if (_diffAngleY < -180)
{
_diffAngleY += 360;
}
}
void RotateTo::update(float time)
{
if (_target)
{
_target->setRotationX(_startAngleX + _diffAngleX * time);
_target->setRotationY(_startAngleY + _diffAngleY * time);
}
}
RotateTo *RotateTo::reverse() const
{
CCASSERT(false, "RotateTo doesn't support the 'reverse' method");
return nullptr;
}
//
// RotateBy
//
RotateBy* RotateBy::create(float fDuration, float fDeltaAngle)
{
RotateBy *pRotateBy = new RotateBy();
pRotateBy->initWithDuration(fDuration, fDeltaAngle);
pRotateBy->autorelease();
return pRotateBy;
}
bool RotateBy::initWithDuration(float fDuration, float fDeltaAngle)
{
if (ActionInterval::initWithDuration(fDuration))
{
_angleX = _angleY = fDeltaAngle;
return true;
}
return false;
}
RotateBy* RotateBy::create(float fDuration, float fDeltaAngleX, float fDeltaAngleY)
{
RotateBy *pRotateBy = new RotateBy();
pRotateBy->initWithDuration(fDuration, fDeltaAngleX, fDeltaAngleY);
pRotateBy->autorelease();
return pRotateBy;
}
bool RotateBy::initWithDuration(float fDuration, float fDeltaAngleX, float fDeltaAngleY)
{
if (ActionInterval::initWithDuration(fDuration))
{
_angleX = fDeltaAngleX;
_angleY = fDeltaAngleY;
return true;
}
return false;
}
RotateBy* RotateBy::clone(void) const
{
// no copy constructor
auto a = new RotateBy();
a->initWithDuration(_duration, _angleX, _angleY);
a->autorelease();
return a;
}
void RotateBy::startWithTarget(Node *target)
{
ActionInterval::startWithTarget(target);
_startAngleX = target->getRotationX();
_startAngleY = target->getRotationY();
}
void RotateBy::update(float time)
{
// XXX: shall I add % 360
if (_target)
{
_target->setRotationX(_startAngleX + _angleX * time);
_target->setRotationY(_startAngleY + _angleY * time);
}
}
RotateBy* RotateBy::reverse() const
{
return RotateBy::create(_duration, -_angleX, -_angleY);
}
//
// MoveBy
//
MoveBy* MoveBy::create(float duration, const Point& deltaPosition)
{
MoveBy *pRet = new MoveBy();
pRet->initWithDuration(duration, deltaPosition);
pRet->autorelease();
return pRet;
}
bool MoveBy::initWithDuration(float duration, const Point& deltaPosition)
{
if (ActionInterval::initWithDuration(duration))
{
_positionDelta = deltaPosition;
return true;
}
return false;
}
MoveBy* MoveBy::clone(void) const
{
// no copy constructor
auto a = new MoveBy();
a->initWithDuration(_duration, _positionDelta);
a->autorelease();
return a;
}
void MoveBy::startWithTarget(Node *target)
{
ActionInterval::startWithTarget(target);
_previousPosition = _startPosition = target->getPosition();
}
MoveBy* MoveBy::reverse() const
{
return MoveBy::create(_duration, Point( -_positionDelta.x, -_positionDelta.y));
}
void MoveBy::update(float t)
{
if (_target)
{
#if CC_ENABLE_STACKABLE_ACTIONS
Point currentPos = _target->getPosition();
Point diff = currentPos - _previousPosition;
_startPosition = _startPosition + diff;
Point newPos = _startPosition + (_positionDelta * t);
_target->setPosition(newPos);
_previousPosition = newPos;
#else
_target->setPosition(ccpAdd( _startPosition, ccpMult(_positionDelta, t)));
#endif // CC_ENABLE_STACKABLE_ACTIONS
}
}
//
// MoveTo
//
MoveTo* MoveTo::create(float duration, const Point& position)
{
MoveTo *pRet = new MoveTo();
pRet->initWithDuration(duration, position);
pRet->autorelease();
return pRet;
}
bool MoveTo::initWithDuration(float duration, const Point& position)
{
if (ActionInterval::initWithDuration(duration))
{
_endPosition = position;
return true;
}
return false;
}
MoveTo* MoveTo::clone(void) const
{
// no copy constructor
auto a = new MoveTo();
a->initWithDuration(_duration, _endPosition);
a->autorelease();
return a;
}
void MoveTo::startWithTarget(Node *target)
{
MoveBy::startWithTarget(target);
_positionDelta = _endPosition - target->getPosition();
}
//
// SkewTo
//
SkewTo* SkewTo::create(float t, float sx, float sy)
{
SkewTo *pSkewTo = new SkewTo();
if (pSkewTo)
{
if (pSkewTo->initWithDuration(t, sx, sy))
{
pSkewTo->autorelease();
}
else
{
CC_SAFE_DELETE(pSkewTo);
}
}
return pSkewTo;
}
bool SkewTo::initWithDuration(float t, float sx, float sy)
{
bool bRet = false;
if (ActionInterval::initWithDuration(t))
{
_endSkewX = sx;
_endSkewY = sy;
bRet = true;
}
return bRet;
}
SkewTo* SkewTo::clone(void) const
{
// no copy constructor
auto a = new SkewTo();
a->initWithDuration(_duration, _endSkewX, _endSkewY);
a->autorelease();
return a;
}
SkewTo* SkewTo::reverse() const
{
CCASSERT(false, "reverse() not supported in SkewTo");
return nullptr;
}
void SkewTo::startWithTarget(Node *target)
{
ActionInterval::startWithTarget(target);
_startSkewX = target->getSkewX();
if (_startSkewX > 0)
{
_startSkewX = fmodf(_startSkewX, 180.f);
}
else
{
_startSkewX = fmodf(_startSkewX, -180.f);
}
_deltaX = _endSkewX - _startSkewX;
if (_deltaX > 180)
{
_deltaX -= 360;
}
if (_deltaX < -180)
{
_deltaX += 360;
}
_startSkewY = target->getSkewY();
if (_startSkewY > 0)
{
_startSkewY = fmodf(_startSkewY, 360.f);
}
else
{
_startSkewY = fmodf(_startSkewY, -360.f);
}
_deltaY = _endSkewY - _startSkewY;
if (_deltaY > 180)
{
_deltaY -= 360;
}
if (_deltaY < -180)
{
_deltaY += 360;
}
}
void SkewTo::update(float t)
{
_target->setSkewX(_startSkewX + _deltaX * t);
_target->setSkewY(_startSkewY + _deltaY * t);
}
SkewTo::SkewTo()
: _skewX(0.0)
, _skewY(0.0)
, _startSkewX(0.0)
, _startSkewY(0.0)
, _endSkewX(0.0)
, _endSkewY(0.0)
, _deltaX(0.0)
, _deltaY(0.0)
{
}
//
// SkewBy
//
SkewBy* SkewBy::create(float t, float sx, float sy)
{
SkewBy *pSkewBy = new SkewBy();
if (pSkewBy)
{
if (pSkewBy->initWithDuration(t, sx, sy))
{
pSkewBy->autorelease();
}
else
{
CC_SAFE_DELETE(pSkewBy);
}
}
return pSkewBy;
}
SkewBy * SkewBy::clone() const
{
// no copy constructor
auto a = new SkewBy();
a->initWithDuration(_duration, _skewX, _skewY);
a->autorelease();
return a;
}
bool SkewBy::initWithDuration(float t, float deltaSkewX, float deltaSkewY)
{
bool bRet = false;
if (SkewTo::initWithDuration(t, deltaSkewX, deltaSkewY))
{
_skewX = deltaSkewX;
_skewY = deltaSkewY;
bRet = true;
}
return bRet;
}
void SkewBy::startWithTarget(Node *target)
{
SkewTo::startWithTarget(target);
_deltaX = _skewX;
_deltaY = _skewY;
_endSkewX = _startSkewX + _deltaX;
_endSkewY = _startSkewY + _deltaY;
}
SkewBy* SkewBy::reverse() const
{
return SkewBy::create(_duration, -_skewX, -_skewY);
}
//
// JumpBy
//
JumpBy* JumpBy::create(float duration, const Point& position, float height, int jumps)
{
JumpBy *pJumpBy = new JumpBy();
pJumpBy->initWithDuration(duration, position, height, jumps);
pJumpBy->autorelease();
return pJumpBy;
}
bool JumpBy::initWithDuration(float duration, const Point& position, float height, int jumps)
{
CCASSERT(jumps>=0, "Number of jumps must be >= 0");
if (ActionInterval::initWithDuration(duration) && jumps>=0)
{
_delta = position;
_height = height;
_jumps = jumps;
return true;
}
return false;
}
JumpBy* JumpBy::clone(void) const
{
// no copy constructor
auto a = new JumpBy();
a->initWithDuration(_duration, _delta, _height, _jumps);
a->autorelease();
return a;
}
void JumpBy::startWithTarget(Node *target)
{
ActionInterval::startWithTarget(target);
_previousPos = _startPosition = target->getPosition();
}
void JumpBy::update(float t)
{
// parabolic jump (since v0.8.2)
if (_target)
{
float frac = fmodf( t * _jumps, 1.0f );
float y = _height * 4 * frac * (1 - frac);
y += _delta.y * t;
float x = _delta.x * t;
#if CC_ENABLE_STACKABLE_ACTIONS
Point currentPos = _target->getPosition();
Point diff = currentPos - _previousPos;
_startPosition = diff + _startPosition;
Point newPos = _startPosition + Point(x,y);
_target->setPosition(newPos);
_previousPos = newPos;
#else
_target->setPosition(ccpAdd( _startPosition, Point(x,y)));
#endif // !CC_ENABLE_STACKABLE_ACTIONS
}
}
JumpBy* JumpBy::reverse() const
{
return JumpBy::create(_duration, Point(-_delta.x, -_delta.y),
_height, _jumps);
}
//
// JumpTo
//
JumpTo* JumpTo::create(float duration, const Point& position, float height, int jumps)
{
JumpTo *pJumpTo = new JumpTo();
pJumpTo->initWithDuration(duration, position, height, jumps);
pJumpTo->autorelease();
return pJumpTo;
}
JumpTo* JumpTo::clone(void) const
{
// no copy constructor
auto a = new JumpTo();
a->initWithDuration(_duration, _delta, _height, _jumps);
a->autorelease();
return a;
}
JumpTo* JumpTo::reverse() const
{
CCASSERT(false, "reverse() not supported in JumpTo");
return nullptr;
}
void JumpTo::startWithTarget(Node *target)
{
JumpBy::startWithTarget(target);
_delta = Point(_delta.x - _startPosition.x, _delta.y - _startPosition.y);
}
// Bezier cubic formula:
// ((1 - t) + t)3 = 1
// Expands to ...
// (1 - t)3 + 3t(1-t)2 + 3t2(1 - t) + t3 = 1
static inline float bezierat( float a, float b, float c, float d, float t )
{
return (powf(1-t,3) * a +
3*t*(powf(1-t,2))*b +
3*powf(t,2)*(1-t)*c +
powf(t,3)*d );
}
//
// BezierBy
//
BezierBy* BezierBy::create(float t, const ccBezierConfig& c)
{
BezierBy *pBezierBy = new BezierBy();
pBezierBy->initWithDuration(t, c);
pBezierBy->autorelease();
return pBezierBy;
}
bool BezierBy::initWithDuration(float t, const ccBezierConfig& c)
{
if (ActionInterval::initWithDuration(t))
{
_config = c;
return true;
}
return false;
}
void BezierBy::startWithTarget(Node *target)
{
ActionInterval::startWithTarget(target);
_previousPosition = _startPosition = target->getPosition();
}
BezierBy* BezierBy::clone(void) const
{
// no copy constructor
auto a = new BezierBy();
a->initWithDuration(_duration, _config);
a->autorelease();
return a;
}
void BezierBy::update(float time)
{
if (_target)
{
float xa = 0;
float xb = _config.controlPoint_1.x;
float xc = _config.controlPoint_2.x;
float xd = _config.endPosition.x;
float ya = 0;
float yb = _config.controlPoint_1.y;
float yc = _config.controlPoint_2.y;
float yd = _config.endPosition.y;
float x = bezierat(xa, xb, xc, xd, time);
float y = bezierat(ya, yb, yc, yd, time);
#if CC_ENABLE_STACKABLE_ACTIONS
Point currentPos = _target->getPosition();
Point diff = currentPos - _previousPosition;
_startPosition = _startPosition + diff;
Point newPos = _startPosition + Point(x,y);
_target->setPosition(newPos);
_previousPosition = newPos;
#else
_target->setPosition(ccpAdd( _startPosition, Point(x,y)));
#endif // !CC_ENABLE_STACKABLE_ACTIONS
}
}
BezierBy* BezierBy::reverse(void) const
{
ccBezierConfig r;
r.endPosition = -_config.endPosition;
r.controlPoint_1 = _config.controlPoint_2 + (-_config.endPosition);
r.controlPoint_2 = _config.controlPoint_1 + (-_config.endPosition);
BezierBy *pAction = BezierBy::create(_duration, r);
return pAction;
}
//
// BezierTo
//
BezierTo* BezierTo::create(float t, const ccBezierConfig& c)
{
BezierTo *pBezierTo = new BezierTo();
pBezierTo->initWithDuration(t, c);
pBezierTo->autorelease();
return pBezierTo;
}
bool BezierTo::initWithDuration(float t, const ccBezierConfig &c)
{
if (ActionInterval::initWithDuration(t))
{
_toConfig = c;
return true;
}
return false;
}
BezierTo* BezierTo::clone(void) const
{
// no copy constructor
auto a = new BezierTo();
a->initWithDuration(_duration, _toConfig);
a->autorelease();
return a;
}
void BezierTo::startWithTarget(Node *target)
{
BezierBy::startWithTarget(target);
_config.controlPoint_1 = _toConfig.controlPoint_1 - _startPosition;
_config.controlPoint_2 = _toConfig.controlPoint_2 - _startPosition;
_config.endPosition = _toConfig.endPosition - _startPosition;
}
BezierTo* BezierTo::reverse() const
{
CCASSERT(false, "CCBezierTo doesn't support the 'reverse' method");
return nullptr;
}
//
// ScaleTo
//
ScaleTo* ScaleTo::create(float duration, float s)
{
ScaleTo *pScaleTo = new ScaleTo();
pScaleTo->initWithDuration(duration, s);
pScaleTo->autorelease();
return pScaleTo;
}
ScaleTo* ScaleTo::create(float duration, float sx, float sy)
{
ScaleTo *pScaleTo = new ScaleTo();
pScaleTo->initWithDuration(duration, sx, sy);
pScaleTo->autorelease();
return pScaleTo;
}
bool ScaleTo::initWithDuration(float duration, float s)
{
if (ActionInterval::initWithDuration(duration))
{
_endScaleX = s;
_endScaleY = s;
return true;
}
return false;
}
bool ScaleTo::initWithDuration(float duration, float sx, float sy)
{
if (ActionInterval::initWithDuration(duration))
{
_endScaleX = sx;
_endScaleY = sy;
return true;
}
return false;
}
ScaleTo* ScaleTo::clone(void) const
{
// no copy constructor
auto a = new ScaleTo();
a->initWithDuration(_duration, _endScaleX, _endScaleY);
a->autorelease();
return a;
}
ScaleTo* ScaleTo::reverse() const
{
CCASSERT(false, "reverse() not supported in ScaleTo");
return nullptr;
}
void ScaleTo::startWithTarget(Node *target)
{
ActionInterval::startWithTarget(target);
_startScaleX = target->getScaleX();
_startScaleY = target->getScaleY();
_deltaX = _endScaleX - _startScaleX;
_deltaY = _endScaleY - _startScaleY;
}
void ScaleTo::update(float time)
{
if (_target)
{
_target->setScaleX(_startScaleX + _deltaX * time);
_target->setScaleY(_startScaleY + _deltaY * time);
}
}
//
// ScaleBy
//
ScaleBy* ScaleBy::create(float duration, float s)
{
ScaleBy *pScaleBy = new ScaleBy();
pScaleBy->initWithDuration(duration, s);
pScaleBy->autorelease();
return pScaleBy;
}
ScaleBy* ScaleBy::create(float duration, float sx, float sy)
{
ScaleBy *pScaleBy = new ScaleBy();
pScaleBy->initWithDuration(duration, sx, sy);
pScaleBy->autorelease();
return pScaleBy;
}
ScaleBy* ScaleBy::clone(void) const
{
// no copy constructor
auto a = new ScaleBy();
a->initWithDuration(_duration, _endScaleX, _endScaleY);
a->autorelease();
return a;
}
void ScaleBy::startWithTarget(Node *target)
{
ScaleTo::startWithTarget(target);
_deltaX = _startScaleX * _endScaleX - _startScaleX;
_deltaY = _startScaleY * _endScaleY - _startScaleY;
}
ScaleBy* ScaleBy::reverse() const
{
return ScaleBy::create(_duration, 1 / _endScaleX, 1 / _endScaleY);
}
//
// Blink
//
Blink* Blink::create(float duration, int blinks)
{
Blink *pBlink = new Blink();
pBlink->initWithDuration(duration, blinks);
pBlink->autorelease();
return pBlink;
}
bool Blink::initWithDuration(float duration, int blinks)
{
CCASSERT(blinks>=0, "blinks should be >= 0");
if (ActionInterval::initWithDuration(duration) && blinks>=0)
{
_times = blinks;
return true;
}
return false;
}
void Blink::stop()
{
_target->setVisible(_originalState);
ActionInterval::stop();
}
void Blink::startWithTarget(Node *target)
{
ActionInterval::startWithTarget(target);
_originalState = target->isVisible();
}
Blink* Blink::clone(void) const
{
// no copy constructor
auto a = new Blink();
a->initWithDuration(_duration, _times);
a->autorelease();
return a;
}
void Blink::update(float time)
{
if (_target && ! isDone())
{
float slice = 1.0f / _times;
float m = fmodf(time, slice);
_target->setVisible(m > slice / 2 ? true : false);
}
}
Blink* Blink::reverse() const
{
return Blink::create(_duration, _times);
}
//
// FadeIn
//
FadeIn* FadeIn::create(float d)
{
FadeIn* pAction = new FadeIn();
pAction->initWithDuration(d);
pAction->autorelease();
return pAction;
}
FadeIn* FadeIn::clone() const
{
// no copy constructor
auto a = new FadeIn();
a->initWithDuration(_duration);
a->autorelease();
return a;
}
void FadeIn::update(float time)
{
RGBAProtocol *pRGBAProtocol = dynamic_cast<RGBAProtocol*>(_target);
if (pRGBAProtocol)
{
pRGBAProtocol->setOpacity((GLubyte)(255 * time));
}
/*_target->setOpacity((GLubyte)(255 * time));*/
}
ActionInterval* FadeIn::reverse() const
{
return FadeOut::create(_duration);
}
//
// FadeOut
//
FadeOut* FadeOut::create(float d)
{
FadeOut* pAction = new FadeOut();
pAction->initWithDuration(d);
pAction->autorelease();
return pAction;
}
FadeOut* FadeOut::clone() const
{
// no copy constructor
auto a = new FadeOut();
a->initWithDuration(_duration);
a->autorelease();
return a;
}
void FadeOut::update(float time)
{
RGBAProtocol *pRGBAProtocol = dynamic_cast<RGBAProtocol*>(_target);
if (pRGBAProtocol)
{
pRGBAProtocol->setOpacity(GLubyte(255 * (1 - time)));
}
/*_target->setOpacity(GLubyte(255 * (1 - time)));*/
}
ActionInterval* FadeOut::reverse() const
{
return FadeIn::create(_duration);
}
//
// FadeTo
//
FadeTo* FadeTo::create(float duration, GLubyte opacity)
{
FadeTo *pFadeTo = new FadeTo();
pFadeTo->initWithDuration(duration, opacity);
pFadeTo->autorelease();
return pFadeTo;
}
bool FadeTo::initWithDuration(float duration, GLubyte opacity)
{
if (ActionInterval::initWithDuration(duration))
{
_toOpacity = opacity;
return true;
}
return false;
}
FadeTo* FadeTo::clone() const
{
// no copy constructor
auto a = new FadeTo();
a->initWithDuration(_duration, _toOpacity);
a->autorelease();
return a;
}
FadeTo* FadeTo::reverse() const
{
CCASSERT(false, "reverse() not supported in FadeTo");
return nullptr;
}
void FadeTo::startWithTarget(Node *target)
{
ActionInterval::startWithTarget(target);
RGBAProtocol *pRGBAProtocol = dynamic_cast<RGBAProtocol*>(target);
if (pRGBAProtocol)
{
_fromOpacity = pRGBAProtocol->getOpacity();
}
/*_fromOpacity = target->getOpacity();*/
}
void FadeTo::update(float time)
{
RGBAProtocol *pRGBAProtocol = dynamic_cast<RGBAProtocol*>(_target);
if (pRGBAProtocol)
{
pRGBAProtocol->setOpacity((GLubyte)(_fromOpacity + (_toOpacity - _fromOpacity) * time));
}
/*_target->setOpacity((GLubyte)(_fromOpacity + (_toOpacity - _fromOpacity) * time));*/
}
//
// TintTo
//
TintTo* TintTo::create(float duration, GLubyte red, GLubyte green, GLubyte blue)
{
TintTo *pTintTo = new TintTo();
pTintTo->initWithDuration(duration, red, green, blue);
pTintTo->autorelease();
return pTintTo;
}
bool TintTo::initWithDuration(float duration, GLubyte red, GLubyte green, GLubyte blue)
{
if (ActionInterval::initWithDuration(duration))
{
_to = Color3B(red, green, blue);
return true;
}
return false;
}
TintTo* TintTo::clone() const
{
// no copy constructor
auto a = new TintTo();
a->initWithDuration(_duration, _to.r, _to.g, _to.b);
a->autorelease();
return a;
}
TintTo* TintTo::reverse() const
{
CCASSERT(false, "reverse() not supported in TintTo");
return nullptr;
}
void TintTo::startWithTarget(Node *target)
{
ActionInterval::startWithTarget(target);
RGBAProtocol *pRGBAProtocol = dynamic_cast<RGBAProtocol*>(_target);
if (pRGBAProtocol)
{
_from = pRGBAProtocol->getColor();
}
/*_from = target->getColor();*/
}
void TintTo::update(float time)
{
RGBAProtocol *pRGBAProtocol = dynamic_cast<RGBAProtocol*>(_target);
if (pRGBAProtocol)
{
pRGBAProtocol->setColor(Color3B(GLubyte(_from.r + (_to.r - _from.r) * time),
(GLubyte)(_from.g + (_to.g - _from.g) * time),
(GLubyte)(_from.b + (_to.b - _from.b) * time)));
}
}
//
// TintBy
//
TintBy* TintBy::create(float duration, GLshort deltaRed, GLshort deltaGreen, GLshort deltaBlue)
{
TintBy *pTintBy = new TintBy();
pTintBy->initWithDuration(duration, deltaRed, deltaGreen, deltaBlue);
pTintBy->autorelease();
return pTintBy;
}
bool TintBy::initWithDuration(float duration, GLshort deltaRed, GLshort deltaGreen, GLshort deltaBlue)
{
if (ActionInterval::initWithDuration(duration))
{
_deltaR = deltaRed;
_deltaG = deltaGreen;
_deltaB = deltaBlue;
return true;
}
return false;
}
TintBy* TintBy::clone() const
{
// no copy constructor
auto a = new TintBy();
a->initWithDuration(_duration, (GLubyte)_deltaR, (GLubyte)_deltaG, (GLubyte)_deltaB);
a->autorelease();
return a;
}
void TintBy::startWithTarget(Node *target)
{
ActionInterval::startWithTarget(target);
RGBAProtocol *pRGBAProtocol = dynamic_cast<RGBAProtocol*>(target);
if (pRGBAProtocol)
{
Color3B color = pRGBAProtocol->getColor();
_fromR = color.r;
_fromG = color.g;
_fromB = color.b;
}
}
void TintBy::update(float time)
{
RGBAProtocol *pRGBAProtocol = dynamic_cast<RGBAProtocol*>(_target);
if (pRGBAProtocol)
{
pRGBAProtocol->setColor(Color3B((GLubyte)(_fromR + _deltaR * time),
(GLubyte)(_fromG + _deltaG * time),
(GLubyte)(_fromB + _deltaB * time)));
}
}
TintBy* TintBy::reverse() const
{
return TintBy::create(_duration, -_deltaR, -_deltaG, -_deltaB);
}
//
// DelayTime
//
DelayTime* DelayTime::create(float d)
{
DelayTime* pAction = new DelayTime();
pAction->initWithDuration(d);
pAction->autorelease();
return pAction;
}
DelayTime* DelayTime::clone() const
{
// no copy constructor
auto a = new DelayTime();
a->initWithDuration(_duration);
a->autorelease();
return a;
}
void DelayTime::update(float time)
{
CC_UNUSED_PARAM(time);
return;
}
DelayTime* DelayTime::reverse() const
{
return DelayTime::create(_duration);
}
//
// ReverseTime
//
ReverseTime* ReverseTime::create(FiniteTimeAction *pAction)
{
// casting to prevent warnings
ReverseTime *pReverseTime = new ReverseTime();
pReverseTime->initWithAction( pAction->clone() );
pReverseTime->autorelease();
return pReverseTime;
}
bool ReverseTime::initWithAction(FiniteTimeAction *pAction)
{
CCASSERT(pAction != NULL, "");
CCASSERT(pAction != _other, "");
if (ActionInterval::initWithDuration(pAction->getDuration()))
{
// Don't leak if action is reused
CC_SAFE_RELEASE(_other);
_other = pAction;
pAction->retain();
return true;
}
return false;
}
ReverseTime* ReverseTime::clone() const
{
// no copy constructor
auto a = new ReverseTime();
a->initWithAction( _other->clone() );
a->autorelease();
return a;
}
ReverseTime::ReverseTime() : _other(NULL)
{
}
ReverseTime::~ReverseTime(void)
{
CC_SAFE_RELEASE(_other);
}
void ReverseTime::startWithTarget(Node *target)
{
ActionInterval::startWithTarget(target);
_other->startWithTarget(target);
}
void ReverseTime::stop(void)
{
_other->stop();
ActionInterval::stop();
}
void ReverseTime::update(float time)
{
if (_other)
{
_other->update(1 - time);
}
}
ReverseTime* ReverseTime::reverse() const
{
// XXX: This looks like a bug
return (ReverseTime*)_other->clone();
}
//
// Animate
//
Animate* Animate::create(Animation *pAnimation)
{
Animate *pAnimate = new Animate();
pAnimate->initWithAnimation(pAnimation);
pAnimate->autorelease();
return pAnimate;
}
Animate::Animate()
: _splitTimes(new std::vector<float>)
, _nextFrame(0)
, _origFrame(NULL)
, _executedLoops(0)
, _animation(NULL)
{
}
Animate::~Animate()
{
CC_SAFE_RELEASE(_animation);
CC_SAFE_RELEASE(_origFrame);
CC_SAFE_DELETE(_splitTimes);
}
bool Animate::initWithAnimation(Animation* animation)
{
CCASSERT( animation!=NULL, "Animate: argument Animation must be non-NULL");
float singleDuration = animation->getDuration();
if ( ActionInterval::initWithDuration(singleDuration * animation->getLoops() ) )
{
_nextFrame = 0;
setAnimation(animation);
_origFrame = NULL;
_executedLoops = 0;
_splitTimes->reserve(animation->getFrames().count());
float accumUnitsOfTime = 0;
float newUnitOfTimeValue = singleDuration / animation->getTotalDelayUnits();
auto frames = animation->getFrames();
for (auto& frame : frames)
{
float value = (accumUnitsOfTime * newUnitOfTimeValue) / singleDuration;
accumUnitsOfTime += frame->getDelayUnits();
_splitTimes->push_back(value);
}
return true;
}
return false;
}
void Animate::setAnimation(cocos2d::Animation *animation)
{
if (_animation != animation)
{
CC_SAFE_RETAIN(animation);
CC_SAFE_RELEASE(_animation);
_animation = animation;
}
}
Animate* Animate::clone() const
{
// no copy constructor
auto a = new Animate();
a->initWithAnimation(_animation->clone());
a->autorelease();
return a;
}
void Animate::startWithTarget(Node *target)
{
ActionInterval::startWithTarget(target);
Sprite *pSprite = static_cast<Sprite*>(target);
CC_SAFE_RELEASE(_origFrame);
if (_animation->getRestoreOriginalFrame())
{
_origFrame = pSprite->getDisplayFrame();
_origFrame->retain();
}
_nextFrame = 0;
_executedLoops = 0;
}
void Animate::stop(void)
{
if (_animation->getRestoreOriginalFrame() && _target)
{
static_cast<Sprite*>(_target)->setDisplayFrame(_origFrame);
}
ActionInterval::stop();
}
void Animate::update(float t)
{
// if t==1, ignore. Animation should finish with t==1
if( t < 1.0f ) {
t *= _animation->getLoops();
// new loop? If so, reset frame counter
unsigned int loopNumber = (unsigned int)t;
if( loopNumber > _executedLoops ) {
_nextFrame = 0;
_executedLoops++;
}
// new t for animations
t = fmodf(t, 1.0f);
}
auto frames = _animation->getFrames();
long numberOfFrames = frames.count();
SpriteFrame *frameToDisplay = NULL;
for( int i=_nextFrame; i < numberOfFrames; i++ ) {
float splitTime = _splitTimes->at(i);
if( splitTime <= t ) {
AnimationFrame* frame = frames[i];
frameToDisplay = frame->getSpriteFrame();
static_cast<Sprite*>(_target)->setDisplayFrame(frameToDisplay);
Dictionary* dict = frame->getUserInfo();
if( dict )
{
//TODO: [[NSNotificationCenter defaultCenter] postNotificationName:AnimationFrameDisplayedNotification object:target_ userInfo:dict];
}
_nextFrame = i+1;
}
// Issue 1438. Could be more than one frame per tick, due to low frame rate or frame delta < 1/FPS
else {
break;
}
}
}
Animate* Animate::reverse() const
{
auto oldArray = _animation->getFrames();
Vector<AnimationFrame*> newArray(oldArray.count());
if (oldArray.count() > 0)
{
for (auto iter = oldArray.crbegin(); iter != oldArray.crend(); ++iter)
{
AnimationFrame* animFrame = *iter;
if (!animFrame)
{
break;
}
newArray.addObject(animFrame->clone());
}
}
Animation *newAnim = Animation::create(newArray, _animation->getDelayPerUnit(), _animation->getLoops());
newAnim->setRestoreOriginalFrame(_animation->getRestoreOriginalFrame());
return Animate::create(newAnim);
}
// TargetedAction
TargetedAction::TargetedAction()
: _action(NULL)
, _forcedTarget(NULL)
{
}
TargetedAction::~TargetedAction()
{
CC_SAFE_RELEASE(_forcedTarget);
CC_SAFE_RELEASE(_action);
}
TargetedAction* TargetedAction::create(Node* target, FiniteTimeAction* pAction)
{
TargetedAction* p = new TargetedAction();
p->initWithTarget(target, pAction);
p->autorelease();
return p;
}
bool TargetedAction::initWithTarget(Node* target, FiniteTimeAction* pAction)
{
if(ActionInterval::initWithDuration(pAction->getDuration()))
{
CC_SAFE_RETAIN(target);
_forcedTarget = target;
CC_SAFE_RETAIN(pAction);
_action = pAction;
return true;
}
return false;
}
TargetedAction* TargetedAction::clone() const
{
// no copy constructor
auto a = new TargetedAction();
// win32 : use the _other's copy object.
a->initWithTarget(_forcedTarget, _action->clone());
a->autorelease();
return a;
}
TargetedAction* TargetedAction::reverse(void) const
{
// just reverse the internal action
auto a = new TargetedAction();
a->initWithTarget(_forcedTarget, _action->reverse());
a->autorelease();
return a;
}
void TargetedAction::startWithTarget(Node *target)
{
ActionInterval::startWithTarget(target);
_action->startWithTarget(_forcedTarget);
}
void TargetedAction::stop(void)
{
_action->stop();
}
void TargetedAction::update(float time)
{
_action->update(time);
}
void TargetedAction::setForcedTarget(Node* forcedTarget)
{
if( _forcedTarget != forcedTarget ) {
CC_SAFE_RETAIN(forcedTarget);
CC_SAFE_RELEASE(_forcedTarget);
_forcedTarget = forcedTarget;
}
}
NS_CC_END