axmol/tests/cpp-tests/Classes/TerrainTest/TerrainTest.cpp

480 lines
14 KiB
C++
Raw Normal View History

2015-01-23 18:09:54 +08:00
#include "TerrainTest.h"
Vec3 camera_offset(0,45,60);
#define PLAYER_HEIGHT 0
2015-01-23 18:09:54 +08:00
static std::function<Layer*()> createFunctions[] =
{
CL(TerrainSimple),
CL(TerrainWalkThru),
};
static int sceneIdx = -1;
#define MAX_LAYER (sizeof(createFunctions) / sizeof(createFunctions[0]))
static Layer* nextTerrainTestAction()
{
sceneIdx++;
sceneIdx = sceneIdx % MAX_LAYER;
auto layer = (createFunctions[sceneIdx])();
return layer;
}
static Layer* backTerrainTestAction()
{
sceneIdx--;
int total = MAX_LAYER;
if( sceneIdx < 0 )
sceneIdx += total;
auto layer = (createFunctions[sceneIdx])();
return layer;
}
static Layer* restartTerrainTestAction()
{
auto layer = (createFunctions[sceneIdx])();
return layer;
}
TerrainSimple::TerrainSimple()
{
Size visibleSize = Director::getInstance()->getVisibleSize();
//use custom camera
2015-01-30 15:46:39 +08:00
_camera = Camera::createPerspective(60,visibleSize.width/visibleSize.height,0.1,800);
_camera->setCameraFlag(CameraFlag::USER1);
addChild(_camera);
Terrain::DetailMap r("TerrainTest/dirt.dds"),g("TerrainTest/Grass2.dds"),b("TerrainTest/road.dds"),a("TerrainTest/Grass1.dds");
Terrain::TerrainData data("TerrainTest/heightmap16.jpg","TerrainTest/alphamap.png",r,g,b,a);
_terrain = Terrain::create(data);
_terrain->setMaxDetailMapAmount(4);
addChild(_terrain);
_terrain->setCameraMask(2);
_terrain->setDrawWire(true);
auto listener = EventListenerTouchAllAtOnce::create();
listener->onTouchesMoved = CC_CALLBACK_2(TerrainSimple::onTouchesMoved, this);
_eventDispatcher->addEventListenerWithSceneGraphPriority(listener, this);
TTFConfig ttfConfig("fonts/arial.ttf", 8);
auto label1 = Label::createWithTTF(ttfConfig,"129*129 1 DETAIL MAP 32*32 chunk Size");
auto menuItem1 = MenuItemLabel::create(label1, CC_CALLBACK_1(TerrainSimple::on127x127_1DetailMap32x32Callback,this));
auto label2 = Label::createWithTTF(ttfConfig,"257*257 1 DETAIL MAP 32*32 chunk Size");
auto menuItem2 = MenuItemLabel::create(label2, CC_CALLBACK_1(TerrainSimple::on257x257_1DetailMap32x32Callback,this));
auto label3 = Label::createWithTTF(ttfConfig,"Enable LOD");
auto menuItem3 = MenuItemLabel::create(label3, CC_CALLBACK_1(TerrainSimple::onEnableLODCallback,this));
auto label4 = Label::createWithTTF(ttfConfig,"disable LOD");
auto menuItem4 = MenuItemLabel::create(label4, CC_CALLBACK_1(TerrainSimple::onDisableLODCallback,this));
auto menu = Menu::create(menuItem1, menuItem2,menuItem3,menuItem4, nullptr);
menu->setPosition(Vec2::ZERO);
menuItem1->setPosition( Vec2(0, VisibleRect::top().y-160) );
menuItem1->setAnchorPoint(Vec2(0,0));
menuItem2->setPosition( Vec2(0, VisibleRect::top().y-190) );
menuItem2->setAnchorPoint(Vec2(0,0));
menuItem3->setPosition( Vec2(0, VisibleRect::top().y-210) );
menuItem3->setAnchorPoint(Vec2(0,0));
menuItem4->setPosition( Vec2(0, VisibleRect::top().y-240) );
menuItem4->setAnchorPoint(Vec2(0,0));
addChild(menu, 0);
2015-01-23 18:09:54 +08:00
}
std::string TerrainSimple::title() const
{
return "Simple Terrain";
}
std::string TerrainSimple::subtitle() const
{
return "Drag to walkThru";
}
2015-01-30 15:46:39 +08:00
void TerrainSimple::onTouchesMoved(const std::vector<cocos2d::Touch*>& touches, cocos2d::Event* event)
{
float delta = Director::getInstance()->getDeltaTime();
auto touch = touches[0];
auto location = touch->getLocation();
auto PreviousLocation = touch->getPreviousLocation();
Point newPos = PreviousLocation - location;
Vec3 cameraDir;
Vec3 cameraRightDir;
_camera->getNodeToWorldTransform().getForwardVector(&cameraDir);
cameraDir.normalize();
cameraDir.y=0;
_camera->getNodeToWorldTransform().getRightVector(&cameraRightDir);
cameraRightDir.normalize();
cameraRightDir.y=0;
Vec3 cameraPos= _camera->getPosition3D();
cameraPos+=cameraDir*newPos.y*0.5*delta;
cameraPos+=cameraRightDir*newPos.x*0.5*delta;
_camera->setPosition3D(cameraPos);
}
void TerrainSimple::on127x127_1DetailMap32x32Callback(Ref* sender)
{
_terrain->removeFromParent();
Terrain::TerrainData data("TerrainTest/heightMap129.jpg","TerrainTest/sand.jpg");
_terrain = Terrain::create(data);
_terrain->setMaxDetailMapAmount(4);
addChild(_terrain);
_terrain->setCameraMask(2);
_terrain->setDrawWire(true);
}
void TerrainSimple::on257x257_1DetailMap32x32Callback(Ref* sender)
{
_terrain->removeFromParent();
Terrain::TerrainData data("TerrainTest/heightmap16.jpg","TerrainTest/sand.jpg");
_terrain = Terrain::create(data);
_terrain->setMaxDetailMapAmount(4);
addChild(_terrain);
_terrain->setCameraMask(2);
_terrain->setDrawWire(true);
}
void TerrainSimple::onEnableLODCallback(Ref* sender)
{
_terrain->setScale(20);
}
void TerrainSimple::onDisableLODCallback(Ref* sender)
{
_terrain->setScale(1);
}
2015-01-23 18:09:54 +08:00
void TerrainTestScene::runThisTest()
{
auto layer = nextTerrainTestAction();
addChild(layer);
Director::getInstance()->replaceScene(this);
}
void TerrainTestDemo::restartCallback(Ref* sender)
{
auto s = new (std::nothrow) TerrainTestScene();
s->addChild(restartTerrainTestAction());
Director::getInstance()->replaceScene(s);
s->release();
}
void TerrainTestDemo::nextCallback(Ref* sender)
{
auto s = new (std::nothrow) TerrainTestScene();
s->addChild( nextTerrainTestAction());
Director::getInstance()->replaceScene(s);
s->release();
}
void TerrainTestDemo::backCallback(Ref* sender)
{
auto s = new (std::nothrow) TerrainTestScene();
s->addChild( backTerrainTestAction());
Director::getInstance()->replaceScene(s);
s->release();
}
std::string TerrainTestDemo::title() const
{
return "No title";
}
std::string TerrainTestDemo::subtitle() const
{
return "";
}
void TerrainTestDemo::onEnter()
{
BaseTest::onEnter();
}
TerrainTestDemo::TerrainTestDemo(void)
:BaseTest()
{
}
TerrainTestDemo::~TerrainTestDemo(void)
{
}
std::string TerrainWalkThru::title() const
{
return " ";
}
std::string TerrainWalkThru::subtitle() const
{
return " ";
}
TerrainWalkThru::TerrainWalkThru()
{
auto listener = EventListenerTouchAllAtOnce::create();
listener->onTouchesBegan = CC_CALLBACK_2(TerrainWalkThru::onTouchesBegan, this);
listener->onTouchesMoved = CC_CALLBACK_2(TerrainWalkThru::onTouchesMoved, this);
listener->onTouchesEnded = CC_CALLBACK_2(TerrainWalkThru::onTouchesEnd, this);
_eventDispatcher->addEventListenerWithSceneGraphPriority(listener, this);
Size visibleSize = Director::getInstance()->getVisibleSize();
//use custom camera
_camera = Camera::createPerspective(60,visibleSize.width/visibleSize.height,0.1,200);
_camera->setCameraFlag(CameraFlag::USER1);
addChild(_camera);
Terrain::TerrainData a("TerrainTest/heightmap16.jpg","TerrainTest/sand.jpg",Size(32,32),3);
2015-01-23 18:09:54 +08:00
_terrain = Terrain::create(a);
_terrain->setScale(20);
_terrain->setCameraMask(2);
_terrain->setDrawWire(false);
_action = new PlayerAction(_camera,_terrain);
_action->retain();
_player =Sprite3D::create("Sprite3DTest/girl.c3b");
2015-01-23 18:09:54 +08:00
_player->runAction(_action);
_player->setCameraMask(2);
_player->setScale(0.08);
2015-01-30 15:46:39 +08:00
_player->setPositionY(_terrain->getHeight(_player->getPositionX(),_player->getPositionZ())+PLAYER_HEIGHT);
auto animation = Animation3D::create("Sprite3DTest/girl.c3b","Take 001");
if (animation)
{
auto animate = Animate3D::create(animation);
_player->runAction(RepeatForever::create(animate));
}
2015-01-23 18:09:54 +08:00
_camera->setPosition3D(_player->getPosition3D()+camera_offset);
_camera->setRotation3D(Vec3(-45,0,0));
2015-01-23 18:09:54 +08:00
forward = Label::create("forward","arial",22);
forward->setPosition(0,200);
forward->setAnchorPoint(Vec2(0,0));
backward = Label::create("backward","arial",22);
backward->setPosition(0,250);
backward->setAnchorPoint(Vec2(0,0));
left = Label::create("turn Left","arial",22);
left->setPosition(0,100);
left->setAnchorPoint(Vec2(0,0));
right = Label::create("turn right","arial",22);
right->setPosition(0,150);
right->setAnchorPoint(Vec2(0,0));
addChild(_player);
addChild(_terrain);
}
void TerrainWalkThru::onTouchesMoved(const std::vector<cocos2d::Touch*>& touches, cocos2d::Event* event)
{
float delta = Director::getInstance()->getDeltaTime();
auto touch = touches[0];
auto location = touch->getLocation();
auto PreviousLocation = touch->getPreviousLocation();
Point newPos = PreviousLocation - location;
Vec3 cameraDir;
Vec3 cameraRightDir;
_camera->getNodeToWorldTransform().getForwardVector(&cameraDir);
cameraDir.normalize();
cameraDir.y=0;
_camera->getNodeToWorldTransform().getRightVector(&cameraRightDir);
cameraRightDir.normalize();
cameraRightDir.y=0;
Vec3 cameraPos= _camera->getPosition3D();
cameraPos+=cameraDir*newPos.y*0.5*delta;
cameraPos+=cameraRightDir*newPos.x*0.5*delta;
_camera->setPosition3D(cameraPos);
}
void TerrainWalkThru::onTouchesBegan(const std::vector<cocos2d::Touch*>& touches, cocos2d::Event* event)
{
}
void TerrainWalkThru::onTouchesEnd(const std::vector<cocos2d::Touch*>& touches, cocos2d::Event* event)
{
auto touch = touches[0];
auto location = touch->getLocationInView();
if(_camera)
{
if(_player)
{
Vec3 nearP(location.x, location.y, 0.0f), farP(location.x, location.y, 1.0f);
auto size = Director::getInstance()->getWinSize();
_camera->unproject(size, &nearP, &nearP);
_camera->unproject(size, &farP, &farP);
Vec3 dir = farP - nearP;
dir.normalize();
Vec3 rayStep = 15*dir;
Vec3 rayPos = nearP;
Vec3 rayStartPosition = nearP;
Vec3 lastRayPosition =rayPos;
rayPos += rayStep;
// Linear search - Loop until find a point inside and outside the terrain Vector3
2015-01-30 15:46:39 +08:00
float height = _terrain->getHeight(rayPos.x,rayPos.z);
2015-01-23 18:09:54 +08:00
while (rayPos.y > height)
{
lastRayPosition = rayPos;
rayPos += rayStep;
2015-01-30 15:46:39 +08:00
height = _terrain->getHeight(rayPos.x,rayPos.z);
2015-01-23 18:09:54 +08:00
}
Vec3 startPosition = lastRayPosition;
Vec3 endPosition = rayPos;
for (int i= 0; i< 32; i++)
{
// Binary search pass
Vec3 middlePoint = (startPosition + endPosition) * 0.5f;
if (middlePoint.y < height)
endPosition = middlePoint;
else
startPosition = middlePoint;
}
Vec3 collisionPoint = (startPosition + endPosition) * 0.5f;
dir = collisionPoint - _player->getPosition3D();
dir.y = 0;
dir.normalize();
_action->_headingAngle = -1*acos(dir.dot(Vec3(0,0,-1)));
dir.cross(dir,Vec3(0,0,-1),&_action->_headingAxis);
2015-01-23 18:09:54 +08:00
_action->_targetPos=collisionPoint;
_action->forward();
}
}
}
bool PlayerAction::isDone() const
{
return false;
}
void PlayerAction::step(float dt)
{
auto player = (Sprite3D * )_target;
switch (_playerState)
{
case PLAYER_STATE_IDLE:
break;
case PLAYER_STATE_FORWARD:
{
Vec3 curPos= player->getPosition3D();
Vec3 newFaceDir = _targetPos - curPos;
newFaceDir.y = 0.0f;
newFaceDir.normalize();
Vec3 offset = newFaceDir * 25.0f * dt;
curPos+=offset;
player->setPosition3D(curPos);
}
break;
case PLAYER_STATE_BACKWARD:
{
Vec3 forward_vec;
player->getNodeToWorldTransform().getForwardVector(&forward_vec);
forward_vec.normalize();
player->setPosition3D(player->getPosition3D()-forward_vec*15*dt);
}
break;
case PLAYER_STATE_LEFT:
{
player->setRotation3D(player->getRotation3D()+Vec3(0,25*dt,0));
}
break;
case PLAYER_STATE_RIGHT:
{
player->setRotation3D(player->getRotation3D()+Vec3(0,-25*dt,0));
}
break;
default:
break;
}
Vec3 Normal;
2015-01-30 15:46:39 +08:00
float player_h = _terrain->getHeight(player->getPositionX(),player->getPositionZ(),&Normal);
float dot_product = Normal.dot(Vec3(0,1,0));
player->setPositionY(player_h+PLAYER_HEIGHT);
Quaternion q2;
q2.createFromAxisAngle(Vec3(0,1,0),-M_PI,&q2);
2015-01-30 15:46:39 +08:00
Quaternion headingQ;
headingQ.createFromAxisAngle(_headingAxis,_headingAngle,&headingQ);
player->setRotationQuat(headingQ*q2);
2015-01-23 18:09:54 +08:00
auto vec_offset =Vec4(camera_offset.x,camera_offset.y,camera_offset.z,1);
vec_offset = player->getNodeToWorldTransform()*vec_offset;
// _cam->setRotation3D(player->getRotation3D());
_cam->setPosition3D(player->getPosition3D() + camera_offset);
2015-01-23 18:09:54 +08:00
updateState();
}
void PlayerAction::turnLeft()
{
_playerState = PLAYER_STATE_LEFT;
}
void PlayerAction::turnRight()
{
_playerState = PLAYER_STATE_RIGHT;
}
void PlayerAction::idle()
{
_playerState = PLAYER_STATE_IDLE;
}
PlayerAction::PlayerAction(Camera * cam,Terrain * terrain)
{
_headingAngle = 0;
2015-01-23 18:09:54 +08:00
_playerState = PLAYER_STATE_IDLE;
_cam = cam;
_terrain = terrain;
}
void PlayerAction::forward()
{
_playerState = PLAYER_STATE_FORWARD;
}
void PlayerAction::backward()
{
_playerState = PLAYER_STATE_BACKWARD;
}
void PlayerAction::updateState()
{
auto player = (Sprite3D * )_target;
switch (_playerState)
{
case PLAYER_STATE_FORWARD:
{
Vec2 player_pos =Vec2(player->getPositionX(),player->getPositionZ());
Vec2 targetPos = Vec2(_targetPos.x,_targetPos.z);
auto dist = player_pos.distance(targetPos);
if(dist<1)
{
_playerState = PLAYER_STATE_IDLE;
}
}
break;
default:
break;
}
}