2010-08-05 14:32:04 +08:00
|
|
|
/****************************************************************************
|
|
|
|
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 "CCAction.h"
|
2010-12-22 15:15:04 +08:00
|
|
|
#include "CCActionInterval.h"
|
2010-08-05 14:32:04 +08:00
|
|
|
#include "ccMacros.h"
|
|
|
|
#include "CCNode.h"
|
2010-08-25 10:19:20 +08:00
|
|
|
#include "CGPointExtension.h"
|
2010-08-05 14:32:04 +08:00
|
|
|
#include "CCDirector.h"
|
|
|
|
#include "NSZone.h"
|
|
|
|
|
|
|
|
namespace cocos2d {
|
|
|
|
//
|
|
|
|
// Action Base Class
|
|
|
|
//
|
|
|
|
|
|
|
|
CCAction::CCAction()
|
|
|
|
:m_pOriginalTarget(NULL)
|
|
|
|
,m_pTarget(NULL)
|
|
|
|
,m_nTag(kCCActionTagInvalid)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
CCAction::~CCAction()
|
|
|
|
{
|
|
|
|
CCLOGINFO("cocos2d: deallocing");
|
|
|
|
}
|
|
|
|
CCAction * CCAction::action()
|
|
|
|
{
|
|
|
|
CCAction * pRet = new CCAction();
|
|
|
|
pRet->autorelease();
|
|
|
|
return pRet;
|
|
|
|
}
|
|
|
|
|
2010-08-06 16:05:19 +08:00
|
|
|
char * CCAction::description()
|
2010-08-05 14:32:04 +08:00
|
|
|
{
|
2010-08-06 16:05:19 +08:00
|
|
|
char *ret = new char[100] ;
|
2010-09-02 16:46:25 +08:00
|
|
|
sprintf(ret,"<CCAction | Tag = %d>", m_nTag);
|
2010-08-05 14:32:04 +08:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
NSObject* CCAction::copyWithZone(NSZone *pZone)
|
|
|
|
{
|
|
|
|
NSZone *pNewZone = NULL;
|
|
|
|
CCAction *pRet = NULL;
|
|
|
|
if (pZone && pZone->m_pCopyObject)
|
|
|
|
{
|
2010-08-27 14:13:32 +08:00
|
|
|
pRet = (CCAction*)(pZone->m_pCopyObject);
|
2010-08-05 14:32:04 +08:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
pRet = new CCAction();
|
|
|
|
pZone = pNewZone = new NSZone(pRet);
|
|
|
|
}
|
|
|
|
//copy member data
|
|
|
|
pRet->m_nTag = m_nTag;
|
|
|
|
CCX_SAFE_DELETE(pNewZone);
|
|
|
|
return pRet;
|
|
|
|
}
|
|
|
|
|
2010-08-28 12:02:10 +08:00
|
|
|
void CCAction::startWithTarget(CCNode *aTarget)
|
2010-08-05 14:32:04 +08:00
|
|
|
{
|
|
|
|
m_pOriginalTarget = m_pTarget = aTarget;
|
|
|
|
}
|
|
|
|
|
|
|
|
void CCAction::stop()
|
|
|
|
{
|
|
|
|
m_pTarget = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CCAction::isDone()
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void CCAction::step(ccTime dt)
|
|
|
|
{
|
|
|
|
CCLOG("[Action step]. override me");
|
|
|
|
}
|
|
|
|
|
|
|
|
void CCAction::update(ccTime time)
|
|
|
|
{
|
|
|
|
CCLOG("[Action update]. override me");
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// FiniteTimeAction
|
|
|
|
//
|
|
|
|
|
|
|
|
CCFiniteTimeAction *CCFiniteTimeAction::reverse()
|
|
|
|
{
|
|
|
|
CCLOG("cocos2d: FiniteTimeAction#reverse: Implement me");
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// Speed
|
|
|
|
//
|
|
|
|
CCSpeed::~CCSpeed()
|
2010-08-28 18:00:38 +08:00
|
|
|
{
|
|
|
|
CCX_SAFE_RELEASE(m_pOther);
|
2010-08-05 14:32:04 +08:00
|
|
|
}
|
|
|
|
|
2010-12-22 15:43:54 +08:00
|
|
|
CCSpeed * CCSpeed::actionWithAction(CCActionInterval *pAction, float fRate)
|
2010-08-05 14:32:04 +08:00
|
|
|
{
|
|
|
|
CCSpeed *pRet = new CCSpeed();
|
2010-08-19 14:09:40 +08:00
|
|
|
if (pRet && pRet->initWithAction(pAction, fRate))
|
2010-08-05 14:32:04 +08:00
|
|
|
{
|
|
|
|
pRet->autorelease();
|
|
|
|
return pRet;
|
|
|
|
}
|
2010-08-19 14:09:40 +08:00
|
|
|
CCX_SAFE_DELETE(pRet)
|
2010-08-05 14:32:04 +08:00
|
|
|
return NULL;
|
2010-08-28 18:00:38 +08:00
|
|
|
}
|
2010-08-25 16:08:10 +08:00
|
|
|
|
2010-12-22 15:43:54 +08:00
|
|
|
bool CCSpeed::initWithAction(CCActionInterval *pAction, float fRate)
|
2010-08-05 14:32:04 +08:00
|
|
|
{
|
2010-09-04 12:02:52 +08:00
|
|
|
assert(pAction != NULL);
|
2010-08-06 11:58:03 +08:00
|
|
|
pAction->retain();
|
|
|
|
m_pOther = pAction;
|
2010-08-05 14:32:04 +08:00
|
|
|
m_fSpeed = fRate;
|
2010-09-04 12:02:52 +08:00
|
|
|
return true;
|
2010-08-28 18:00:38 +08:00
|
|
|
}
|
2010-08-25 16:08:10 +08:00
|
|
|
|
2010-08-05 14:32:04 +08:00
|
|
|
NSObject *CCSpeed::copyWithZone(NSZone *pZone)
|
|
|
|
{
|
|
|
|
NSZone* pNewZone = NULL;
|
|
|
|
CCSpeed* pRet = NULL;
|
|
|
|
if(pZone && pZone->m_pCopyObject) //in case of being called at sub class
|
|
|
|
{
|
2010-08-30 13:53:20 +08:00
|
|
|
pRet = (CCSpeed*)(pZone->m_pCopyObject);
|
2010-08-05 14:32:04 +08:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
pRet = new CCSpeed();
|
|
|
|
pZone = pNewZone = new NSZone(pRet);
|
|
|
|
}
|
2010-09-02 14:54:42 +08:00
|
|
|
CCAction::copyWithZone(pZone);
|
2010-08-28 18:00:38 +08:00
|
|
|
|
2010-12-22 15:43:54 +08:00
|
|
|
pRet->initWithAction( (CCActionInterval*)(m_pOther->copy()->autorelease()) , m_fSpeed );
|
2010-08-25 16:08:10 +08:00
|
|
|
|
2010-08-05 14:32:04 +08:00
|
|
|
CCX_SAFE_DELETE(pNewZone);
|
|
|
|
return pRet;
|
|
|
|
}
|
|
|
|
|
2010-08-28 12:02:10 +08:00
|
|
|
void CCSpeed::startWithTarget(CCNode* pTarget)
|
2010-08-05 14:32:04 +08:00
|
|
|
{
|
2010-09-02 14:54:42 +08:00
|
|
|
CCAction::startWithTarget(pTarget);
|
2010-08-28 18:00:38 +08:00
|
|
|
m_pOther->startWithTarget(pTarget);
|
2010-08-05 14:32:04 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void CCSpeed::stop()
|
2010-08-28 18:00:38 +08:00
|
|
|
{
|
|
|
|
m_pOther->stop();
|
2010-09-02 14:54:42 +08:00
|
|
|
CCAction::stop();
|
2010-08-05 14:32:04 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void CCSpeed::step(ccTime dt)
|
2010-08-28 18:00:38 +08:00
|
|
|
{
|
|
|
|
m_pOther->step(dt * m_fSpeed);
|
2010-08-05 14:32:04 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
bool CCSpeed::isDone()
|
2010-08-28 18:00:38 +08:00
|
|
|
{
|
|
|
|
return m_pOther->isDone();
|
2010-08-05 14:32:04 +08:00
|
|
|
}
|
|
|
|
|
2010-12-22 15:43:54 +08:00
|
|
|
CCActionInterval *CCSpeed::reverse()
|
2010-08-28 18:00:38 +08:00
|
|
|
{
|
2010-12-22 15:43:54 +08:00
|
|
|
return (CCActionInterval*)(CCSpeed::actionWithAction(m_pOther->reverse(), m_fSpeed));
|
2010-08-05 14:32:04 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// Follow
|
|
|
|
//
|
|
|
|
CCFollow::~CCFollow()
|
|
|
|
{
|
|
|
|
m_pobFollowedNode->release();
|
|
|
|
}
|
|
|
|
|
|
|
|
CCFollow *CCFollow::actionWithTarget(CCNode *pFollowedNode)
|
|
|
|
{
|
|
|
|
CCFollow *pRet = new CCFollow();
|
2010-08-19 14:09:40 +08:00
|
|
|
if (pRet && pRet->initWithTarget(pFollowedNode))
|
2010-08-05 14:32:04 +08:00
|
|
|
{
|
|
|
|
pRet->autorelease();
|
|
|
|
return pRet;
|
|
|
|
}
|
2010-08-19 14:09:40 +08:00
|
|
|
CCX_SAFE_DELETE(pRet)
|
2010-08-05 14:32:04 +08:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
CCFollow *CCFollow::actionWithTarget(CCNode *pFollowedNode, CGRect rect)
|
|
|
|
{
|
|
|
|
CCFollow *pRet = new CCFollow();
|
2010-08-19 14:09:40 +08:00
|
|
|
if (pRet && pRet->initWithTarget(pFollowedNode, rect))
|
2010-08-05 14:32:04 +08:00
|
|
|
{
|
|
|
|
pRet->autorelease();
|
|
|
|
return pRet;
|
|
|
|
}
|
2010-08-19 14:09:40 +08:00
|
|
|
CCX_SAFE_DELETE(pRet)
|
2010-08-05 14:32:04 +08:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2010-09-04 12:02:52 +08:00
|
|
|
bool CCFollow::initWithTarget(CCNode *pFollowedNode)
|
2010-08-05 14:32:04 +08:00
|
|
|
{
|
2010-09-04 12:02:52 +08:00
|
|
|
assert(pFollowedNode != NULL);
|
2010-08-05 14:32:04 +08:00
|
|
|
pFollowedNode->retain();
|
|
|
|
m_pobFollowedNode = pFollowedNode;
|
|
|
|
m_bBoundarySet = false;
|
|
|
|
m_bBoundaryFullyCovered = false;
|
|
|
|
|
2010-11-11 11:18:58 +08:00
|
|
|
CGSize winSize = CCDirector::sharedDirector()->getWinSize();
|
2010-08-05 14:32:04 +08:00
|
|
|
m_obFullScreenSize = CGPointMake(winSize.width, winSize.height);
|
|
|
|
m_obHalfScreenSize = ccpMult(m_obFullScreenSize, 0.5f);
|
2010-09-04 12:02:52 +08:00
|
|
|
return true;
|
2010-08-05 14:32:04 +08:00
|
|
|
}
|
|
|
|
|
2010-09-04 12:02:52 +08:00
|
|
|
bool CCFollow::initWithTarget(CCNode *pFollowedNode, CGRect rect)
|
2010-08-05 14:32:04 +08:00
|
|
|
{
|
2010-09-04 12:02:52 +08:00
|
|
|
assert(pFollowedNode != NULL);
|
2010-08-05 14:32:04 +08:00
|
|
|
pFollowedNode->retain();
|
|
|
|
m_pobFollowedNode = pFollowedNode;
|
|
|
|
m_bBoundarySet = true;
|
|
|
|
m_bBoundaryFullyCovered = false;
|
|
|
|
|
2010-11-11 11:18:58 +08:00
|
|
|
CGSize winSize = CCDirector::sharedDirector()->getWinSize();
|
2010-08-05 14:32:04 +08:00
|
|
|
m_obFullScreenSize = CGPointMake(winSize.width, winSize.height);
|
|
|
|
m_obHalfScreenSize = ccpMult(m_obFullScreenSize, 0.5f);
|
|
|
|
|
|
|
|
m_fLeftBoundary = -((rect.origin.x+rect.size.width) - m_obFullScreenSize.x);
|
|
|
|
m_fRightBoundary = -rect.origin.x ;
|
|
|
|
m_fTopBoundary = -rect.origin.y;
|
|
|
|
m_fBottomBoundary = -((rect.origin.y+rect.size.height) - m_obFullScreenSize.y);
|
|
|
|
|
|
|
|
if(m_fRightBoundary < m_fLeftBoundary)
|
|
|
|
{
|
|
|
|
// screen width is larger than world's boundary width
|
|
|
|
//set both in the middle of the world
|
|
|
|
m_fRightBoundary = m_fLeftBoundary = (m_fLeftBoundary + m_fRightBoundary) / 2;
|
|
|
|
}
|
|
|
|
if(m_fTopBoundary < m_fBottomBoundary)
|
|
|
|
{
|
|
|
|
// screen width is larger than world's boundary width
|
|
|
|
//set both in the middle of the world
|
|
|
|
m_fTopBoundary = m_fBottomBoundary = (m_fTopBoundary + m_fBottomBoundary) / 2;
|
|
|
|
}
|
|
|
|
|
|
|
|
if( (m_fTopBoundary == m_fBottomBoundary) && (m_fLeftBoundary == m_fRightBoundary) )
|
|
|
|
{
|
|
|
|
m_bBoundaryFullyCovered = true;
|
|
|
|
}
|
2010-09-04 12:02:52 +08:00
|
|
|
return true;
|
2010-08-05 14:32:04 +08:00
|
|
|
}
|
|
|
|
NSObject *CCFollow::copyWithZone(NSZone *pZone)
|
|
|
|
{
|
|
|
|
NSZone *pNewZone = NULL;
|
|
|
|
CCFollow *pRet = NULL;
|
|
|
|
if(pZone && pZone->m_pCopyObject) //in case of being called at sub class
|
|
|
|
{
|
2010-08-30 13:53:20 +08:00
|
|
|
pRet = (CCFollow*)(pZone->m_pCopyObject);
|
2010-08-05 14:32:04 +08:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
pRet = new CCFollow();
|
|
|
|
pZone = pNewZone = new NSZone(pRet);
|
|
|
|
}
|
2010-09-02 14:54:42 +08:00
|
|
|
CCAction::copyWithZone(pZone);
|
2010-08-05 14:32:04 +08:00
|
|
|
// copy member data
|
|
|
|
pRet->m_nTag = m_nTag;
|
|
|
|
CCX_SAFE_DELETE(pNewZone);
|
|
|
|
return pRet;
|
|
|
|
}
|
|
|
|
void CCFollow::step(ccTime dt)
|
|
|
|
{
|
|
|
|
#define CLAMP(x,y,z) MIN(MAX(x,y),z)
|
|
|
|
|
|
|
|
if(m_bBoundarySet)
|
|
|
|
{
|
|
|
|
// whole map fits inside a single screen, no need to modify the position - unless map boundaries are increased
|
|
|
|
if(m_bBoundaryFullyCovered)
|
|
|
|
return;
|
|
|
|
|
|
|
|
CGPoint tempPos = ccpSub( m_obHalfScreenSize, m_pobFollowedNode->getPosition());
|
|
|
|
|
2010-08-30 13:53:20 +08:00
|
|
|
m_pTarget->setPosition(ccp(CLAMP(tempPos.x, m_fLeftBoundary, m_fRightBoundary),
|
2010-08-05 14:32:04 +08:00
|
|
|
CLAMP(tempPos.y, m_fBottomBoundary, m_fTopBoundary)));
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2010-08-30 13:53:20 +08:00
|
|
|
m_pTarget->setPosition(ccpSub(m_obHalfScreenSize, m_pobFollowedNode->getPosition()));
|
2010-08-05 14:32:04 +08:00
|
|
|
}
|
|
|
|
#undef CLAMP
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CCFollow::isDone()
|
|
|
|
{
|
|
|
|
return ( !m_pobFollowedNode->getIsRunning() );
|
|
|
|
}
|
|
|
|
void CCFollow::stop()
|
|
|
|
{
|
|
|
|
m_pTarget = NULL;
|
2010-09-02 14:54:42 +08:00
|
|
|
CCAction::stop();
|
2010-08-05 14:32:04 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
}//namespace cocos2d
|
|
|
|
|
|
|
|
|