2019-11-23 20:27:39 +08:00
|
|
|
/****************************************************************************
|
|
|
|
Copyright (c) 2009 On-Core
|
|
|
|
Copyright (c) 2010-2012 cocos2d-x.org
|
|
|
|
Copyright (c) 2013-2016 Chukong Technologies Inc.
|
|
|
|
Copyright (c) 2017-2018 Xiamen Yaji Software Co., Ltd.
|
2021-12-25 10:04:45 +08:00
|
|
|
|
2019-11-23 20:27:39 +08:00
|
|
|
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 "2d/CCActionTiledGrid.h"
|
|
|
|
#include "2d/CCGrid.h"
|
|
|
|
#include "2d/CCNodeGrid.h"
|
|
|
|
#include "base/CCDirector.h"
|
|
|
|
#include "base/ccMacros.h"
|
|
|
|
|
|
|
|
NS_CC_BEGIN
|
|
|
|
|
|
|
|
struct Tile
|
|
|
|
{
|
2021-12-25 10:04:45 +08:00
|
|
|
Vec2 position;
|
|
|
|
Vec2 startPosition;
|
|
|
|
Vec2 delta;
|
2019-11-23 20:27:39 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
// implementation of ShakyTiles3D
|
|
|
|
|
2021-10-23 23:27:14 +08:00
|
|
|
ShakyTiles3D* ShakyTiles3D::create(float duration, const Vec2& gridSize, int range, bool shakeZ)
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
2021-12-25 10:04:45 +08:00
|
|
|
ShakyTiles3D* action = new ShakyTiles3D();
|
2019-11-23 20:27:39 +08:00
|
|
|
|
2021-12-08 00:11:53 +08:00
|
|
|
if (action->initWithDuration(duration, gridSize, range, shakeZ))
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
|
|
|
action->autorelease();
|
|
|
|
return action;
|
|
|
|
}
|
|
|
|
|
|
|
|
delete action;
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
2021-10-23 23:27:14 +08:00
|
|
|
bool ShakyTiles3D::initWithDuration(float duration, const Vec2& gridSize, int range, bool shakeZ)
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
|
|
|
if (TiledGrid3DAction::initWithDuration(duration, gridSize))
|
|
|
|
{
|
|
|
|
_randrange = range;
|
2021-12-25 10:04:45 +08:00
|
|
|
_shakeZ = shakeZ;
|
2019-11-23 20:27:39 +08:00
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
ShakyTiles3D* ShakyTiles3D::clone() const
|
|
|
|
{
|
|
|
|
// no copy constructor
|
|
|
|
return ShakyTiles3D::create(_duration, _gridSize, _randrange, _shakeZ);
|
|
|
|
}
|
|
|
|
|
|
|
|
void ShakyTiles3D::update(float /*time*/)
|
|
|
|
{
|
|
|
|
int i, j;
|
|
|
|
|
|
|
|
for (i = 0; i < _gridSize.width; ++i)
|
|
|
|
{
|
|
|
|
for (j = 0; j < _gridSize.height; ++j)
|
|
|
|
{
|
|
|
|
Vec2 pos((float)i, (float)j);
|
|
|
|
Quad3 coords = getOriginalTile(pos);
|
|
|
|
|
|
|
|
// X
|
2021-12-25 10:04:45 +08:00
|
|
|
coords.bl.x += (rand() % (_randrange * 2)) - _randrange;
|
|
|
|
coords.br.x += (rand() % (_randrange * 2)) - _randrange;
|
|
|
|
coords.tl.x += (rand() % (_randrange * 2)) - _randrange;
|
|
|
|
coords.tr.x += (rand() % (_randrange * 2)) - _randrange;
|
2019-11-23 20:27:39 +08:00
|
|
|
|
|
|
|
// Y
|
2021-12-25 10:04:45 +08:00
|
|
|
coords.bl.y += (rand() % (_randrange * 2)) - _randrange;
|
|
|
|
coords.br.y += (rand() % (_randrange * 2)) - _randrange;
|
|
|
|
coords.tl.y += (rand() % (_randrange * 2)) - _randrange;
|
|
|
|
coords.tr.y += (rand() % (_randrange * 2)) - _randrange;
|
2019-11-23 20:27:39 +08:00
|
|
|
|
|
|
|
if (_shakeZ)
|
|
|
|
{
|
2021-12-25 10:04:45 +08:00
|
|
|
coords.bl.z += (rand() % (_randrange * 2)) - _randrange;
|
|
|
|
coords.br.z += (rand() % (_randrange * 2)) - _randrange;
|
|
|
|
coords.tl.z += (rand() % (_randrange * 2)) - _randrange;
|
|
|
|
coords.tr.z += (rand() % (_randrange * 2)) - _randrange;
|
2019-11-23 20:27:39 +08:00
|
|
|
}
|
2021-12-25 10:04:45 +08:00
|
|
|
|
2019-11-23 20:27:39 +08:00
|
|
|
setTile(pos, coords);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// implementation of ShatteredTiles3D
|
|
|
|
|
2021-10-23 23:27:14 +08:00
|
|
|
ShatteredTiles3D* ShatteredTiles3D::create(float duration, const Vec2& gridSize, int range, bool shatterZ)
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
2021-12-25 10:04:45 +08:00
|
|
|
ShatteredTiles3D* action = new ShatteredTiles3D();
|
2019-11-23 20:27:39 +08:00
|
|
|
|
2021-12-08 00:11:53 +08:00
|
|
|
if (action->initWithDuration(duration, gridSize, range, shatterZ))
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
|
|
|
action->autorelease();
|
|
|
|
return action;
|
|
|
|
}
|
|
|
|
|
|
|
|
delete action;
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
2021-10-23 23:27:14 +08:00
|
|
|
bool ShatteredTiles3D::initWithDuration(float duration, const Vec2& gridSize, int range, bool shatterZ)
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
|
|
|
if (TiledGrid3DAction::initWithDuration(duration, gridSize))
|
|
|
|
{
|
2021-12-25 10:04:45 +08:00
|
|
|
_once = false;
|
2019-11-23 20:27:39 +08:00
|
|
|
_randrange = range;
|
2021-12-25 10:04:45 +08:00
|
|
|
_shatterZ = shatterZ;
|
2019-11-23 20:27:39 +08:00
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
ShatteredTiles3D* ShatteredTiles3D::clone() const
|
|
|
|
{
|
|
|
|
// no copy constructor
|
|
|
|
return ShatteredTiles3D::create(_duration, _gridSize, _randrange, _shatterZ);
|
|
|
|
}
|
|
|
|
|
|
|
|
void ShatteredTiles3D::update(float /*time*/)
|
|
|
|
{
|
|
|
|
int i, j;
|
|
|
|
|
|
|
|
if (_once == false)
|
|
|
|
{
|
|
|
|
for (i = 0; i < _gridSize.width; ++i)
|
|
|
|
{
|
|
|
|
for (j = 0; j < _gridSize.height; ++j)
|
|
|
|
{
|
|
|
|
Vec2 pos((float)i, (float)j);
|
|
|
|
Quad3 coords = getOriginalTile(pos);
|
2021-12-25 10:04:45 +08:00
|
|
|
|
2019-11-23 20:27:39 +08:00
|
|
|
// X
|
2021-12-25 10:04:45 +08:00
|
|
|
coords.bl.x += (rand() % (_randrange * 2)) - _randrange;
|
|
|
|
coords.br.x += (rand() % (_randrange * 2)) - _randrange;
|
|
|
|
coords.tl.x += (rand() % (_randrange * 2)) - _randrange;
|
|
|
|
coords.tr.x += (rand() % (_randrange * 2)) - _randrange;
|
|
|
|
|
2019-11-23 20:27:39 +08:00
|
|
|
// Y
|
2021-12-25 10:04:45 +08:00
|
|
|
coords.bl.y += (rand() % (_randrange * 2)) - _randrange;
|
|
|
|
coords.br.y += (rand() % (_randrange * 2)) - _randrange;
|
|
|
|
coords.tl.y += (rand() % (_randrange * 2)) - _randrange;
|
|
|
|
coords.tr.y += (rand() % (_randrange * 2)) - _randrange;
|
2019-11-23 20:27:39 +08:00
|
|
|
|
2021-12-25 10:04:45 +08:00
|
|
|
if (_shatterZ)
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
2021-12-25 10:04:45 +08:00
|
|
|
coords.bl.z += (rand() % (_randrange * 2)) - _randrange;
|
|
|
|
coords.br.z += (rand() % (_randrange * 2)) - _randrange;
|
|
|
|
coords.tl.z += (rand() % (_randrange * 2)) - _randrange;
|
|
|
|
coords.tr.z += (rand() % (_randrange * 2)) - _randrange;
|
2019-11-23 20:27:39 +08:00
|
|
|
}
|
2021-12-25 10:04:45 +08:00
|
|
|
|
2019-11-23 20:27:39 +08:00
|
|
|
setTile(pos, coords);
|
|
|
|
}
|
|
|
|
}
|
2021-12-25 10:04:45 +08:00
|
|
|
|
2019-11-23 20:27:39 +08:00
|
|
|
_once = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// implementation of ShuffleTiles
|
|
|
|
|
2021-10-23 23:27:14 +08:00
|
|
|
ShuffleTiles* ShuffleTiles::create(float duration, const Vec2& gridSize, unsigned int seed)
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
2021-12-25 10:04:45 +08:00
|
|
|
ShuffleTiles* action = new ShuffleTiles();
|
2019-11-23 20:27:39 +08:00
|
|
|
|
2021-12-08 00:11:53 +08:00
|
|
|
if (action->initWithDuration(duration, gridSize, seed))
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
|
|
|
action->autorelease();
|
|
|
|
return action;
|
|
|
|
}
|
|
|
|
|
|
|
|
delete action;
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
2021-10-23 23:27:14 +08:00
|
|
|
bool ShuffleTiles::initWithDuration(float duration, const Vec2& gridSize, unsigned int seed)
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
|
|
|
if (TiledGrid3DAction::initWithDuration(duration, gridSize))
|
|
|
|
{
|
2021-12-25 10:04:45 +08:00
|
|
|
_seed = seed;
|
2019-11-23 20:27:39 +08:00
|
|
|
_tilesOrder = nullptr;
|
2021-12-25 10:04:45 +08:00
|
|
|
_tiles = nullptr;
|
2019-11-23 20:27:39 +08:00
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
ShuffleTiles* ShuffleTiles::clone() const
|
|
|
|
{
|
|
|
|
// no copy constructor
|
|
|
|
return ShuffleTiles::create(_duration, _gridSize, _seed);
|
|
|
|
}
|
|
|
|
|
|
|
|
ShuffleTiles::~ShuffleTiles()
|
|
|
|
{
|
|
|
|
CC_SAFE_DELETE_ARRAY(_tilesOrder);
|
|
|
|
CC_SAFE_DELETE_ARRAY(_tiles);
|
|
|
|
}
|
|
|
|
|
2021-12-25 10:04:45 +08:00
|
|
|
void ShuffleTiles::shuffle(unsigned int* array, unsigned int len)
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
2021-12-25 10:04:45 +08:00
|
|
|
for (int i = len - 1; i >= 0; i--)
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
2021-12-25 10:04:45 +08:00
|
|
|
unsigned int j = rand() % (i + 1);
|
2019-11-23 20:27:39 +08:00
|
|
|
unsigned int v = array[i];
|
2021-12-25 10:04:45 +08:00
|
|
|
array[i] = array[j];
|
|
|
|
array[j] = v;
|
2019-11-23 20:27:39 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-10-23 23:27:14 +08:00
|
|
|
Vec2 ShuffleTiles::getDelta(const Vec2& pos) const
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
|
|
|
unsigned int idx = static_cast<unsigned int>(pos.width * _gridSize.height + pos.height);
|
2021-12-25 10:04:45 +08:00
|
|
|
Vec2 pos2;
|
2019-11-23 20:27:39 +08:00
|
|
|
pos2.x = (float)(_tilesOrder[idx] / (int)_gridSize.height);
|
|
|
|
pos2.y = (float)(_tilesOrder[idx] % (int)_gridSize.height);
|
|
|
|
|
2021-12-28 21:27:32 +08:00
|
|
|
return Vec2(static_cast<float>((int)(pos2.x - pos.width)), static_cast<float>((int)(pos2.y - pos.height)));
|
2019-11-23 20:27:39 +08:00
|
|
|
}
|
|
|
|
|
2021-12-25 10:04:45 +08:00
|
|
|
void ShuffleTiles::placeTile(const Vec2& pos, Tile* t)
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
|
|
|
Quad3 coords = getOriginalTile(pos);
|
|
|
|
|
|
|
|
Vec2 step = _gridNodeTarget->getGrid()->getStep();
|
|
|
|
coords.bl.x += (int)(t->position.x * step.x);
|
|
|
|
coords.bl.y += (int)(t->position.y * step.y);
|
|
|
|
|
|
|
|
coords.br.x += (int)(t->position.x * step.x);
|
|
|
|
coords.br.y += (int)(t->position.y * step.y);
|
|
|
|
|
|
|
|
coords.tl.x += (int)(t->position.x * step.x);
|
|
|
|
coords.tl.y += (int)(t->position.y * step.y);
|
|
|
|
|
|
|
|
coords.tr.x += (int)(t->position.x * step.x);
|
|
|
|
coords.tr.y += (int)(t->position.y * step.y);
|
|
|
|
|
|
|
|
setTile(pos, coords);
|
|
|
|
}
|
|
|
|
|
2021-12-25 10:04:45 +08:00
|
|
|
void ShuffleTiles::startWithTarget(Node* target)
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
|
|
|
TiledGrid3DAction::startWithTarget(target);
|
|
|
|
|
|
|
|
if (_seed != (unsigned int)-1)
|
|
|
|
{
|
|
|
|
std::srand(_seed);
|
|
|
|
}
|
|
|
|
|
|
|
|
_tilesCount = (unsigned int)(_gridSize.width * _gridSize.height);
|
|
|
|
_tilesOrder = new unsigned int[_tilesCount];
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Use k to loop. Because _tilesCount is unsigned int,
|
|
|
|
* and i is used later for int.
|
|
|
|
*/
|
|
|
|
for (unsigned int k = 0; k < _tilesCount; ++k)
|
|
|
|
{
|
|
|
|
_tilesOrder[k] = k;
|
|
|
|
}
|
|
|
|
|
|
|
|
shuffle(_tilesOrder, _tilesCount);
|
|
|
|
|
2021-12-25 10:04:45 +08:00
|
|
|
_tiles = (struct Tile*)new Tile[_tilesCount];
|
|
|
|
Tile* tileArray = (Tile*)_tiles;
|
2019-11-23 20:27:39 +08:00
|
|
|
|
|
|
|
for (int i = 0; i < _gridSize.width; ++i)
|
|
|
|
{
|
2021-12-25 10:04:45 +08:00
|
|
|
for (int j = 0; j < _gridSize.height; ++j)
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
|
|
|
tileArray->position.set((float)i, (float)j);
|
|
|
|
tileArray->startPosition.set((float)i, (float)j);
|
2021-10-23 23:27:14 +08:00
|
|
|
tileArray->delta = getDelta(Vec2((float)i, (float)j));
|
2019-11-23 20:27:39 +08:00
|
|
|
++tileArray;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void ShuffleTiles::update(float time)
|
|
|
|
{
|
2021-12-25 10:04:45 +08:00
|
|
|
Tile* tileArray = (Tile*)_tiles;
|
2019-11-23 20:27:39 +08:00
|
|
|
|
|
|
|
for (int i = 0; i < _gridSize.width; ++i)
|
|
|
|
{
|
|
|
|
for (int j = 0; j < _gridSize.height; ++j)
|
|
|
|
{
|
|
|
|
tileArray->position = Vec2((float)tileArray->delta.width, (float)tileArray->delta.height) * time;
|
|
|
|
placeTile(Vec2((float)i, (float)j), tileArray);
|
|
|
|
++tileArray;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// implementation of FadeOutTRTiles
|
|
|
|
|
2021-10-23 23:27:14 +08:00
|
|
|
FadeOutTRTiles* FadeOutTRTiles::create(float duration, const Vec2& gridSize)
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
2021-12-25 10:04:45 +08:00
|
|
|
FadeOutTRTiles* action = new FadeOutTRTiles();
|
2019-11-23 20:27:39 +08:00
|
|
|
|
2021-12-08 00:11:53 +08:00
|
|
|
if (action->initWithDuration(duration, gridSize))
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
|
|
|
action->autorelease();
|
|
|
|
return action;
|
|
|
|
}
|
|
|
|
|
|
|
|
delete action;
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
FadeOutTRTiles* FadeOutTRTiles::clone() const
|
|
|
|
{
|
|
|
|
// no copy constructor
|
|
|
|
return FadeOutTRTiles::create(_duration, _gridSize);
|
|
|
|
}
|
|
|
|
|
2021-10-23 23:27:14 +08:00
|
|
|
float FadeOutTRTiles::testFunc(const Vec2& pos, float time)
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
|
|
|
Vec2 n = Vec2((float)_gridSize.width, (float)_gridSize.height) * time;
|
|
|
|
if ((n.x + n.y) == 0.0f)
|
|
|
|
{
|
|
|
|
return 1.0f;
|
|
|
|
}
|
|
|
|
|
|
|
|
return powf((pos.width + pos.height) / (n.x + n.y), 6);
|
|
|
|
}
|
|
|
|
|
|
|
|
void FadeOutTRTiles::turnOnTile(const Vec2& pos)
|
|
|
|
{
|
|
|
|
setTile(pos, getOriginalTile(pos));
|
|
|
|
}
|
|
|
|
|
|
|
|
void FadeOutTRTiles::turnOffTile(const Vec2& pos)
|
|
|
|
{
|
|
|
|
Quad3 coords;
|
|
|
|
memset(&coords, 0, sizeof(Quad3));
|
|
|
|
setTile(pos, coords);
|
|
|
|
}
|
|
|
|
|
|
|
|
void FadeOutTRTiles::transformTile(const Vec2& pos, float distance)
|
|
|
|
{
|
|
|
|
Quad3 coords = getOriginalTile(pos);
|
2021-12-25 10:04:45 +08:00
|
|
|
Vec2 step = _gridNodeTarget->getGrid()->getStep();
|
2019-11-23 20:27:39 +08:00
|
|
|
|
|
|
|
coords.bl.x += (step.x / 2) * (1.0f - distance);
|
|
|
|
coords.bl.y += (step.y / 2) * (1.0f - distance);
|
|
|
|
|
|
|
|
coords.br.x -= (step.x / 2) * (1.0f - distance);
|
|
|
|
coords.br.y += (step.y / 2) * (1.0f - distance);
|
|
|
|
|
|
|
|
coords.tl.x += (step.x / 2) * (1.0f - distance);
|
|
|
|
coords.tl.y -= (step.y / 2) * (1.0f - distance);
|
|
|
|
|
|
|
|
coords.tr.x -= (step.x / 2) * (1.0f - distance);
|
|
|
|
coords.tr.y -= (step.y / 2) * (1.0f - distance);
|
|
|
|
|
|
|
|
setTile(pos, coords);
|
|
|
|
}
|
|
|
|
|
|
|
|
void FadeOutTRTiles::update(float time)
|
|
|
|
{
|
|
|
|
for (int i = 0; i < _gridSize.width; ++i)
|
|
|
|
{
|
|
|
|
for (int j = 0; j < _gridSize.height; ++j)
|
|
|
|
{
|
|
|
|
Vec2 pos((float)i, (float)j);
|
2021-10-23 23:27:14 +08:00
|
|
|
float distance = testFunc(Vec2((float)i, (float)j), time);
|
2021-12-25 10:04:45 +08:00
|
|
|
if (distance == 0)
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
|
|
|
turnOffTile(pos);
|
2021-12-25 10:04:45 +08:00
|
|
|
}
|
|
|
|
else if (distance < 1)
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
|
|
|
transformTile(pos, distance);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
turnOnTile(pos);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// implementation of FadeOutBLTiles
|
|
|
|
|
2021-10-23 23:27:14 +08:00
|
|
|
FadeOutBLTiles* FadeOutBLTiles::create(float duration, const Vec2& gridSize)
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
2021-12-25 10:04:45 +08:00
|
|
|
FadeOutBLTiles* action = new FadeOutBLTiles();
|
2019-11-23 20:27:39 +08:00
|
|
|
|
2021-12-08 00:11:53 +08:00
|
|
|
if (action->initWithDuration(duration, gridSize))
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
|
|
|
action->autorelease();
|
|
|
|
return action;
|
|
|
|
}
|
|
|
|
|
|
|
|
delete action;
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
FadeOutBLTiles* FadeOutBLTiles::clone() const
|
|
|
|
{
|
|
|
|
// no copy constructor
|
|
|
|
return FadeOutBLTiles::create(_duration, _gridSize);
|
|
|
|
}
|
|
|
|
|
2021-10-23 23:27:14 +08:00
|
|
|
float FadeOutBLTiles::testFunc(const Vec2& pos, float time)
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
|
|
|
Vec2 n = Vec2((float)_gridSize.width, (float)_gridSize.height) * (1.0f - time);
|
|
|
|
if ((pos.width + pos.height) == 0)
|
|
|
|
{
|
|
|
|
return 1.0f;
|
|
|
|
}
|
|
|
|
|
|
|
|
return powf((n.x + n.y) / (pos.width + pos.height), 6);
|
|
|
|
}
|
|
|
|
|
|
|
|
// implementation of FadeOutUpTiles
|
|
|
|
|
2021-10-23 23:27:14 +08:00
|
|
|
FadeOutUpTiles* FadeOutUpTiles::create(float duration, const Vec2& gridSize)
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
2021-12-25 10:04:45 +08:00
|
|
|
FadeOutUpTiles* action = new FadeOutUpTiles();
|
2019-11-23 20:27:39 +08:00
|
|
|
|
2021-12-08 00:11:53 +08:00
|
|
|
if (action->initWithDuration(duration, gridSize))
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
|
|
|
action->autorelease();
|
|
|
|
return action;
|
|
|
|
}
|
|
|
|
|
|
|
|
delete action;
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
FadeOutUpTiles* FadeOutUpTiles::clone() const
|
|
|
|
{
|
|
|
|
// no copy constructor
|
|
|
|
return FadeOutUpTiles::create(_duration, _gridSize);
|
|
|
|
}
|
|
|
|
|
2021-10-23 23:27:14 +08:00
|
|
|
float FadeOutUpTiles::testFunc(const Vec2& pos, float time)
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
|
|
|
Vec2 n = Vec2((float)_gridSize.width, (float)_gridSize.height) * time;
|
|
|
|
if (n.y == 0.0f)
|
|
|
|
{
|
|
|
|
return 1.0f;
|
|
|
|
}
|
|
|
|
|
|
|
|
return powf(pos.height / n.y, 6);
|
|
|
|
}
|
|
|
|
|
|
|
|
void FadeOutUpTiles::transformTile(const Vec2& pos, float distance)
|
|
|
|
{
|
|
|
|
Quad3 coords = getOriginalTile(pos);
|
2021-12-25 10:04:45 +08:00
|
|
|
Vec2 step = _gridNodeTarget->getGrid()->getStep();
|
2019-11-23 20:27:39 +08:00
|
|
|
|
|
|
|
coords.bl.y += (step.y / 2) * (1.0f - distance);
|
|
|
|
coords.br.y += (step.y / 2) * (1.0f - distance);
|
|
|
|
coords.tl.y -= (step.y / 2) * (1.0f - distance);
|
|
|
|
coords.tr.y -= (step.y / 2) * (1.0f - distance);
|
|
|
|
|
|
|
|
setTile(pos, coords);
|
|
|
|
}
|
|
|
|
|
|
|
|
// implementation of FadeOutDownTiles
|
|
|
|
|
2021-10-23 23:27:14 +08:00
|
|
|
FadeOutDownTiles* FadeOutDownTiles::create(float duration, const Vec2& gridSize)
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
2021-12-25 10:04:45 +08:00
|
|
|
FadeOutDownTiles* action = new FadeOutDownTiles();
|
2019-11-23 20:27:39 +08:00
|
|
|
|
2021-12-08 00:11:53 +08:00
|
|
|
if (action->initWithDuration(duration, gridSize))
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
|
|
|
action->autorelease();
|
|
|
|
return action;
|
|
|
|
}
|
|
|
|
|
|
|
|
delete action;
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
FadeOutDownTiles* FadeOutDownTiles::clone() const
|
|
|
|
{
|
|
|
|
// no copy constructor
|
|
|
|
return FadeOutDownTiles::create(_duration, _gridSize);
|
|
|
|
}
|
|
|
|
|
2021-10-23 23:27:14 +08:00
|
|
|
float FadeOutDownTiles::testFunc(const Vec2& pos, float time)
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
|
|
|
Vec2 n = Vec2((float)_gridSize.width, (float)_gridSize.height) * (1.0f - time);
|
|
|
|
return powf(n.y / (pos.height > 0.0f ? pos.height : 0.1f), 6);
|
|
|
|
}
|
|
|
|
|
|
|
|
// implementation of TurnOffTiles
|
|
|
|
|
2021-10-23 23:27:14 +08:00
|
|
|
TurnOffTiles* TurnOffTiles::create(float duration, const Vec2& gridSize)
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
2021-12-08 00:11:53 +08:00
|
|
|
TurnOffTiles* action = new TurnOffTiles();
|
|
|
|
if (action->initWithDuration(duration, gridSize, 0))
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
|
|
|
action->autorelease();
|
|
|
|
return action;
|
|
|
|
}
|
2021-12-25 10:04:45 +08:00
|
|
|
|
2019-11-23 20:27:39 +08:00
|
|
|
delete action;
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
2021-10-23 23:27:14 +08:00
|
|
|
TurnOffTiles* TurnOffTiles::create(float duration, const Vec2& gridSize, unsigned int seed)
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
2021-12-25 10:04:45 +08:00
|
|
|
TurnOffTiles* action = new TurnOffTiles();
|
2019-11-23 20:27:39 +08:00
|
|
|
|
2021-12-08 00:11:53 +08:00
|
|
|
if (action->initWithDuration(duration, gridSize, seed))
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
|
|
|
action->autorelease();
|
|
|
|
return action;
|
|
|
|
}
|
|
|
|
|
|
|
|
delete action;
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
2021-10-23 23:27:14 +08:00
|
|
|
bool TurnOffTiles::initWithDuration(float duration, const Vec2& gridSize, unsigned int seed)
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
|
|
|
if (TiledGrid3DAction::initWithDuration(duration, gridSize))
|
|
|
|
{
|
2021-12-25 10:04:45 +08:00
|
|
|
_seed = seed;
|
2019-11-23 20:27:39 +08:00
|
|
|
_tilesOrder = nullptr;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
TurnOffTiles* TurnOffTiles::clone() const
|
|
|
|
{
|
|
|
|
// no copy constructor
|
|
|
|
return TurnOffTiles::create(_duration, _gridSize, _seed);
|
|
|
|
}
|
|
|
|
|
|
|
|
TurnOffTiles::~TurnOffTiles()
|
|
|
|
{
|
|
|
|
CC_SAFE_DELETE_ARRAY(_tilesOrder);
|
|
|
|
}
|
|
|
|
|
2021-12-25 10:04:45 +08:00
|
|
|
void TurnOffTiles::shuffle(unsigned int* array, unsigned int len)
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
|
|
|
for (int i = len - 1; i >= 0; i--)
|
|
|
|
{
|
2021-12-25 10:04:45 +08:00
|
|
|
unsigned int j = rand() % (i + 1);
|
2019-11-23 20:27:39 +08:00
|
|
|
unsigned int v = array[i];
|
2021-12-25 10:04:45 +08:00
|
|
|
array[i] = array[j];
|
|
|
|
array[j] = v;
|
2019-11-23 20:27:39 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void TurnOffTiles::turnOnTile(const Vec2& pos)
|
|
|
|
{
|
|
|
|
setTile(pos, getOriginalTile(pos));
|
|
|
|
}
|
|
|
|
|
|
|
|
void TurnOffTiles::turnOffTile(const Vec2& pos)
|
|
|
|
{
|
|
|
|
Quad3 coords;
|
|
|
|
|
|
|
|
memset(&coords, 0, sizeof(Quad3));
|
|
|
|
setTile(pos, coords);
|
|
|
|
}
|
|
|
|
|
2021-12-25 10:04:45 +08:00
|
|
|
void TurnOffTiles::startWithTarget(Node* target)
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
|
|
|
TiledGrid3DAction::startWithTarget(target);
|
|
|
|
|
|
|
|
if (_seed != (unsigned int)-1)
|
|
|
|
{
|
|
|
|
std::srand(_seed);
|
|
|
|
}
|
|
|
|
|
|
|
|
_tilesCount = (unsigned int)(_gridSize.width * _gridSize.height);
|
|
|
|
_tilesOrder = new unsigned int[_tilesCount];
|
|
|
|
|
|
|
|
for (unsigned int i = 0; i < _tilesCount; ++i)
|
|
|
|
{
|
|
|
|
_tilesOrder[i] = i;
|
|
|
|
}
|
|
|
|
|
|
|
|
shuffle(_tilesOrder, _tilesCount);
|
|
|
|
}
|
|
|
|
|
|
|
|
void TurnOffTiles::update(float time)
|
|
|
|
{
|
|
|
|
unsigned int l = (unsigned int)(time * (float)_tilesCount);
|
|
|
|
|
|
|
|
unsigned int t = 0;
|
2021-12-25 10:04:45 +08:00
|
|
|
for (unsigned int i = 0; i < _tilesCount; i++)
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
|
|
|
t = _tilesOrder[i];
|
2021-12-25 10:04:45 +08:00
|
|
|
// needs integer value
|
2019-11-23 20:27:39 +08:00
|
|
|
unsigned int x = (unsigned int)(t / _gridSize.height);
|
|
|
|
Vec2 tilePos((float)x, (float)(t % (unsigned int)_gridSize.height));
|
|
|
|
|
2021-12-25 10:04:45 +08:00
|
|
|
if (i < l)
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
|
|
|
turnOffTile(tilePos);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
turnOnTile(tilePos);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// implementation of WavesTiles3D
|
|
|
|
|
2021-10-23 23:27:14 +08:00
|
|
|
WavesTiles3D* WavesTiles3D::create(float duration, const Vec2& gridSize, unsigned int waves, float amplitude)
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
2021-12-25 10:04:45 +08:00
|
|
|
WavesTiles3D* action = new WavesTiles3D();
|
2019-11-23 20:27:39 +08:00
|
|
|
|
2021-12-08 00:11:53 +08:00
|
|
|
if (action->initWithDuration(duration, gridSize, waves, amplitude))
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
|
|
|
action->autorelease();
|
|
|
|
return action;
|
|
|
|
}
|
|
|
|
|
|
|
|
delete action;
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
2021-10-23 23:27:14 +08:00
|
|
|
bool WavesTiles3D::initWithDuration(float duration, const Vec2& gridSize, unsigned int waves, float amplitude)
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
|
|
|
if (TiledGrid3DAction::initWithDuration(duration, gridSize))
|
|
|
|
{
|
2021-12-25 10:04:45 +08:00
|
|
|
_waves = waves;
|
|
|
|
_amplitude = amplitude;
|
2019-11-23 20:27:39 +08:00
|
|
|
_amplitudeRate = 1.0f;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
WavesTiles3D* WavesTiles3D::clone() const
|
|
|
|
{
|
|
|
|
// no copy constructor
|
|
|
|
return WavesTiles3D::create(_duration, _gridSize, _waves, _amplitude);
|
|
|
|
}
|
|
|
|
|
|
|
|
void WavesTiles3D::update(float time)
|
|
|
|
{
|
2021-12-25 10:04:45 +08:00
|
|
|
for (int i = 0; i < _gridSize.width; i++)
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
2021-12-25 10:04:45 +08:00
|
|
|
for (int j = 0; j < _gridSize.height; j++)
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
|
|
|
Vec2 pos((float)i, (float)j);
|
|
|
|
Quad3 coords = getOriginalTile(pos);
|
|
|
|
|
2021-12-25 10:04:45 +08:00
|
|
|
coords.bl.z = (sinf(time * (float)M_PI * _waves * 2 + (coords.bl.y + coords.bl.x) * .01f) * _amplitude *
|
|
|
|
_amplitudeRate);
|
|
|
|
coords.br.z = coords.bl.z;
|
2019-11-23 20:27:39 +08:00
|
|
|
coords.tl.z = coords.bl.z;
|
|
|
|
coords.tr.z = coords.bl.z;
|
|
|
|
|
|
|
|
setTile(pos, coords);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// implementation of JumpTiles3D
|
|
|
|
|
2021-10-23 23:27:14 +08:00
|
|
|
JumpTiles3D* JumpTiles3D::create(float duration, const Vec2& gridSize, unsigned int numberOfJumps, float amplitude)
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
2021-12-25 10:04:45 +08:00
|
|
|
JumpTiles3D* action = new JumpTiles3D();
|
2019-11-23 20:27:39 +08:00
|
|
|
|
2021-12-08 00:11:53 +08:00
|
|
|
if (action->initWithDuration(duration, gridSize, numberOfJumps, amplitude))
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
|
|
|
action->autorelease();
|
|
|
|
return action;
|
|
|
|
}
|
|
|
|
|
|
|
|
delete action;
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
2021-10-23 23:27:14 +08:00
|
|
|
bool JumpTiles3D::initWithDuration(float duration, const Vec2& gridSize, unsigned int numberOfJumps, float amplitude)
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
|
|
|
if (TiledGrid3DAction::initWithDuration(duration, gridSize))
|
|
|
|
{
|
2021-12-25 10:04:45 +08:00
|
|
|
_jumps = numberOfJumps;
|
|
|
|
_amplitude = amplitude;
|
2019-11-23 20:27:39 +08:00
|
|
|
_amplitudeRate = 1.0f;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
JumpTiles3D* JumpTiles3D::clone() const
|
|
|
|
{
|
|
|
|
// no copy constructor
|
|
|
|
return JumpTiles3D::create(_duration, _gridSize, _jumps, _amplitude);
|
|
|
|
}
|
|
|
|
|
|
|
|
void JumpTiles3D::update(float time)
|
|
|
|
{
|
2021-12-25 10:04:45 +08:00
|
|
|
float sinz = (sinf((float)M_PI * time * _jumps * 2) * _amplitude * _amplitudeRate);
|
|
|
|
float sinz2 = (sinf((float)M_PI * (time * _jumps * 2 + 1)) * _amplitude * _amplitudeRate);
|
2019-11-23 20:27:39 +08:00
|
|
|
|
2021-12-25 10:04:45 +08:00
|
|
|
for (int i = 0; i < _gridSize.width; i++)
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
2021-12-25 10:04:45 +08:00
|
|
|
for (int j = 0; j < _gridSize.height; j++)
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
|
|
|
Vec2 pos((float)i, (float)j);
|
|
|
|
Quad3 coords = getOriginalTile(pos);
|
|
|
|
|
2021-12-25 10:04:45 +08:00
|
|
|
if (((i + j) % 2) == 0)
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
|
|
|
coords.bl.z += sinz;
|
|
|
|
coords.br.z += sinz;
|
|
|
|
coords.tl.z += sinz;
|
|
|
|
coords.tr.z += sinz;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
coords.bl.z += sinz2;
|
|
|
|
coords.br.z += sinz2;
|
|
|
|
coords.tl.z += sinz2;
|
|
|
|
coords.tr.z += sinz2;
|
|
|
|
}
|
|
|
|
|
|
|
|
setTile(pos, coords);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// implementation of SplitRows
|
|
|
|
|
|
|
|
SplitRows* SplitRows::create(float duration, unsigned int nRows)
|
|
|
|
{
|
2021-12-25 10:04:45 +08:00
|
|
|
SplitRows* action = new SplitRows();
|
2019-11-23 20:27:39 +08:00
|
|
|
|
2021-12-08 00:11:53 +08:00
|
|
|
if (action->initWithDuration(duration, nRows))
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
|
|
|
action->autorelease();
|
|
|
|
return action;
|
|
|
|
}
|
|
|
|
|
|
|
|
delete action;
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool SplitRows::initWithDuration(float duration, unsigned int rows)
|
|
|
|
{
|
|
|
|
_rows = rows;
|
|
|
|
|
2021-10-23 23:27:14 +08:00
|
|
|
return TiledGrid3DAction::initWithDuration(duration, Vec2(1.0f, (float)rows));
|
2019-11-23 20:27:39 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
SplitRows* SplitRows::clone() const
|
|
|
|
{
|
|
|
|
// no copy constructor
|
|
|
|
return SplitRows::create(_duration, _rows);
|
|
|
|
}
|
|
|
|
|
2021-12-25 10:04:45 +08:00
|
|
|
void SplitRows::startWithTarget(Node* target)
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
|
|
|
TiledGrid3DAction::startWithTarget(target);
|
|
|
|
_winSize = Director::getInstance()->getWinSizeInPixels();
|
|
|
|
}
|
|
|
|
|
|
|
|
void SplitRows::update(float time)
|
|
|
|
{
|
|
|
|
for (int j = 0; j < _gridSize.height; ++j)
|
|
|
|
{
|
|
|
|
Vec2 pos(0, (float)j);
|
2021-12-25 10:04:45 +08:00
|
|
|
Quad3 coords = getOriginalTile(pos);
|
|
|
|
float direction = 1;
|
2019-11-23 20:27:39 +08:00
|
|
|
|
2021-12-25 10:04:45 +08:00
|
|
|
if ((j % 2) == 0)
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
|
|
|
direction = -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
coords.bl.x += direction * _winSize.width * time;
|
|
|
|
coords.br.x += direction * _winSize.width * time;
|
|
|
|
coords.tl.x += direction * _winSize.width * time;
|
|
|
|
coords.tr.x += direction * _winSize.width * time;
|
|
|
|
|
|
|
|
setTile(pos, coords);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// implementation of SplitCols
|
|
|
|
|
|
|
|
SplitCols* SplitCols::create(float duration, unsigned int cols)
|
|
|
|
{
|
2021-12-25 10:04:45 +08:00
|
|
|
SplitCols* action = new SplitCols();
|
2019-11-23 20:27:39 +08:00
|
|
|
|
2021-12-08 00:11:53 +08:00
|
|
|
if (action->initWithDuration(duration, cols))
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
|
|
|
action->autorelease();
|
|
|
|
return action;
|
|
|
|
}
|
|
|
|
|
|
|
|
delete action;
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool SplitCols::initWithDuration(float duration, unsigned int cols)
|
|
|
|
{
|
|
|
|
_cols = cols;
|
2021-10-23 23:27:14 +08:00
|
|
|
return TiledGrid3DAction::initWithDuration(duration, Vec2((float)cols, 1.0f));
|
2019-11-23 20:27:39 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
SplitCols* SplitCols::clone() const
|
|
|
|
{
|
|
|
|
// no copy constructor
|
|
|
|
return SplitCols::create(_duration, _cols);
|
|
|
|
}
|
|
|
|
|
2021-12-25 10:04:45 +08:00
|
|
|
void SplitCols::startWithTarget(Node* target)
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
|
|
|
TiledGrid3DAction::startWithTarget(target);
|
|
|
|
_winSize = Director::getInstance()->getWinSizeInPixels();
|
|
|
|
}
|
|
|
|
|
|
|
|
void SplitCols::update(float time)
|
|
|
|
{
|
|
|
|
for (unsigned int i = 0; i < _gridSize.width; ++i)
|
|
|
|
{
|
|
|
|
Vec2 pos((float)i, 0);
|
2021-12-25 10:04:45 +08:00
|
|
|
Quad3 coords = getOriginalTile(pos);
|
|
|
|
float direction = 1;
|
2019-11-23 20:27:39 +08:00
|
|
|
|
2021-12-25 10:04:45 +08:00
|
|
|
if ((i % 2) == 0)
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
|
|
|
direction = -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
coords.bl.y += direction * _winSize.height * time;
|
|
|
|
coords.br.y += direction * _winSize.height * time;
|
|
|
|
coords.tl.y += direction * _winSize.height * time;
|
|
|
|
coords.tr.y += direction * _winSize.height * time;
|
|
|
|
|
|
|
|
setTile(pos, coords);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_CC_END
|