/**************************************************************************** Copyright (c) 2010 cocos2d-x.org 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 "CCIntervalAction.h" #include "CCSprite.h" #include "CCNode.h" #include "CGPointExtension.h" #include "platform/CCXMath.h" #include namespace cocos2d { // // IntervalAction // CCIntervalAction* CCIntervalAction::actionWithDuration(ccTime d) { CCIntervalAction *pAction = new CCIntervalAction(); pAction->initWithDuration(d); pAction->autorelease(); return pAction; } bool CCIntervalAction::initWithDuration(cocos2d::ccTime d) { m_fDuration = 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 (m_fDuration == 0) { m_fDuration = FLT_EPSILON; } m_elapsed = 0; m_bFirstTick = true; return true; } NSObject* CCIntervalAction::copyWithZone(NSZone *pZone) { NSZone* pNewZone = NULL; CCIntervalAction* pCopy = NULL; if(pZone && pZone->m_pCopyObject) { //in case of being called at sub class pCopy = (CCIntervalAction*)(pZone->m_pCopyObject); } else { // action's base class , must be called using __super::copyWithZone(), after overriding from derived class assert(0); pCopy = new CCIntervalAction(); pZone = pNewZone = new NSZone(pCopy); } CCFiniteTimeAction::copyWithZone(pZone); CCX_SAFE_DELETE(pNewZone); pCopy->initWithDuration(m_fDuration); return pCopy; } bool CCIntervalAction::isDone(void) { return m_elapsed >= m_fDuration; } void CCIntervalAction::step(ccTime dt) { if (m_bFirstTick) { m_bFirstTick = false; m_elapsed = 0; } else { m_elapsed += dt; } // update(min(1, m_elapsed/m_fDuration)); update(1 > m_elapsed/m_fDuration ? m_elapsed/m_fDuration : 1); } void CCIntervalAction::setAmplitudeRate(CGFloat amp) { // Abstract class needs implementation assert(0); } CGFloat CCIntervalAction::getAmplitudeRate(void) { // Abstract class needs implementation assert(0); return 0; } void CCIntervalAction::startWithTarget(CCNode *pTarget) { CCFiniteTimeAction::startWithTarget(pTarget); m_elapsed = 0.0f; m_bFirstTick = true; } CCIntervalAction* CCIntervalAction::reverse(void) { /* NSException* myException = [NSException exceptionWithName:@"ReverseActionNotImplemented" reason:@"Reverse Action not implemented" userInfo:nil]; @throw myException; */ return NULL; } // // Sequence // CCSequence* CCSequence::actionOneTwo(cocos2d::CCFiniteTimeAction *pActionOne, cocos2d::CCFiniteTimeAction *pActionTwo) { CCSequence *pSequence = new CCSequence(); pSequence->initOneTwo(pActionOne, pActionTwo); pSequence->autorelease(); return pSequence; } CCFiniteTimeAction* CCSequence::actions(cocos2d::CCFiniteTimeAction *pAction1, ...) { va_list params; va_start(params, pAction1); CCFiniteTimeAction *pNow; CCFiniteTimeAction *pPrev = pAction1; while (pAction1) { pNow = va_arg(params, CCFiniteTimeAction*); if (pNow) { pPrev = actionOneTwo(pPrev, pNow); } else { break; } } va_end(params); return pPrev; } bool CCSequence::initOneTwo(cocos2d::CCFiniteTimeAction *pActionOne, cocos2d::CCFiniteTimeAction *pActionTwo) { assert(pActionOne != NULL); assert(pActionTwo != NULL); ccTime d = pActionOne->getDuration() + pActionTwo->getDuration(); CCIntervalAction::initWithDuration(d); m_pActions[0] = pActionOne; pActionOne->retain(); m_pActions[1] = pActionTwo; pActionTwo->retain(); return true; } NSObject* CCSequence::copyWithZone(NSZone *pZone) { NSZone* pNewZone = NULL; CCSequence* pCopy = NULL; if(pZone && pZone->m_pCopyObject) { //in case of being called at sub class pCopy = (CCSequence*)(pZone->m_pCopyObject); } else { pCopy = new CCSequence(); pZone = pNewZone = new NSZone(pCopy); } CCIntervalAction::copyWithZone(pZone); pCopy->initOneTwo((CCFiniteTimeAction*)(m_pActions[0]->copy()->autorelease()), (CCFiniteTimeAction*)(m_pActions[1]->copy()->autorelease())); CCX_SAFE_DELETE(pNewZone); return pCopy; } CCSequence::~CCSequence(void) { m_pActions[0]->release(); m_pActions[1]->release(); } void CCSequence::startWithTarget(CCNode *pTarget) { CCIntervalAction::startWithTarget(pTarget); m_split = m_pActions[0]->getDuration() / m_fDuration; m_last = -1; } void CCSequence::stop(void) { m_pActions[0]->stop(); m_pActions[1]->stop(); CCIntervalAction::stop(); } void CCSequence::update(cocos2d::ccTime time) { int found = 0; ccTime new_t = 0.0f; if (time >= m_split) { found = 1; if (m_split == 1) { new_t = 1; } else { new_t = (time - m_split) / (1 - m_split); } } else { found = 0; if (m_split != 0) { new_t = time / m_split; } else { new_t = 1; } } if (m_last == -1 && found == 1) { m_pActions[0]->startWithTarget(m_pTarget); m_pActions[0]->update(1.0f); m_pActions[0]->stop(); } if (m_last != found) { if (m_last != -1) { m_pActions[m_last]->update(1.0f); m_pActions[m_last]->stop(); } m_pActions[found]->startWithTarget(m_pTarget); } m_pActions[found]->update(new_t); m_last = found; } CCIntervalAction* CCSequence::reverse(void) { return CCSequence::actionOneTwo(m_pActions[1]->reverse(), m_pActions[0]->reverse()); } // // Repeat // CCRepeat* CCRepeat::actionWithAction(cocos2d::CCFiniteTimeAction *pAction, unsigned int times) { CCRepeat* pRepeat = new CCRepeat(); pRepeat->initWithAction(pAction, times); pRepeat->autorelease(); return pRepeat; } bool CCRepeat::initWithAction(cocos2d::CCFiniteTimeAction *pAction, unsigned int times) { ccTime d = pAction->getDuration() * times; if (CCIntervalAction::initWithDuration(d)) { m_uTimes = times; m_pOther = pAction; pAction->retain(); m_uTotal = 0; return true; } return false; } NSObject* CCRepeat::copyWithZone(cocos2d::NSZone *pZone) { NSZone* pNewZone = NULL; CCRepeat* pCopy = NULL; if(pZone && pZone->m_pCopyObject) { //in case of being called at sub class pCopy = (CCRepeat*)(pZone->m_pCopyObject); } else { pCopy = new CCRepeat(); pZone = pNewZone = new NSZone(pCopy); } CCIntervalAction::copyWithZone(pZone); pCopy->initWithAction((CCFiniteTimeAction*)(m_pOther->copy()->autorelease()), m_uTimes); CCX_SAFE_DELETE(pNewZone); return pCopy; } CCRepeat::~CCRepeat(void) { m_pOther->release(); } void CCRepeat::startWithTarget(CCNode *pTarget) { m_uTotal = 0; CCIntervalAction::startWithTarget(pTarget); m_pOther->startWithTarget(pTarget); } void CCRepeat::stop(void) { m_pOther->stop(); CCIntervalAction::stop(); } // issue #80. Instead of hooking step:, hook update: since it can be called by any // container action like Repeat, Sequence, AccelDeccel, etc.. void CCRepeat::update(cocos2d::ccTime time) { ccTime t = time * m_uTimes; if (t > m_uTotal + 1) { m_pOther->update(1.0f); m_uTotal++; m_pOther->stop(); m_pOther->startWithTarget(m_pTarget); // repeat is over? if (m_uTotal == m_uTimes) { // so, set it in the original position m_pOther->update(0); } else { // no ? start next repeat with the right update // to prevent jerk (issue #390) m_pOther->update(t - m_uTotal); } } else { float r = fmodf(t, 1.0f); // fix last repeat position // else it could be 0. if (time == 1.0f) { r = 1.0f; m_uTotal++; // this is the added line } // m_pOther->update(min(r, 1)); m_pOther->update(r > 1 ? 1 : r); } } bool CCRepeat::isDone(void) { return m_uTotal == m_uTimes; } CCIntervalAction* CCRepeat::reverse(void) { return CCRepeat::actionWithAction(m_pOther->reverse(), m_uTimes); } // // RepeatForever // CCRepeatForever::~CCRepeatForever() { m_pOther->release(); } CCRepeatForever *CCRepeatForever::actionWithAction(CCIntervalAction *pAction) { CCRepeatForever *pRet = new CCRepeatForever(); if (pRet && pRet->initWithAction(pAction)) { pRet->autorelease(); return pRet; } CCX_SAFE_DELETE(pRet); return NULL; } CCRepeatForever *CCRepeatForever::initWithAction(CCIntervalAction *pAction) { pAction->retain(); m_pOther = pAction; return this; } NSObject* CCRepeatForever::copyWithZone(NSZone *pZone) { NSZone* pNewZone = NULL; CCRepeatForever* pRet = NULL; if(pZone && pZone->m_pCopyObject) //in case of being called at sub class { pRet = (CCRepeatForever*)(pZone->m_pCopyObject); } else { pRet = new CCRepeatForever(); pZone = pNewZone = new NSZone(pRet); } CCIntervalAction::copyWithZone(pZone); // win32 : use the m_pOther's copy object. pRet->initWithAction((CCIntervalAction*)(m_pOther->copy()->autorelease())); CCX_SAFE_DELETE(pNewZone); return pRet; } void CCRepeatForever::startWithTarget(CCNode* pTarget) { CCIntervalAction::startWithTarget(pTarget); m_pOther->startWithTarget(pTarget); } void CCRepeatForever::step(ccTime dt) { m_pOther->step(dt); if (m_pOther->isDone()) { ccTime diff = dt + m_pOther->getDuration() - m_pOther->getElapsed(); m_pOther->startWithTarget(m_pTarget); // to prevent jerk. issue #390 m_pOther->step(diff); } } bool CCRepeatForever::isDone() { return false; } CCIntervalAction *CCRepeatForever::reverse() { return (CCIntervalAction*)(CCRepeatForever::actionWithAction(m_pOther->reverse())); } // // Spawn // CCFiniteTimeAction* CCSpawn::actions(cocos2d::CCFiniteTimeAction *pAction1, ...) { va_list params; va_start(params, pAction1); CCFiniteTimeAction *pNow; CCFiniteTimeAction *pPrev = pAction1; while (pAction1) { pNow = va_arg(params, CCFiniteTimeAction*); if (pNow) { pPrev = actionOneTwo(pPrev, pNow); } else { break; } } va_end(params); return pPrev; } CCSpawn* CCSpawn::actionOneTwo(cocos2d::CCFiniteTimeAction *pAction1, cocos2d::CCFiniteTimeAction *pAction2) { CCSpawn *pSpawn = new CCSpawn(); pSpawn->initOneTwo(pAction1, pAction2); pSpawn->autorelease(); return pSpawn; } bool CCSpawn:: initOneTwo(CCFiniteTimeAction *pAction1, CCFiniteTimeAction *pAction2) { assert(pAction1 != NULL); assert(pAction2 != NULL); ccTime d1 = pAction1->getDuration(); ccTime d2 = pAction2->getDuration(); // __super::initWithDuration(fmaxf(d1, d2)); float maxd = (d1 >= d2 || CCXMath::isnanCocos2d(d2)) ? d1 : d2; CCIntervalAction::initWithDuration(maxd); m_pOne = pAction1; m_pTwo = pAction2; if (d1 > d2) { m_pTwo = CCSequence::actionOneTwo(pAction2, CCDelayTime::actionWithDuration(d1 - d2)); } else if (d1 < d2) { m_pOne = CCSequence::actionOneTwo(pAction1, CCDelayTime::actionWithDuration(d2 - d1)); } pAction1->retain(); pAction2->retain(); return true; } NSObject* CCSpawn::copyWithZone(cocos2d::NSZone *pZone) { NSZone* pNewZone = NULL; CCSpawn* pCopy = NULL; if(pZone && pZone->m_pCopyObject) { //in case of being called at sub class pCopy = (CCSpawn*)(pZone->m_pCopyObject); } else { pCopy = new CCSpawn(); pZone = pNewZone = new NSZone(pCopy); } CCIntervalAction::copyWithZone(pZone); pCopy->initOneTwo((CCFiniteTimeAction*)(m_pOne->copy()->autorelease()), (CCFiniteTimeAction*)(m_pTwo->copy()->autorelease())); CCX_SAFE_DELETE(pNewZone); return pCopy; } CCSpawn::~CCSpawn(void) { m_pOne->release(); m_pTwo->release(); } void CCSpawn::startWithTarget(CCNode *pTarget) { CCIntervalAction::startWithTarget(pTarget); m_pOne->startWithTarget(pTarget); m_pTwo->startWithTarget(pTarget); } void CCSpawn::stop(void) { m_pOne->stop(); m_pTwo->stop(); CCIntervalAction::stop(); } void CCSpawn::update(cocos2d::ccTime time) { m_pOne->update(time); m_pTwo->update(time); } CCIntervalAction* CCSpawn::reverse(void) { return CCSpawn::actionOneTwo(m_pOne->reverse(), m_pTwo->reverse()); } // // RotateTo // CCRotateTo* CCRotateTo::actionWithDuration(cocos2d::ccTime duration, float fDeltaAngle) { CCRotateTo* pRotateTo = new CCRotateTo(); pRotateTo->initWithDuration(duration, fDeltaAngle); pRotateTo->autorelease(); return pRotateTo; } bool CCRotateTo::initWithDuration(cocos2d::ccTime duration, float fDeltaAngle) { if (CCIntervalAction::initWithDuration(duration)) { m_fDstAngle = fDeltaAngle; return true; } return false; } NSObject* CCRotateTo::copyWithZone(cocos2d::NSZone *pZone) { NSZone* pNewZone = NULL; CCRotateTo* pCopy = NULL; if(pZone && pZone->m_pCopyObject) { //in case of being called at sub class pCopy = (CCRotateTo*)(pZone->m_pCopyObject); } else { pCopy = new CCRotateTo(); pZone = pNewZone = new NSZone(pCopy); } CCIntervalAction::copyWithZone(pZone); pCopy->initWithDuration(m_fDuration, m_fDstAngle); //Action *copy = [[[self class] allocWithZone: zone] initWithDuration:[self duration] angle: angle]; CCX_SAFE_DELETE(pNewZone); return pCopy; } void CCRotateTo::startWithTarget(CCNode *pTarget) { CCIntervalAction::startWithTarget(pTarget); m_fStartAngle = pTarget->getRotation(); if (m_fStartAngle > 0) { m_fStartAngle = fmodf(m_fStartAngle, 360.0f); } else { m_fStartAngle = fmodf(m_fStartAngle, -360.0f); } m_fDiffAngle = m_fDstAngle - m_fStartAngle; if (m_fDiffAngle > 180) { m_fDiffAngle -= 360; } if (m_fDiffAngle < -180) { m_fDiffAngle += 360; } } void CCRotateTo::update(cocos2d::ccTime time) { m_pTarget->setRotation(m_fStartAngle + m_fDiffAngle * time); } // // RotateBy // CCRotateBy* CCRotateBy::actionWithDuration(cocos2d::ccTime duration, float fDeltaAngle) { CCRotateBy *pRotateBy = new CCRotateBy(); pRotateBy->initWithDuration(duration, fDeltaAngle); pRotateBy->autorelease(); return pRotateBy; } bool CCRotateBy::initWithDuration(cocos2d::ccTime duration, float fDeltaAngle) { if (CCIntervalAction::initWithDuration(duration)) { m_fAngle = fDeltaAngle; return true; } return false; } NSObject* CCRotateBy::copyWithZone(cocos2d::NSZone *pZone) { NSZone* pNewZone = NULL; CCRotateBy* pCopy = NULL; if(pZone && pZone->m_pCopyObject) { //in case of being called at sub class pCopy = (CCRotateBy*)(pZone->m_pCopyObject); } else { pCopy = new CCRotateBy(); pZone = pNewZone = new NSZone(pCopy); } CCIntervalAction::copyWithZone(pZone); pCopy->initWithDuration(m_fDuration, m_fAngle); CCX_SAFE_DELETE(pNewZone); return pCopy; } void CCRotateBy::startWithTarget(CCNode *pTarget) { CCIntervalAction::startWithTarget(pTarget); m_fStartAngle = pTarget->getRotation(); } void CCRotateBy::update(cocos2d::ccTime time) { // XXX: shall I add % 360 m_pTarget->setRotation(m_fStartAngle + m_fAngle * time); } CCIntervalAction* CCRotateBy::reverse(void) { return CCRotateBy::actionWithDuration(m_fDuration, -m_fAngle); } // // MoveTo // CCMoveTo* CCMoveTo::actionWithDuration(cocos2d::ccTime duration, cocos2d::CGPoint position) { CCMoveTo *pMoveTo = new CCMoveTo(); pMoveTo->initWithDuration(duration, position); pMoveTo->autorelease(); return pMoveTo; } bool CCMoveTo::initWithDuration(cocos2d::ccTime duration, cocos2d::CGPoint position) { if (CCIntervalAction::initWithDuration(duration)) { m_endPosition = position; return true; } return false; } NSObject* CCMoveTo::copyWithZone(cocos2d::NSZone *pZone) { NSZone* pNewZone = NULL; CCMoveTo* pCopy = NULL; if(pZone && pZone->m_pCopyObject) { //in case of being called at sub class pCopy = (CCMoveTo*)(pZone->m_pCopyObject); } else { pCopy = new CCMoveTo(); pZone = pNewZone = new NSZone(pCopy); } CCIntervalAction::copyWithZone(pZone); pCopy->initWithDuration(m_fDuration, m_endPosition); CCX_SAFE_DELETE(pNewZone); return pCopy; } void CCMoveTo::startWithTarget(CCNode *pTarget) { CCIntervalAction::startWithTarget(pTarget); m_startPosition = pTarget->getPosition(); m_delta = ccpSub(m_endPosition, m_startPosition); } void CCMoveTo::update(cocos2d::ccTime time) { m_pTarget->setPosition(ccp(m_startPosition.x + m_delta.x * time, m_startPosition.y + m_delta.y * time)); } // // MoveBy // CCMoveBy* CCMoveBy::actionWithDuration(cocos2d::ccTime duration, cocos2d::CGPoint position) { CCMoveBy *pMoveBy = new CCMoveBy(); pMoveBy->initWithDuration(duration, position); pMoveBy->autorelease(); return pMoveBy; } bool CCMoveBy::initWithDuration(cocos2d::ccTime duration, cocos2d::CGPoint position) { if (CCIntervalAction::initWithDuration(duration)) { m_delta = position; return true; } return false; } NSObject* CCMoveBy::copyWithZone(cocos2d::NSZone *pZone) { NSZone* pNewZone = NULL; CCMoveBy* pCopy = NULL; if(pZone && pZone->m_pCopyObject) { //in case of being called at sub class pCopy = (CCMoveBy*)(pZone->m_pCopyObject); } else { pCopy = new CCMoveBy(); pZone = pNewZone = new NSZone(pCopy); } CCMoveTo::copyWithZone(pZone); pCopy->initWithDuration(m_fDuration, m_delta); CCX_SAFE_DELETE(pNewZone); return pCopy; } void CCMoveBy::startWithTarget(CCNode *pTarget) { CGPoint dTmp = m_delta; CCMoveTo::startWithTarget(pTarget); m_delta = dTmp; } CCIntervalAction* CCMoveBy::reverse(void) { return CCMoveBy::actionWithDuration(m_fDuration, ccp(-m_delta.x, -m_delta.y)); } // // JumpBy // CCJumpBy* CCJumpBy::actionWithDuration(cocos2d::ccTime duration, cocos2d::CGPoint position, cocos2d::ccTime height, int jumps) { CCJumpBy *pJumpBy = new CCJumpBy(); pJumpBy->initWithDuration(duration, position, height, jumps); pJumpBy->autorelease(); return pJumpBy; } bool CCJumpBy::initWithDuration(cocos2d::ccTime duration, cocos2d::CGPoint position, cocos2d::ccTime height, int jumps) { if (CCIntervalAction::initWithDuration(duration)) { m_delta = position; m_height = height; m_nJumps = jumps; return true; } return false; } NSObject* CCJumpBy::copyWithZone(cocos2d::NSZone *pZone) { NSZone* pNewZone = NULL; CCJumpBy* pCopy = NULL; if(pZone && pZone->m_pCopyObject) { //in case of being called at sub class pCopy = (CCJumpBy*)(pZone->m_pCopyObject); } else { pCopy = new CCJumpBy(); pZone = pNewZone = new NSZone(pCopy); } CCIntervalAction::copyWithZone(pZone); pCopy->initWithDuration(m_fDuration, m_delta, m_height, m_nJumps); CCX_SAFE_DELETE(pNewZone); return pCopy; } void CCJumpBy::startWithTarget(CCNode *pTarget) { CCIntervalAction::startWithTarget(pTarget); m_startPosition = pTarget->getPosition(); } void CCJumpBy::update(cocos2d::ccTime time) { // parabolic jump (since v0.8.2) ccTime frac = fmodf(time * m_nJumps, 1.0f); ccTime y = m_height * 4 * frac * (1 - frac); y += m_delta.y * time; ccTime x = m_delta.x * time; m_pTarget->setPosition(ccp(m_startPosition.x + x, m_startPosition.y + y)); } CCIntervalAction* CCJumpBy::reverse(void) { return CCJumpBy::actionWithDuration(m_fDuration, ccp(-m_delta.x, -m_delta.y), m_height, m_nJumps); } // // JumpTo // CCJumpTo* CCJumpTo::actionWithDuration(cocos2d::ccTime duration, cocos2d::CGPoint position, cocos2d::ccTime height, int jumps) { CCJumpTo *pJumpTo = new CCJumpTo(); pJumpTo->initWithDuration(duration, position, height, jumps); pJumpTo->autorelease(); return pJumpTo; } NSObject* CCJumpTo::copyWithZone(NSZone* pZone) { NSZone* pNewZone = NULL; CCJumpTo* pCopy = NULL; if(pZone && pZone->m_pCopyObject) { //in case of being called at sub class pCopy = (CCJumpTo*)(pZone->m_pCopyObject); } else { pCopy = new CCJumpTo(); pZone = pNewZone = new NSZone(pCopy); } CCJumpBy::copyWithZone(pZone); pCopy->initWithDuration(m_fDuration, m_delta, m_height, m_nJumps); CCX_SAFE_DELETE(pNewZone); return pCopy; } void CCJumpTo::startWithTarget(CCNode *pTarget) { CCJumpBy::startWithTarget(pTarget); m_delta = ccp(m_delta.x - m_startPosition.x, m_delta.y - m_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, ccTime 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 // CCBezierBy* CCBezierBy::actionWithDuration(cocos2d::ccTime t, cocos2d::ccBezierConfig c) { CCBezierBy *pBezierBy = new CCBezierBy(); pBezierBy->initWithDuration(t, c); pBezierBy->autorelease(); return pBezierBy; } bool CCBezierBy::initWithDuration(cocos2d::ccTime t, cocos2d::ccBezierConfig c) { if (CCIntervalAction::initWithDuration(t)) { m_sConfig = c; return true; } return false; } void CCBezierBy::startWithTarget(CCNode *pTarget) { CCIntervalAction::startWithTarget(pTarget); m_startPosition = pTarget->getPosition(); } NSObject* CCBezierBy::copyWithZone(cocos2d::NSZone *pZone) { NSZone* pNewZone = NULL; CCBezierBy* pCopy = NULL; if(pZone && pZone->m_pCopyObject) { //in case of being called at sub class pCopy = (CCBezierBy*)(pZone->m_pCopyObject); } else { pCopy = new CCBezierBy(); pZone = pNewZone = new NSZone(pCopy); } CCIntervalAction::copyWithZone(pZone); pCopy->initWithDuration(m_fDuration, m_sConfig); CCX_SAFE_DELETE(pNewZone); return pCopy; } void CCBezierBy::update(cocos2d::ccTime time) { float xa = 0; float xb = m_sConfig.controlPoint_1.x; float xc = m_sConfig.controlPoint_2.x; float xd = m_sConfig.endPosition.x; float ya = 0; float yb = m_sConfig.controlPoint_1.y; float yc = m_sConfig.controlPoint_2.y; float yd = m_sConfig.endPosition.y; float x = bezierat(xa, xb, xc, xd, time); float y = bezierat(ya, yb, yc, yd, time); m_pTarget->setPosition(ccpAdd(m_startPosition, ccp(x, y))); } CCIntervalAction* CCBezierBy::reverse(void) { ccBezierConfig r; r.endPosition = ccpNeg(m_sConfig.endPosition); r.controlPoint_1 = ccpAdd(m_sConfig.controlPoint_2, ccpNeg(m_sConfig.endPosition)); r.controlPoint_2 = ccpAdd(m_sConfig.controlPoint_1, ccpNeg(m_sConfig.endPosition)); CCBezierBy *pAction = CCBezierBy::actionWithDuration(m_fDuration, r); return pAction; } // // BezierTo // CCBezierTo* CCBezierTo::actionWithDuration(ccTime t, ccBezierConfig c) { CCBezierTo *pBezierTo = new CCBezierTo(); pBezierTo->initWithDuration(t, c); pBezierTo->autorelease(); return pBezierTo; } NSObject* CCBezierTo::copyWithZone(cocos2d::NSZone *pZone) { NSZone* pNewZone = NULL; CCBezierBy* pCopy = NULL; if(pZone && pZone->m_pCopyObject) { //in case of being called at sub class pCopy = (CCBezierTo*)(pZone->m_pCopyObject); } else { pCopy = new CCBezierTo(); pZone = pNewZone = new NSZone(pCopy); } CCBezierBy::copyWithZone(pZone); pCopy->initWithDuration(m_fDuration, m_sConfig); CCX_SAFE_DELETE(pNewZone); return pCopy; } void CCBezierTo::startWithTarget(CCNode *pTarget) { CCBezierBy::startWithTarget(pTarget); m_sConfig.controlPoint_1 = ccpSub(m_sConfig.controlPoint_1, m_startPosition); m_sConfig.controlPoint_2 = ccpSub(m_sConfig.controlPoint_2, m_startPosition); m_sConfig.endPosition = ccpSub(m_sConfig.endPosition, m_startPosition); } // // ScaleTo // CCScaleTo* CCScaleTo::actionWithDuration(cocos2d::ccTime duration, float s) { CCScaleTo *pScaleTo = new CCScaleTo(); pScaleTo->initWithDuration(duration, s); pScaleTo->autorelease(); return pScaleTo; } bool CCScaleTo::initWithDuration(cocos2d::ccTime duration, float s) { if (CCIntervalAction::initWithDuration(duration)) { m_fEndScaleX = s; m_fEndScaleY = s; return true; } return false; } CCScaleTo* CCScaleTo::actionWithDuration(cocos2d::ccTime duration, float sx, float sy) { CCScaleTo *pScaleTo = new CCScaleTo(); pScaleTo->initWithDuration(duration, sx, sy); pScaleTo->autorelease(); return pScaleTo; } bool CCScaleTo::initWithDuration(cocos2d::ccTime duration, float sx, float sy) { if (CCIntervalAction::initWithDuration(duration)) { m_fEndScaleX = sx; m_fEndScaleY = sy; return true; } return false; } NSObject* CCScaleTo::copyWithZone(cocos2d::NSZone *pZone) { NSZone* pNewZone = NULL; CCScaleTo* pCopy = NULL; if(pZone && pZone->m_pCopyObject) { //in case of being called at sub class pCopy = (CCScaleTo*)(pZone->m_pCopyObject); } else { pCopy = new CCScaleTo(); pZone = pNewZone = new NSZone(pCopy); } CCIntervalAction::copyWithZone(pZone); pCopy->initWithDuration(m_fDuration, m_fEndScaleX, m_fEndScaleY); CCX_SAFE_DELETE(pNewZone); return pCopy; } void CCScaleTo::startWithTarget(CCNode *pTarget) { CCIntervalAction::startWithTarget(pTarget); m_fStartScaleX = pTarget->getScaleX(); m_fStartScaleY = pTarget->getScaleY(); m_fDeltaX = m_fEndScaleX - m_fStartScaleX; m_fDeltaY = m_fEndScaleY - m_fStartScaleY; } void CCScaleTo::update(cocos2d::ccTime time) { m_pTarget->setScaleX(m_fStartScaleX + m_fDeltaX * time); m_pTarget->setScaleY(m_fStartScaleY + m_fDeltaY * time); } // // ScaleBy // CCScaleBy* CCScaleBy::actionWithDuration(cocos2d::ccTime duration, float s) { CCScaleBy *pScaleBy = new CCScaleBy(); pScaleBy->initWithDuration(duration, s); pScaleBy->autorelease(); return pScaleBy; } CCScaleBy* CCScaleBy::actionWithDuration(cocos2d::ccTime duration, float sx, float sy) { CCScaleBy *pScaleBy = new CCScaleBy(); pScaleBy->initWithDuration(duration, sx, sy); pScaleBy->autorelease(); return pScaleBy; } NSObject* CCScaleBy::copyWithZone(cocos2d::NSZone *pZone) { NSZone* pNewZone = NULL; CCScaleTo* pCopy = NULL; if(pZone && pZone->m_pCopyObject) { //in case of being called at sub class pCopy = (CCScaleBy*)(pZone->m_pCopyObject); } else { pCopy = new CCScaleBy(); pZone = pNewZone = new NSZone(pCopy); } CCScaleTo::copyWithZone(pZone); pCopy->initWithDuration(m_fDuration, m_fEndScaleX, m_fEndScaleY); CCX_SAFE_DELETE(pNewZone); return pCopy; } void CCScaleBy::startWithTarget(CCNode *pTarget) { CCScaleTo::startWithTarget(pTarget); m_fDeltaX = m_fStartScaleX * m_fEndScaleX - m_fStartScaleX; m_fDeltaY = m_fStartScaleY * m_fEndScaleY - m_fStartScaleY; } CCIntervalAction* CCScaleBy::reverse(void) { return CCScaleBy::actionWithDuration(m_fDuration, 1 / m_fEndScaleX, 1 / m_fEndScaleY); } // // Blink // CCBlink* CCBlink::actionWithDuration(cocos2d::ccTime duration, unsigned int uBlinks) { CCBlink *pBlink = new CCBlink(); pBlink->initWithDuration(duration, uBlinks); pBlink->autorelease(); return pBlink; } bool CCBlink::initWithDuration(cocos2d::ccTime duration, unsigned int uBlinks) { if (CCIntervalAction::initWithDuration(duration)) { m_nTimes = uBlinks; return true; } return false; } NSObject* CCBlink::copyWithZone(cocos2d::NSZone *pZone) { NSZone* pNewZone = NULL; CCBlink* pCopy = NULL; if(pZone && pZone->m_pCopyObject) { //in case of being called at sub class pCopy = (CCBlink*)(pZone->m_pCopyObject); } else { pCopy = new CCBlink(); pZone = pNewZone = new NSZone(pCopy); } CCIntervalAction::copyWithZone(pZone); pCopy->initWithDuration(m_fDuration, (unsigned int)m_nTimes); CCX_SAFE_DELETE(pNewZone); return pCopy; } void CCBlink::update(cocos2d::ccTime time) { ccTime slice = 1.0f / m_nTimes; ccTime m = fmodf(time, slice); m_pTarget->setIsVisible(m > slice / 2 ? true : false); } CCIntervalAction* CCBlink::reverse(void) { // return 'self' return CCBlink::actionWithDuration(m_fDuration, m_nTimes); } // // FadeIn // CCFadeIn* CCFadeIn::actionWithDuration(cocos2d::ccTime d) { CCFadeIn* pAction = new CCFadeIn(); pAction->initWithDuration(d); pAction->autorelease(); return pAction; } NSObject* CCFadeIn::copyWithZone(cocos2d::NSZone *pZone) { NSZone* pNewZone = NULL; CCFadeIn* pCopy = NULL; if(pZone && pZone->m_pCopyObject) { //in case of being called at sub class pCopy = (CCFadeIn*)(pZone->m_pCopyObject); } else { pCopy = new CCFadeIn(); pZone = pNewZone = new NSZone(pCopy); } CCIntervalAction::copyWithZone(pZone); CCX_SAFE_DELETE(pNewZone); return pCopy; } void CCFadeIn::update(cocos2d::ccTime time) { CCRGBAProtocol *pRGBAProtocol = m_pTarget->convertToRGBAProtocol(); if (pRGBAProtocol) { pRGBAProtocol->setOpacity((GLubyte)(255 * time)); } /*m_pTarget->setOpacity((GLubyte)(255 * time));*/ } CCIntervalAction* CCFadeIn::reverse(void) { return CCFadeOut::actionWithDuration(m_fDuration); } // // FadeOut // CCFadeOut* CCFadeOut::actionWithDuration(cocos2d::ccTime d) { CCFadeOut* pAction = new CCFadeOut(); pAction->initWithDuration(d); pAction->autorelease(); return pAction; } NSObject* CCFadeOut::copyWithZone(cocos2d::NSZone *pZone) { NSZone* pNewZone = NULL; CCFadeOut* pCopy = NULL; if(pZone && pZone->m_pCopyObject) { //in case of being called at sub class pCopy = (CCFadeOut*)(pZone->m_pCopyObject); } else { pCopy = new CCFadeOut(); pZone = pNewZone = new NSZone(pCopy); } CCIntervalAction::copyWithZone(pZone); CCX_SAFE_DELETE(pNewZone); return pCopy; } void CCFadeOut::update(cocos2d::ccTime time) { CCRGBAProtocol *pRGBAProtocol = m_pTarget->convertToRGBAProtocol(); if (pRGBAProtocol) { pRGBAProtocol->setOpacity(GLubyte(255 * (1 - time))); } /*m_pTarget->setOpacity(GLubyte(255 * (1 - time)));*/ } CCIntervalAction* CCFadeOut::reverse(void) { return CCFadeIn::actionWithDuration(m_fDuration); } // // FadeTo // CCFadeTo* CCFadeTo::actionWithDuration(cocos2d::ccTime duration, GLubyte opacity) { CCFadeTo *pFadeTo = new CCFadeTo(); pFadeTo->initWithDuration(duration, opacity); pFadeTo->autorelease(); return pFadeTo; } bool CCFadeTo::initWithDuration(cocos2d::ccTime duration, GLubyte opacity) { if (CCIntervalAction::initWithDuration(duration)) { m_toOpacity = opacity; return true; } return false; } NSObject* CCFadeTo::copyWithZone(cocos2d::NSZone *pZone) { NSZone* pNewZone = NULL; CCFadeTo* pCopy = NULL; if(pZone && pZone->m_pCopyObject) { //in case of being called at sub class pCopy = (CCFadeTo*)(pZone->m_pCopyObject); } else { pCopy = new CCFadeTo(); pZone = pNewZone = new NSZone(pCopy); } CCIntervalAction::copyWithZone(pZone); pCopy->initWithDuration(m_fDuration, m_toOpacity); CCX_SAFE_DELETE(pNewZone); return pCopy; } void CCFadeTo::startWithTarget(CCNode *pTarget) { CCIntervalAction::startWithTarget(pTarget); CCRGBAProtocol *pRGBAProtocol = pTarget->convertToRGBAProtocol(); if (pRGBAProtocol) { m_fromOpacity = pRGBAProtocol->getOpacity(); } /*m_fromOpacity = pTarget->getOpacity();*/ } void CCFadeTo::update(cocos2d::ccTime time) { CCRGBAProtocol *pRGBAProtocol = m_pTarget->convertToRGBAProtocol(); if (pRGBAProtocol) { pRGBAProtocol->setOpacity((GLubyte)(m_fromOpacity + (m_toOpacity - m_fromOpacity) * time)); } /*m_pTarget->setOpacity((GLubyte)(m_fromOpacity + (m_toOpacity - m_fromOpacity) * time));*/ } // // TintTo // CCTintTo* CCTintTo::actionWithDuration(cocos2d::ccTime duration, GLubyte red, GLubyte green, GLubyte blue) { CCTintTo *pTintTo = new CCTintTo(); pTintTo->initWithDuration(duration, red, green, blue); pTintTo->autorelease(); return pTintTo; } bool CCTintTo::initWithDuration(cocos2d::ccTime duration, GLubyte red, GLubyte green, GLubyte blue) { if (CCIntervalAction::initWithDuration(duration)) { m_to = ccc3(red, green, blue); return true; } return false; } NSObject* CCTintTo::copyWithZone(cocos2d::NSZone *pZone) { NSZone* pNewZone = NULL; CCTintTo* pCopy = NULL; if(pZone && pZone->m_pCopyObject) { //in case of being called at sub class pCopy = (CCTintTo*)(pZone->m_pCopyObject); } else { pCopy = new CCTintTo(); pZone = pNewZone = new NSZone(pCopy); } CCIntervalAction::copyWithZone(pZone); pCopy->initWithDuration(m_fDuration, m_to.r, m_to.g, m_to.b); CCX_SAFE_DELETE(pNewZone); return pCopy; } void CCTintTo::startWithTarget(CCNode *pTarget) { CCIntervalAction::startWithTarget(pTarget); CCRGBAProtocol *pRGBAProtocol = m_pTarget->convertToRGBAProtocol(); if (pRGBAProtocol) { m_from = pRGBAProtocol->getColor(); } /*m_from = pTarget->getColor();*/ } void CCTintTo::update(cocos2d::ccTime time) { CCRGBAProtocol *pRGBAProtocol = m_pTarget->convertToRGBAProtocol(); if (pRGBAProtocol) { pRGBAProtocol->setColor(ccc3(GLubyte(m_from.r + (m_to.r - m_from.r) * time), (GLbyte)(m_from.g + (m_to.g - m_from.g) * time), (GLbyte)(m_from.b + (m_to.b - m_from.b) * time))); } } // // TintBy // CCTintBy* CCTintBy::actionWithDuration(cocos2d::ccTime duration, GLshort deltaRed, GLshort deltaGreen, GLshort deltaBlue) { CCTintBy *pTintBy = new CCTintBy(); pTintBy->initWithDuration(duration, deltaRed, deltaGreen, deltaBlue); pTintBy->autorelease(); return pTintBy; } bool CCTintBy::initWithDuration(cocos2d::ccTime duration, GLshort deltaRed, GLshort deltaGreen, GLshort deltaBlue) { if (CCIntervalAction::initWithDuration(duration)) { m_deltaR = deltaRed; m_deltaG = deltaGreen; m_deltaB = deltaBlue; return true; } return false; } NSObject* CCTintBy::copyWithZone(cocos2d::NSZone *pZone) { NSZone* pNewZone = NULL; CCTintBy* pCopy = NULL; if(pZone && pZone->m_pCopyObject) { //in case of being called at sub class pCopy = (CCTintBy*)(pZone->m_pCopyObject); } else { pCopy = new CCTintBy(); pZone = pNewZone = new NSZone(pCopy); } CCIntervalAction::copyWithZone(pZone); pCopy->initWithDuration(m_fDuration, (GLubyte)m_deltaR, (GLubyte)m_deltaG, (GLubyte)m_deltaB); CCX_SAFE_DELETE(pNewZone); return pCopy; } void CCTintBy::startWithTarget(CCNode *pTarget) { CCIntervalAction::startWithTarget(pTarget); CCRGBAProtocol *pRGBAProtocol = pTarget->convertToRGBAProtocol(); if (pRGBAProtocol) { ccColor3B color = pRGBAProtocol->getColor(); m_fromR = color.r; m_fromG = color.g; m_fromB = color.b; } } void CCTintBy::update(cocos2d::ccTime time) { CCRGBAProtocol *pRGBAProtocol = m_pTarget->convertToRGBAProtocol(); if (pRGBAProtocol) { pRGBAProtocol->setColor(ccc3((GLubyte)(m_fromR + m_deltaR * time), (GLubyte)(m_fromG + m_deltaG * time), (GLubyte)(m_fromB + m_deltaB * time))); } } CCIntervalAction* CCTintBy::reverse(void) { return CCTintBy::actionWithDuration(m_fDuration, -m_deltaR, -m_deltaG, -m_deltaB); } // // DelayTime // CCDelayTime* CCDelayTime::actionWithDuration(cocos2d::ccTime d) { CCDelayTime* pAction = new CCDelayTime(); pAction->initWithDuration(d); pAction->autorelease(); return pAction; } NSObject* CCDelayTime::copyWithZone(cocos2d::NSZone *pZone) { NSZone* pNewZone = NULL; CCDelayTime* pCopy = NULL; if(pZone && pZone->m_pCopyObject) { //in case of being called at sub class pCopy = (CCDelayTime*)(pZone->m_pCopyObject); } else { pCopy = new CCDelayTime(); pZone = pNewZone = new NSZone(pCopy); } CCIntervalAction::copyWithZone(pZone); CCX_SAFE_DELETE(pNewZone); return pCopy; } void CCDelayTime::update(cocos2d::ccTime time) { return; } CCIntervalAction* CCDelayTime::reverse(void) { return CCDelayTime::actionWithDuration(m_fDuration); } // // ReverseTime // CCReverseTime* CCReverseTime::actionWithAction(cocos2d::CCFiniteTimeAction *pAction) { // casting to prevent warnings CCReverseTime *pReverseTime = new CCReverseTime(); pReverseTime->initWithAction(pAction); pReverseTime->autorelease(); return pReverseTime; } bool CCReverseTime::initWithAction(cocos2d::CCFiniteTimeAction *pAction) { if (CCIntervalAction::initWithDuration(pAction->getDuration())) { m_pOther = pAction; pAction->retain(); return true; } return false; } NSObject* CCReverseTime::copyWithZone(cocos2d::NSZone *pZone) { NSZone* pNewZone = NULL; CCReverseTime* pCopy = NULL; if(pZone && pZone->m_pCopyObject) { //in case of being called at sub class pCopy = (CCReverseTime*)(pZone->m_pCopyObject); } else { pCopy = new CCReverseTime(); pZone = pNewZone = new NSZone(pCopy); } CCIntervalAction::copyWithZone(pZone); pCopy->initWithAction((CCFiniteTimeAction*)(m_pOther->copy()->autorelease())); CCX_SAFE_DELETE(pNewZone); return pCopy; } CCReverseTime::~CCReverseTime(void) { m_pOther->release(); } void CCReverseTime::startWithTarget(CCNode *pTarget) { CCIntervalAction::startWithTarget(pTarget); m_pOther->startWithTarget(pTarget); } void CCReverseTime::stop(void) { m_pOther->stop(); CCIntervalAction::stop(); } void CCReverseTime::update(cocos2d::ccTime time) { m_pOther->update(1 - time); } CCIntervalAction* CCReverseTime::reverse(void) { return (CCIntervalAction*)(m_pOther->copy()->autorelease()); } // // Animate // CCAnimate* CCAnimate::actionWithAnimation(cocos2d::CCAnimation *pAnimation) { CCAnimate *pAnimate = new CCAnimate(); pAnimate->initWithAnimation(pAnimation, true); pAnimate->autorelease(); return pAnimate; } bool CCAnimate::initWithAnimation(cocos2d::CCAnimation *pAnimation) { assert(pAnimation != NULL); return initWithAnimation(pAnimation, true); } CCAnimate* CCAnimate::actionWithAnimation(cocos2d::CCAnimation *pAnimation, bool bRestoreOriginalFrame) { CCAnimate *pAnimate = new CCAnimate(); pAnimate->initWithAnimation(pAnimation, bRestoreOriginalFrame); pAnimate->autorelease(); return pAnimate; } bool CCAnimate::initWithAnimation(cocos2d::CCAnimation *pAnimation, bool bRestoreOriginalFrame) { assert(pAnimation); if (CCIntervalAction::initWithDuration(pAnimation->getFrames()->count() * pAnimation->getDelay())) { m_bRestoreOriginalFrame = bRestoreOriginalFrame; m_pAnimation = pAnimation; m_pAnimation->retain(); m_pOrigFrame = NULL; return true; } return false; } CCAnimate* CCAnimate::actionWithDuration(cocos2d::ccTime duration, cocos2d::CCAnimation *pAnimation, bool bRestoreOriginalFrame) { CCAnimate *pAnimate = new CCAnimate(); pAnimate->initWithDuration(duration, pAnimation, bRestoreOriginalFrame); pAnimate->autorelease(); return pAnimate; } bool CCAnimate::initWithDuration(cocos2d::ccTime duration, cocos2d::CCAnimation *pAnimation, bool bRestoreOriginalFrame) { assert(pAnimation != NULL); if (CCIntervalAction::initWithDuration(duration)) { m_bRestoreOriginalFrame = bRestoreOriginalFrame; m_pAnimation = pAnimation; m_pAnimation->retain(); m_pOrigFrame = NULL; return true; } return false; } NSObject* CCAnimate::copyWithZone(cocos2d::NSZone *pZone) { NSZone* pNewZone = NULL; CCAnimate* pCopy = NULL; if(pZone && pZone->m_pCopyObject) { //in case of being called at sub class pCopy = (CCAnimate*)(pZone->m_pCopyObject); } else { pCopy = new CCAnimate(); pZone = pNewZone = new NSZone(pCopy); } CCIntervalAction::copyWithZone(pZone); pCopy->initWithDuration(m_fDuration, m_pAnimation, m_bRestoreOriginalFrame); CCX_SAFE_DELETE(pNewZone); return pCopy; } CCAnimate::~CCAnimate(void) { CCX_SAFE_RELEASE(m_pAnimation); CCX_SAFE_RELEASE(m_pOrigFrame); } void CCAnimate::startWithTarget(CCNode *pTarget) { CCIntervalAction::startWithTarget(pTarget); CCSprite *pSprite = (CCSprite*)(pTarget); CCX_SAFE_RELEASE(m_pOrigFrame); if (m_bRestoreOriginalFrame) { m_pOrigFrame = pSprite->displayedFrame(); m_pOrigFrame->retain(); } } void CCAnimate::stop(void) { if (m_bRestoreOriginalFrame) { ((CCSprite*)(m_pTarget))->setDisplayFrame(m_pOrigFrame); } CCIntervalAction::stop(); } void CCAnimate::update(cocos2d::ccTime time) { NSMutableArray *pFrames = m_pAnimation->getFrames(); unsigned int numberOfFrames = pFrames->count(); unsigned int idx = (unsigned int)(time * numberOfFrames); if (idx >= numberOfFrames) { idx = numberOfFrames - 1; } CCSprite *pSprite = (CCSprite*)(m_pTarget); if (! pSprite->isFrameDisplayed(pFrames->getObjectAtIndex(idx))) { pSprite->setDisplayFrame(pFrames->getObjectAtIndex(idx)); } } CCIntervalAction* CCAnimate::reverse(void) { NSMutableArray *pOldArray = m_pAnimation->getFrames(); NSMutableArray *pNewArray = new NSMutableArray(pOldArray->count()); if (pOldArray->count() > 0) { CCSpriteFrame *pElement; NSMutableArray::NSMutableArrayRevIterator iter; for (iter = pOldArray->rbegin(); iter != pOldArray->rend(); iter++) { pElement = *iter; if (! pElement) { break; } pNewArray->addObject((CCSpriteFrame*)(pElement->copy()->autorelease())); } } CCAnimation *pNewAnim = CCAnimation::animationWithName(m_pAnimation->getName(), m_pAnimation->getDelay(), pNewArray); pNewArray->release(); return CCAnimate::actionWithDuration(m_fDuration, pNewAnim, m_bRestoreOriginalFrame); } }