axmol/cocos2dx/actions/CCActionCatmullRom.cpp

566 lines
14 KiB
C++
Raw Normal View History

2012-06-08 13:55:28 +08:00
/*
* Copyright (c) 2010-2012 cocos2d-x.org
* cocos2d for iPhone: http://www.cocos2d-iphone.org
*
* Copyright (c) 2008 Radu Gruian
*
* Copyright (c) 2011 Vit Valentin
*
*
* 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.
*
*
* Original code by Radu Gruian: http://www.codeproject.com/Articles/30838/Overhauser-Catmull-Rom-Splines-for-Camera-Animatio.So
2012-06-08 13:55:28 +08:00
*
* Adapted to cocos2d-x by Vit Valentin
*
* Adapted from cocos2d-x to cocos2d-iphone by Ricardo Quesada
*/
#include "ccMacros.h"
#include "CCActionCatmullRom.h"
using namespace std;
2012-06-08 13:55:28 +08:00
NS_CC_BEGIN;
/*
* Implementation of PointArray
2012-06-08 13:55:28 +08:00
*/
PointArray* PointArray::create(unsigned int capacity)
2012-06-08 13:55:28 +08:00
{
PointArray* ret = new PointArray();
2012-06-08 13:55:28 +08:00
if (ret)
{
if (ret->initWithCapacity(capacity))
{
ret->autorelease();
}
else
{
delete ret;
ret = NULL;
}
}
2012-06-08 13:55:28 +08:00
return ret;
}
bool PointArray::initWithCapacity(unsigned int capacity)
2012-06-08 13:55:28 +08:00
{
_controlPoints = new vector<Point*>();
2012-06-08 13:55:28 +08:00
return true;
}
PointArray* PointArray::clone() const
2012-06-08 13:55:28 +08:00
{
vector<Point*> *newArray = new vector<Point*>();
vector<Point*>::iterator iter;
for (iter = _controlPoints->begin(); iter != _controlPoints->end(); ++iter)
{
newArray->push_back(new Point((*iter)->x, (*iter)->y));
}
PointArray *points = new PointArray();
points->initWithCapacity(10);
2012-06-08 13:55:28 +08:00
points->setControlPoints(newArray);
points->autorelease();
2012-06-08 13:55:28 +08:00
return points;
}
PointArray::~PointArray()
2012-06-08 13:55:28 +08:00
{
vector<Point*>::iterator iter;
for (iter = _controlPoints->begin(); iter != _controlPoints->end(); ++iter)
{
delete *iter;
}
delete _controlPoints;
2012-06-08 13:55:28 +08:00
}
PointArray::PointArray() :_controlPoints(NULL){}
2012-06-08 13:55:28 +08:00
const std::vector<Point*>* PointArray::getControlPoints() const
2012-06-08 13:55:28 +08:00
{
return _controlPoints;
}
void PointArray::setControlPoints(vector<Point*> *controlPoints)
{
CCASSERT(controlPoints != NULL, "control points should not be NULL");
// delete old points
vector<Point*>::iterator iter;
for (iter = _controlPoints->begin(); iter != _controlPoints->end(); ++iter)
{
delete *iter;
}
delete _controlPoints;
_controlPoints = controlPoints;
}
void PointArray::addControlPoint(Point controlPoint)
{
_controlPoints->push_back(new Point(controlPoint.x, controlPoint.y));
2012-06-08 13:55:28 +08:00
}
void PointArray::insertControlPoint(Point &controlPoint, unsigned int index)
2012-06-08 13:55:28 +08:00
{
Point *temp = new Point(controlPoint.x, controlPoint.y);
_controlPoints->insert(_controlPoints->begin() + index, temp);
2012-06-08 13:55:28 +08:00
}
Point PointArray::getControlPointAtIndex(unsigned int index)
2012-06-08 13:55:28 +08:00
{
index = MIN(_controlPoints->size()-1, MAX(index, 0));
return *(_controlPoints->at(index));
2012-06-08 13:55:28 +08:00
}
void PointArray::replaceControlPoint(cocos2d::Point &controlPoint, unsigned int index)
2012-06-08 13:55:28 +08:00
{
Point *temp = _controlPoints->at(index);
temp->x = controlPoint.x;
temp->y = controlPoint.y;
2012-06-08 13:55:28 +08:00
}
void PointArray::removeControlPointAtIndex(unsigned int index)
2012-06-08 13:55:28 +08:00
{
vector<Point*>::iterator iter = _controlPoints->begin() + index;
Point* pRemovedPoint = *iter;
_controlPoints->erase(iter);
delete pRemovedPoint;
2012-06-08 13:55:28 +08:00
}
unsigned int PointArray::count() const
2012-06-08 13:55:28 +08:00
{
return _controlPoints->size();
2012-06-08 13:55:28 +08:00
}
PointArray* PointArray::reverse() const
2012-06-08 13:55:28 +08:00
{
vector<Point*> *newArray = new vector<Point*>();
vector<Point*>::reverse_iterator iter;
Point *point = NULL;
for (iter = _controlPoints->rbegin(); iter != _controlPoints->rend(); ++iter)
2012-06-08 13:55:28 +08:00
{
point = *iter;
newArray->push_back(new Point(point->x, point->y));
2012-06-08 13:55:28 +08:00
}
PointArray *config = PointArray::create(0);
2012-06-08 13:55:28 +08:00
config->setControlPoints(newArray);
return config;
}
void PointArray::reverseInline()
2012-06-08 13:55:28 +08:00
{
unsigned long l = _controlPoints->size();
Point *p1 = NULL;
Point *p2 = NULL;
int x, y;
2012-06-08 13:55:28 +08:00
for (unsigned int i = 0; i < l/2; ++i)
{
p1 = _controlPoints->at(i);
p2 = _controlPoints->at(l-i-1);
x = p1->x;
y = p1->y;
p1->x = p2->x;
p1->y = p2->y;
p2->x = x;
p2->y = y;
2012-06-08 13:55:28 +08:00
}
}
// CatmullRom Spline formula:
Point ccCardinalSplineAt(Point &p0, Point &p1, Point &p2, Point &p3, float tension, float t)
2012-06-08 13:55:28 +08:00
{
float t2 = t * t;
float t3 = t2 * t;
/*
* Formula: s(-ttt + 2tt - t)P1 + s(-ttt + tt)P2 + (2ttt - 3tt + 1)P2 + s(ttt - 2tt + t)P3 + (-2ttt + 3tt)P3 + s(ttt - tt)P4
2012-06-08 13:55:28 +08:00
*/
float s = (1 - tension) / 2;
float b1 = s * ((-t3 + (2 * t2)) - t); // s(-t3 + 2 t2 - t)P1
float b2 = s * (-t3 + t2) + (2 * t3 - 3 * t2 + 1); // s(-t3 + t2)P2 + (2 t3 - 3 t2 + 1)P2
float b3 = s * (t3 - 2 * t2 + t) + (-2 * t3 + 3 * t2); // s(t3 - 2 t2 + t)P3 + (-2 t3 + 3 t2)P3
float b4 = s * (t3 - t2); // s(t3 - t2)P4
2012-06-08 13:55:28 +08:00
float x = (p0.x*b1 + p1.x*b2 + p2.x*b3 + p3.x*b4);
float y = (p0.y*b1 + p1.y*b2 + p2.y*b3 + p3.y*b4);
2013-07-12 14:11:55 +08:00
return Point(x,y);
2012-06-08 13:55:28 +08:00
}
/* Implementation of CardinalSplineTo
2012-06-08 13:55:28 +08:00
*/
CardinalSplineTo* CardinalSplineTo::create(float duration, cocos2d::PointArray *points, float tension)
2012-06-08 13:55:28 +08:00
{
CardinalSplineTo *ret = new CardinalSplineTo();
2012-06-08 13:55:28 +08:00
if (ret)
{
if (ret->initWithDuration(duration, points, tension))
{
ret->autorelease();
}
else
{
CC_SAFE_RELEASE_NULL(ret);
}
}
2012-06-08 13:55:28 +08:00
return ret;
}
bool CardinalSplineTo::initWithDuration(float duration, cocos2d::PointArray *points, float tension)
2012-06-08 13:55:28 +08:00
{
CCASSERT(points->count() > 0, "Invalid configuration. It must at least have one control point");
2012-06-08 13:55:28 +08:00
if (ActionInterval::initWithDuration(duration))
2012-06-08 13:55:28 +08:00
{
this->setPoints(points);
this->_tension = tension;
2012-06-08 13:55:28 +08:00
return true;
}
return false;
}
CardinalSplineTo::~CardinalSplineTo()
2012-06-08 13:55:28 +08:00
{
CC_SAFE_RELEASE_NULL(_points);
2012-06-08 13:55:28 +08:00
}
CardinalSplineTo::CardinalSplineTo()
: _points(NULL)
, _deltaT(0.f)
, _tension(0.f)
2012-06-08 13:55:28 +08:00
{
}
void CardinalSplineTo::startWithTarget(cocos2d::Node *target)
2012-06-08 13:55:28 +08:00
{
ActionInterval::startWithTarget(target);
2012-06-08 13:55:28 +08:00
// _deltaT = (float) 1 / _points->count();
2012-11-14 18:05:15 +08:00
// Issue #1441
_deltaT = (float) 1 / (_points->count() - 1);
_previousPosition = target->getPosition();
_accumulatedDiff = Point::ZERO;
2012-06-08 13:55:28 +08:00
}
CardinalSplineTo* CardinalSplineTo::clone() const
{
// no copy constructor
auto a = new CardinalSplineTo();
a->initWithDuration(this->_duration, this->_points->clone(), this->_tension);
a->autorelease();
return a;
}
void CardinalSplineTo::update(float time)
2012-06-08 13:55:28 +08:00
{
unsigned int p;
float lt;
2012-11-14 18:05:15 +08:00
// eg.
// p..p..p..p..p..p..p
// 1..2..3..4..5..6..7
// want p to be 1, 2, 3, 4, 5, 6
2012-06-08 13:55:28 +08:00
if (time == 1)
{
p = _points->count() - 1;
2012-06-08 13:55:28 +08:00
lt = 1;
}
else
{
p = time / _deltaT;
lt = (time - _deltaT * (float)p) / _deltaT;
2012-06-08 13:55:28 +08:00
}
// Interpolate
Point pp0 = _points->getControlPointAtIndex(p-1);
Point pp1 = _points->getControlPointAtIndex(p+0);
Point pp2 = _points->getControlPointAtIndex(p+1);
Point pp3 = _points->getControlPointAtIndex(p+2);
2012-06-08 13:55:28 +08:00
Point newPos = ccCardinalSplineAt(pp0, pp1, pp2, pp3, _tension, lt);
2012-06-08 13:55:28 +08:00
2013-02-27 09:38:30 +08:00
#if CC_ENABLE_STACKABLE_ACTIONS
// Support for stacked actions
Node *node = _target;
Point diff = node->getPosition() - _previousPosition;
if( diff.x !=0 || diff.y != 0 ) {
_accumulatedDiff = _accumulatedDiff + diff;
newPos = newPos + _accumulatedDiff;
}
2013-02-27 09:38:30 +08:00
#endif
2012-06-08 13:55:28 +08:00
this->updatePosition(newPos);
}
void CardinalSplineTo::updatePosition(cocos2d::Point &newPos)
2012-06-08 13:55:28 +08:00
{
_target->setPosition(newPos);
_previousPosition = newPos;
2012-06-08 13:55:28 +08:00
}
CardinalSplineTo* CardinalSplineTo::reverse() const
2012-06-08 13:55:28 +08:00
{
PointArray *pReverse = _points->reverse();
2012-06-08 13:55:28 +08:00
return CardinalSplineTo::create(_duration, pReverse, _tension);
2012-06-08 13:55:28 +08:00
}
/* CardinalSplineBy
2012-06-08 13:55:28 +08:00
*/
CardinalSplineBy* CardinalSplineBy::create(float duration, cocos2d::PointArray *points, float tension)
2012-06-08 13:55:28 +08:00
{
CardinalSplineBy *ret = new CardinalSplineBy();
2012-06-08 13:55:28 +08:00
if (ret)
{
if (ret->initWithDuration(duration, points, tension))
{
ret->autorelease();
}
else
{
CC_SAFE_RELEASE_NULL(ret);
}
}
2012-06-08 13:55:28 +08:00
return ret;
}
CardinalSplineBy::CardinalSplineBy() : _startPosition(0,0)
2012-06-08 13:55:28 +08:00
{
}
void CardinalSplineBy::updatePosition(cocos2d::Point &newPos)
2012-06-08 13:55:28 +08:00
{
Point p = newPos + _startPosition;
_target->setPosition(p);
_previousPosition = p;
2012-06-08 13:55:28 +08:00
}
CardinalSplineBy* CardinalSplineBy::reverse() const
2012-06-08 13:55:28 +08:00
{
PointArray *copyConfig = _points->clone();
2012-06-08 13:55:28 +08:00
//
// convert "absolutes" to "diffs"
//
Point p = copyConfig->getControlPointAtIndex(0);
2012-06-08 13:55:28 +08:00
for (unsigned int i = 1; i < copyConfig->count(); ++i)
{
Point current = copyConfig->getControlPointAtIndex(i);
Point diff = current - p;
2012-06-08 13:55:28 +08:00
copyConfig->replaceControlPoint(diff, i);
p = current;
}
// convert to "diffs" to "reverse absolute"
PointArray *pReverse = copyConfig->reverse();
2012-06-08 13:55:28 +08:00
// 1st element (which should be 0,0) should be here too
p = pReverse->getControlPointAtIndex(pReverse->count()-1);
pReverse->removeControlPointAtIndex(pReverse->count()-1);
2012-06-08 13:55:28 +08:00
p = -p;
pReverse->insertControlPoint(p, 0);
2012-06-08 13:55:28 +08:00
for (unsigned int i = 1; i < pReverse->count(); ++i)
2012-06-08 13:55:28 +08:00
{
Point current = pReverse->getControlPointAtIndex(i);
current = -current;
Point abs = current + p;
pReverse->replaceControlPoint(abs, i);
2012-06-08 13:55:28 +08:00
p = abs;
}
return CardinalSplineBy::create(_duration, pReverse, _tension);
2012-06-08 13:55:28 +08:00
}
void CardinalSplineBy::startWithTarget(cocos2d::Node *target)
2012-06-08 13:55:28 +08:00
{
CardinalSplineTo::startWithTarget(target);
_startPosition = target->getPosition();
2012-06-08 13:55:28 +08:00
}
CardinalSplineBy* CardinalSplineBy::clone() const
{
// no copy constructor
auto a = new CardinalSplineBy();
a->initWithDuration(this->_duration, this->_points->clone(), this->_tension);
a->autorelease();
return a;
}
/* CatmullRomTo
2012-06-08 13:55:28 +08:00
*/
CatmullRomTo* CatmullRomTo::create(float dt, cocos2d::PointArray *points)
2012-06-08 13:55:28 +08:00
{
CatmullRomTo *ret = new CatmullRomTo();
2012-06-08 13:55:28 +08:00
if (ret)
{
if (ret->initWithDuration(dt, points))
{
ret->autorelease();
}
else
{
CC_SAFE_RELEASE_NULL(ret);
}
}
2012-06-08 13:55:28 +08:00
return ret;
}
bool CatmullRomTo::initWithDuration(float dt, cocos2d::PointArray *points)
2012-06-08 13:55:28 +08:00
{
if (CardinalSplineTo::initWithDuration(dt, points, 0.5f))
2012-06-08 13:55:28 +08:00
{
return true;
}
return false;
}
CatmullRomTo* CatmullRomTo::clone() const
{
// no copy constructor
auto a = new CatmullRomTo();
a->initWithDuration(this->_duration, this->_points->clone());
a->autorelease();
return a;
}
CatmullRomTo* CatmullRomTo::reverse() const
{
PointArray *pReverse = _points->reverse();
return CatmullRomTo::create(_duration, pReverse);
}
/* CatmullRomBy
2012-06-08 13:55:28 +08:00
*/
CatmullRomBy* CatmullRomBy::create(float dt, cocos2d::PointArray *points)
2012-06-08 13:55:28 +08:00
{
CatmullRomBy *ret = new CatmullRomBy();
2012-06-08 13:55:28 +08:00
if (ret)
{
if (ret->initWithDuration(dt, points))
{
ret->autorelease();
}
else
{
CC_SAFE_RELEASE_NULL(ret);
}
}
2012-06-08 13:55:28 +08:00
return ret;
}
bool CatmullRomBy::initWithDuration(float dt, cocos2d::PointArray *points)
2012-06-08 13:55:28 +08:00
{
if (CardinalSplineTo::initWithDuration(dt, points, 0.5f))
2012-06-08 13:55:28 +08:00
{
return true;
}
return false;
}
CatmullRomBy* CatmullRomBy::clone() const
{
// no copy constructor
auto a = new CatmullRomBy();
a->initWithDuration(this->_duration, this->_points->clone());
a->autorelease();
return a;
}
CatmullRomBy* CatmullRomBy::reverse() const
{
PointArray *copyConfig = _points->clone();
//
// convert "absolutes" to "diffs"
//
Point p = copyConfig->getControlPointAtIndex(0);
for (unsigned int i = 1; i < copyConfig->count(); ++i)
{
Point current = copyConfig->getControlPointAtIndex(i);
Point diff = current - p;
copyConfig->replaceControlPoint(diff, i);
p = current;
}
// convert to "diffs" to "reverse absolute"
PointArray *pReverse = copyConfig->reverse();
// 1st element (which should be 0,0) should be here too
p = pReverse->getControlPointAtIndex(pReverse->count()-1);
pReverse->removeControlPointAtIndex(pReverse->count()-1);
p = -p;
pReverse->insertControlPoint(p, 0);
for (unsigned int i = 1; i < pReverse->count(); ++i)
{
Point current = pReverse->getControlPointAtIndex(i);
current = -current;
Point abs = current + p;
pReverse->replaceControlPoint(abs, i);
p = abs;
}
return CatmullRomBy::create(_duration, pReverse);
}
2012-06-08 13:55:28 +08:00
NS_CC_END;