2014-08-07 15:23:31 +08:00
|
|
|
/****************************************************************************
|
|
|
|
Copyright (c) 2012 cocos2d-x.org
|
|
|
|
Copyright (c) 2013-2014 Chukong Technologies Inc.
|
|
|
|
|
|
|
|
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 "Camera3DTest.h"
|
|
|
|
#include <algorithm>
|
|
|
|
#include "../testResource.h"
|
|
|
|
|
2014-08-08 15:01:24 +08:00
|
|
|
////////////DrawLine/////////////////////
|
|
|
|
|
|
|
|
class DrawLine3D: public Node
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
/** creates and initialize a node */
|
|
|
|
static DrawLine3D* create();
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Draw 3D Line
|
|
|
|
*/
|
|
|
|
void drawLine(const Vec3 &from, const Vec3 &to, const Color4F &color);
|
|
|
|
|
|
|
|
/** Clear the geometry in the node's buffer. */
|
|
|
|
void clear()
|
|
|
|
{
|
|
|
|
_buffer.clear();
|
|
|
|
}
|
|
|
|
|
|
|
|
void onDraw(const Mat4 &transform, uint32_t flags);
|
|
|
|
|
|
|
|
// Overrides
|
|
|
|
virtual void draw(Renderer *renderer, const Mat4 &transform, uint32_t flags) override;
|
|
|
|
|
|
|
|
CC_CONSTRUCTOR_ACCESS:
|
|
|
|
DrawLine3D()
|
|
|
|
{
|
|
|
|
|
|
|
|
}
|
|
|
|
virtual ~DrawLine3D()
|
|
|
|
{
|
|
|
|
|
|
|
|
}
|
|
|
|
virtual bool init();
|
|
|
|
|
|
|
|
protected:
|
|
|
|
struct V3F_C4B
|
|
|
|
{
|
|
|
|
Vec3 vertices;
|
|
|
|
Color4B colors;
|
|
|
|
};
|
|
|
|
|
|
|
|
std::vector<V3F_C4B> _buffer;
|
|
|
|
|
|
|
|
CustomCommand _customCommand;
|
|
|
|
|
|
|
|
private:
|
|
|
|
CC_DISALLOW_COPY_AND_ASSIGN(DrawLine3D);
|
|
|
|
};
|
|
|
|
|
|
|
|
DrawLine3D* DrawLine3D::create()
|
|
|
|
{
|
2014-08-28 07:31:57 +08:00
|
|
|
auto ret = new (std::nothrow) DrawLine3D();
|
2014-08-08 15:01:24 +08:00
|
|
|
if (ret && ret->init())
|
|
|
|
return ret;
|
|
|
|
CC_SAFE_DELETE(ret);
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool DrawLine3D::init()
|
|
|
|
{
|
|
|
|
setGLProgramState(GLProgramState::getOrCreateWithGLProgramName(GLProgram::SHADER_NAME_POSITION_COLOR));
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void DrawLine3D::drawLine(const Vec3 &from, const Vec3 &to, const Color4F &color)
|
|
|
|
{
|
|
|
|
Color4B col = Color4B(color);
|
|
|
|
DrawLine3D::V3F_C4B vertex;
|
|
|
|
vertex.vertices = from;
|
|
|
|
vertex.colors = col;
|
|
|
|
_buffer.push_back(vertex);
|
|
|
|
vertex.vertices = to;
|
|
|
|
_buffer.push_back(vertex);
|
|
|
|
}
|
|
|
|
|
|
|
|
void DrawLine3D::draw(Renderer *renderer, const Mat4 &transform, uint32_t flags)
|
|
|
|
{
|
|
|
|
_customCommand.init(_globalZOrder);
|
|
|
|
_customCommand.func = CC_CALLBACK_0(DrawLine3D::onDraw, this, transform, flags);
|
|
|
|
renderer->addCommand(&_customCommand);
|
|
|
|
}
|
|
|
|
|
|
|
|
void DrawLine3D::onDraw(const Mat4 &transform, uint32_t flags)
|
|
|
|
{
|
|
|
|
auto glProgram = getGLProgram();
|
|
|
|
glProgram->use();
|
|
|
|
glProgram->setUniformsForBuiltins(transform);
|
|
|
|
glEnable(GL_DEPTH_TEST);
|
|
|
|
glBindBuffer(GL_ARRAY_BUFFER, 0);
|
|
|
|
glEnableVertexAttribArray(GLProgram::VERTEX_ATTRIB_POSITION);
|
|
|
|
glVertexAttribPointer(GLProgram::VERTEX_ATTRIB_POSITION, 3, GL_FLOAT, GL_FALSE, sizeof(V3F_C4B), &(_buffer[0].vertices));
|
|
|
|
|
|
|
|
glEnableVertexAttribArray(GLProgram::VERTEX_ATTRIB_COLOR);
|
|
|
|
glVertexAttribPointer(GLProgram::VERTEX_ATTRIB_COLOR, 4, GL_UNSIGNED_BYTE, GL_TRUE, sizeof(V3F_C4B), &(_buffer[0].colors));
|
2014-08-21 16:21:23 +08:00
|
|
|
glDrawArrays(GL_LINES, 0, static_cast<int>(_buffer.size()));
|
2014-08-08 15:01:24 +08:00
|
|
|
glDisable(GL_DEPTH_TEST);
|
|
|
|
}
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
2014-08-07 15:23:31 +08:00
|
|
|
enum
|
|
|
|
{
|
|
|
|
IDC_NEXT = 100,
|
|
|
|
IDC_BACK,
|
|
|
|
IDC_RESTART
|
|
|
|
};
|
|
|
|
|
|
|
|
static int sceneIdx = -1;
|
|
|
|
|
|
|
|
static std::function<Layer*()> createFunctions[] =
|
|
|
|
{
|
|
|
|
CL(Camera3DTestDemo),
|
|
|
|
};
|
|
|
|
#define MAX_LAYER (sizeof(createFunctions) / sizeof(createFunctions[0]))
|
|
|
|
|
|
|
|
static Layer* nextSpriteTestAction()
|
|
|
|
{
|
|
|
|
sceneIdx++;
|
|
|
|
sceneIdx = sceneIdx % MAX_LAYER;
|
|
|
|
auto layer = (createFunctions[sceneIdx])();
|
|
|
|
return layer;
|
|
|
|
}
|
|
|
|
|
|
|
|
static Layer* backSpriteTestAction()
|
|
|
|
{
|
|
|
|
sceneIdx--;
|
|
|
|
int total = MAX_LAYER;
|
|
|
|
if( sceneIdx < 0 )
|
|
|
|
sceneIdx += total;
|
|
|
|
|
|
|
|
auto layer = (createFunctions[sceneIdx])();
|
|
|
|
return layer;
|
|
|
|
}
|
|
|
|
|
|
|
|
static Layer* restartSpriteTestAction()
|
|
|
|
{
|
|
|
|
auto layer = (createFunctions[sceneIdx])();
|
|
|
|
return layer;
|
|
|
|
}
|
|
|
|
|
|
|
|
//------------------------------------------------------------------
|
|
|
|
//
|
|
|
|
// SpriteTestDemo
|
|
|
|
//
|
|
|
|
//------------------------------------------------------------------
|
|
|
|
|
|
|
|
Camera3DTestDemo::Camera3DTestDemo(void)
|
|
|
|
: BaseTest()
|
|
|
|
, _camera(nullptr)
|
2014-08-11 10:55:14 +08:00
|
|
|
, _incRot(nullptr)
|
|
|
|
, _decRot(nullptr)
|
2014-08-07 15:23:31 +08:00
|
|
|
{
|
|
|
|
}
|
|
|
|
Camera3DTestDemo::~Camera3DTestDemo(void)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
void Camera3DTestDemo::reachEndCallBack()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
std::string Camera3DTestDemo::title() const
|
|
|
|
{
|
|
|
|
return "Testing Camera";
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string Camera3DTestDemo::subtitle() const
|
|
|
|
{
|
|
|
|
return "";
|
|
|
|
}
|
|
|
|
void Camera3DTestDemo::scaleCameraCallback(Ref* sender,float value)
|
|
|
|
{
|
|
|
|
if(_camera&& _cameraType!=CameraType::FirstCamera)
|
|
|
|
{
|
|
|
|
Vec3 cameraPos= _camera->getPosition3D();
|
|
|
|
cameraPos+= cameraPos.getNormalized()*value;
|
|
|
|
_camera->setPosition3D(cameraPos);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
void Camera3DTestDemo::rotateCameraCallback(Ref* sender,float value)
|
|
|
|
{
|
|
|
|
if(_cameraType==CameraType::FreeCamera || _cameraType==CameraType::FirstCamera)
|
|
|
|
{
|
|
|
|
Vec3 rotation3D= _camera->getRotation3D();
|
|
|
|
rotation3D.y+= value;
|
|
|
|
_camera->setRotation3D(rotation3D);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
void Camera3DTestDemo::SwitchViewCallback(Ref* sender, CameraType cameraType)
|
|
|
|
{
|
|
|
|
if(_cameraType==cameraType)
|
|
|
|
{
|
|
|
|
return ;
|
|
|
|
}
|
|
|
|
_cameraType = cameraType;
|
|
|
|
if(_cameraType==CameraType::FreeCamera)
|
|
|
|
{
|
|
|
|
_camera->setPosition3D(Vec3(0, 130, 130) + _sprite3D->getPosition3D());
|
|
|
|
_camera->lookAt(_sprite3D->getPosition3D(), Vec3(0,1,0));
|
2014-08-11 10:55:14 +08:00
|
|
|
_incRot->setEnabled(true);
|
|
|
|
_decRot->setEnabled(true);
|
2014-08-07 15:23:31 +08:00
|
|
|
}
|
|
|
|
else if(_cameraType==CameraType::FirstCamera)
|
|
|
|
{
|
2014-08-11 10:55:14 +08:00
|
|
|
Vec3 newFaceDir;
|
|
|
|
_sprite3D->getWorldToNodeTransform().getForwardVector(&newFaceDir);
|
|
|
|
newFaceDir.normalize();
|
|
|
|
_camera->setPosition3D(Vec3(0,35,0) + _sprite3D->getPosition3D());
|
|
|
|
_camera->lookAt(_sprite3D->getPosition3D() + newFaceDir*50, Vec3(0, 1, 0));
|
|
|
|
_incRot->setEnabled(true);
|
|
|
|
_decRot->setEnabled(true);
|
2014-08-07 15:23:31 +08:00
|
|
|
}
|
|
|
|
else if(_cameraType==CameraType::ThirdCamera)
|
|
|
|
{
|
2014-08-11 10:55:14 +08:00
|
|
|
_camera->setPosition3D(Vec3(0, 130, 130) + _sprite3D->getPosition3D());
|
|
|
|
_camera->lookAt(_sprite3D->getPosition3D(), Vec3(0,1,0));
|
|
|
|
_incRot->setEnabled(false);
|
|
|
|
_decRot->setEnabled(false);
|
2014-08-07 15:23:31 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
void Camera3DTestDemo::onEnter()
|
|
|
|
{
|
|
|
|
BaseTest::onEnter();
|
|
|
|
_sprite3D=nullptr;
|
|
|
|
auto s = Director::getInstance()->getWinSize();
|
|
|
|
auto listener = EventListenerTouchAllAtOnce::create();
|
|
|
|
listener->onTouchesBegan = CC_CALLBACK_2(Camera3DTestDemo::onTouchesBegan, this);
|
|
|
|
listener->onTouchesMoved = CC_CALLBACK_2(Camera3DTestDemo::onTouchesMoved, this);
|
|
|
|
listener->onTouchesEnded = CC_CALLBACK_2(Camera3DTestDemo::onTouchesEnded, this);
|
|
|
|
_eventDispatcher->addEventListenerWithSceneGraphPriority(listener, this);
|
|
|
|
auto layer3D=Layer::create();
|
|
|
|
addChild(layer3D,0);
|
|
|
|
_layer3D=layer3D;
|
|
|
|
_curState=State_None;
|
|
|
|
addNewSpriteWithCoords( Vec3(0,0,0),"Sprite3DTest/girl.c3b",true,0.2,true);
|
|
|
|
TTFConfig ttfConfig("fonts/arial.ttf", 20);
|
|
|
|
auto label1 = Label::createWithTTF(ttfConfig,"zoom out");
|
|
|
|
auto menuItem1 = MenuItemLabel::create(label1, CC_CALLBACK_1(Camera3DTestDemo::scaleCameraCallback,this,1));
|
|
|
|
auto label2 = Label::createWithTTF(ttfConfig,"zoom in");
|
|
|
|
auto menuItem2 = MenuItemLabel::create(label2, CC_CALLBACK_1(Camera3DTestDemo::scaleCameraCallback,this,-1));
|
|
|
|
auto label3 = Label::createWithTTF(ttfConfig,"rotate+");
|
|
|
|
auto menuItem3 = MenuItemLabel::create(label3, CC_CALLBACK_1(Camera3DTestDemo::rotateCameraCallback,this,10));
|
2014-08-11 10:55:14 +08:00
|
|
|
_incRot = menuItem3;
|
2014-08-07 15:23:31 +08:00
|
|
|
auto label4 = Label::createWithTTF(ttfConfig,"rotate-");
|
|
|
|
auto menuItem4 = MenuItemLabel::create(label4, CC_CALLBACK_1(Camera3DTestDemo::rotateCameraCallback,this,-10));
|
2014-08-11 10:55:14 +08:00
|
|
|
_decRot = menuItem4;
|
2014-08-07 15:23:31 +08:00
|
|
|
auto label5 = Label::createWithTTF(ttfConfig,"free ");
|
|
|
|
auto menuItem5 = MenuItemLabel::create(label5, CC_CALLBACK_1(Camera3DTestDemo::SwitchViewCallback,this,CameraType::FreeCamera));
|
|
|
|
auto label6 = Label::createWithTTF(ttfConfig,"third person");
|
|
|
|
auto menuItem6 = MenuItemLabel::create(label6, CC_CALLBACK_1(Camera3DTestDemo::SwitchViewCallback,this,CameraType::ThirdCamera));
|
|
|
|
auto label7 = Label::createWithTTF(ttfConfig,"first person");
|
|
|
|
auto menuItem7 = MenuItemLabel::create(label7, CC_CALLBACK_1(Camera3DTestDemo::SwitchViewCallback,this,CameraType::FirstCamera));
|
|
|
|
auto menu = Menu::create(menuItem1,menuItem2,menuItem3,menuItem4,menuItem5,menuItem6,menuItem7,NULL);
|
|
|
|
|
|
|
|
menu->setPosition(Vec2::ZERO);
|
2014-08-28 11:41:18 +08:00
|
|
|
menuItem1->setPosition(s.width-50, VisibleRect::top().y-50 );
|
|
|
|
menuItem2->setPosition(s.width-50, VisibleRect::top().y-100);
|
|
|
|
menuItem3->setPosition(s.width-50, VisibleRect::top().y-150);
|
|
|
|
menuItem4->setPosition(s.width-50, VisibleRect::top().y-200);
|
|
|
|
menuItem5->setPosition(VisibleRect::left().x+100, VisibleRect::top().y-50);
|
|
|
|
menuItem6->setPosition(VisibleRect::left().x+100, VisibleRect::top().y -100);
|
|
|
|
menuItem7->setPosition(VisibleRect::left().x+100, VisibleRect::top().y -150);
|
2014-08-07 15:23:31 +08:00
|
|
|
addChild(menu, 0);
|
2014-10-04 00:38:36 +08:00
|
|
|
schedule(CC_SCHEDULE_SELECTOR(Camera3DTestDemo::updateCamera), 0.0f);
|
2014-08-07 15:23:31 +08:00
|
|
|
if (_camera == nullptr)
|
|
|
|
{
|
|
|
|
_camera=Camera::createPerspective(60, (GLfloat)s.width/s.height, 1, 1000);
|
|
|
|
_camera->setCameraFlag(CameraFlag::USER1);
|
|
|
|
_layer3D->addChild(_camera);
|
|
|
|
}
|
|
|
|
SwitchViewCallback(this,CameraType::ThirdCamera);
|
2014-08-08 15:01:24 +08:00
|
|
|
DrawLine3D* line =DrawLine3D::create();
|
2014-08-07 15:23:31 +08:00
|
|
|
//draw x
|
|
|
|
for( int j =-20; j<=20 ;j++)
|
|
|
|
{
|
|
|
|
line->drawLine(Vec3(-100, 0, 5*j),Vec3(100,0,5*j),Color4F(1,0,0,1));
|
|
|
|
}
|
|
|
|
//draw z
|
|
|
|
for( int j =-20; j<=20 ;j++)
|
|
|
|
{
|
|
|
|
line->drawLine(Vec3(5*j, 0, -100),Vec3(5*j,0,100),Color4F(0,0,1,1));
|
|
|
|
}
|
|
|
|
//draw y
|
|
|
|
line->drawLine(Vec3(0, -50, 0),Vec3(0,0,0),Color4F(0,0.5,0,1));
|
|
|
|
line->drawLine(Vec3(0, 0, 0),Vec3(0,50,0),Color4F(0,1,0,1));
|
|
|
|
_layer3D->addChild(line);
|
|
|
|
_layer3D->setCameraMask(2);
|
|
|
|
}
|
|
|
|
void Camera3DTestDemo::onExit()
|
|
|
|
{
|
|
|
|
BaseTest::onExit();
|
|
|
|
if (_camera)
|
|
|
|
{
|
|
|
|
_camera = nullptr;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Camera3DTestDemo::restartCallback(Ref* sender)
|
|
|
|
{
|
2014-08-28 07:31:57 +08:00
|
|
|
auto s = new (std::nothrow) Camera3DTestScene();
|
2014-08-07 15:23:31 +08:00
|
|
|
s->addChild(restartSpriteTestAction());
|
|
|
|
|
|
|
|
Director::getInstance()->replaceScene(s);
|
|
|
|
s->release();
|
|
|
|
}
|
|
|
|
|
|
|
|
void Camera3DTestDemo::nextCallback(Ref* sender)
|
|
|
|
{
|
2014-08-28 07:31:57 +08:00
|
|
|
auto s = new (std::nothrow) Camera3DTestScene();
|
2014-08-07 15:23:31 +08:00
|
|
|
s->addChild( nextSpriteTestAction() );
|
|
|
|
Director::getInstance()->replaceScene(s);
|
|
|
|
s->release();
|
|
|
|
}
|
|
|
|
void Camera3DTestDemo::backCallback(Ref* sender)
|
|
|
|
{
|
2014-08-28 07:31:57 +08:00
|
|
|
auto s = new (std::nothrow) Camera3DTestScene();
|
2014-08-07 15:23:31 +08:00
|
|
|
s->addChild( backSpriteTestAction() );
|
|
|
|
Director::getInstance()->replaceScene(s);
|
|
|
|
s->release();
|
|
|
|
}
|
|
|
|
void Camera3DTestDemo::addNewSpriteWithCoords(Vec3 p,std::string fileName,bool playAnimation,float scale,bool bindCamera)
|
|
|
|
{
|
|
|
|
|
|
|
|
auto sprite = Sprite3D::create(fileName);
|
|
|
|
_layer3D->addChild(sprite);
|
|
|
|
float globalZOrder=sprite->getGlobalZOrder();
|
|
|
|
sprite->setPosition3D( Vec3( p.x, p.y,p.z) );
|
|
|
|
sprite->setGlobalZOrder(globalZOrder);
|
|
|
|
if(playAnimation)
|
|
|
|
{
|
|
|
|
auto animation = Animation3D::create(fileName,"Take 001");
|
|
|
|
if (animation)
|
|
|
|
{
|
|
|
|
auto animate = Animate3D::create(animation);
|
|
|
|
sprite->runAction(RepeatForever::create(animate));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if(bindCamera)
|
|
|
|
{
|
|
|
|
_sprite3D=sprite;
|
|
|
|
}
|
|
|
|
sprite->setScale(scale);
|
|
|
|
|
|
|
|
}
|
|
|
|
void Camera3DTestDemo::onTouchesBegan(const std::vector<Touch*>& touches, cocos2d::Event *event)
|
|
|
|
{
|
|
|
|
for ( auto &item: touches )
|
|
|
|
{
|
|
|
|
auto touch = item;
|
|
|
|
auto location = touch->getLocation();
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
void Camera3DTestDemo::onTouchesMoved(const std::vector<Touch*>& touches, cocos2d::Event *event)
|
|
|
|
{
|
|
|
|
if(touches.size()==1)
|
|
|
|
{
|
|
|
|
auto touch = touches[0];
|
|
|
|
auto location = touch->getLocation();
|
|
|
|
Point newPos = touch->getPreviousLocation()-location;
|
|
|
|
if(_cameraType==CameraType::FreeCamera || _cameraType==CameraType::FirstCamera)
|
|
|
|
{
|
|
|
|
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.1;
|
|
|
|
cameraPos+=cameraRightDir*newPos.x*0.1;
|
|
|
|
_camera->setPosition3D(cameraPos);
|
|
|
|
if(_sprite3D && _cameraType==CameraType::FirstCamera)
|
|
|
|
{
|
|
|
|
_sprite3D->setPosition3D(Vec3(_camera->getPositionX(),0,_camera->getPositionZ()));
|
|
|
|
_targetPos=_sprite3D->getPosition3D();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
void Camera3DTestDemo::move3D(float elapsedTime)
|
|
|
|
{
|
|
|
|
if(_sprite3D)
|
|
|
|
{
|
|
|
|
Vec3 curPos= _sprite3D->getPosition3D();
|
|
|
|
Vec3 newFaceDir = _targetPos - curPos;
|
|
|
|
newFaceDir.y = 0.0f;
|
|
|
|
newFaceDir.normalize();
|
|
|
|
Vec3 offset = newFaceDir * 25.0f * elapsedTime;
|
|
|
|
curPos+=offset;
|
|
|
|
_sprite3D->setPosition3D(curPos);
|
|
|
|
offset.x=offset.x;
|
|
|
|
offset.z=offset.z;
|
|
|
|
if(_cameraType==CameraType::ThirdCamera)
|
|
|
|
{
|
|
|
|
Vec3 cameraPos= _camera->getPosition3D();
|
|
|
|
cameraPos.x+=offset.x;
|
|
|
|
cameraPos.z+=offset.z;
|
|
|
|
_camera->setPosition3D(cameraPos);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
void Camera3DTestDemo::updateState(float elapsedTime)
|
|
|
|
{
|
|
|
|
if(_sprite3D)
|
|
|
|
{
|
|
|
|
Vec3 curPos= _sprite3D->getPosition3D();
|
|
|
|
Vec3 curFaceDir;
|
|
|
|
_sprite3D->getNodeToWorldTransform().getForwardVector(&curFaceDir);
|
|
|
|
curFaceDir=-curFaceDir;
|
|
|
|
curFaceDir.normalize();
|
|
|
|
Vec3 newFaceDir = _targetPos - curPos;
|
|
|
|
newFaceDir.y = 0.0f;
|
|
|
|
newFaceDir.normalize();
|
|
|
|
float cosAngle = std::fabs(Vec3::dot(curFaceDir,newFaceDir) - 1.0f);
|
|
|
|
float dist = curPos.distanceSquared(_targetPos);
|
|
|
|
if(dist<=4.0f)
|
|
|
|
{
|
|
|
|
if(cosAngle<=0.01f)
|
|
|
|
_curState = State_Idle;
|
|
|
|
else
|
|
|
|
_curState = State_Rotate;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if(cosAngle>0.01f)
|
|
|
|
_curState = State_Rotate | State_Move;
|
|
|
|
else
|
|
|
|
_curState = State_Move;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
void Camera3DTestDemo::onTouchesEnded(const std::vector<Touch*>& touches, cocos2d::Event *event)
|
|
|
|
{
|
|
|
|
for ( auto &item: touches )
|
|
|
|
{
|
|
|
|
auto touch = item;
|
|
|
|
auto location = touch->getLocationInView();
|
|
|
|
if(_camera)
|
|
|
|
{
|
|
|
|
if(_sprite3D && _cameraType==CameraType::ThirdCamera)
|
|
|
|
{
|
|
|
|
Vec3 nearP(location.x, location.y, -1.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);
|
|
|
|
float dist=0.0f;
|
|
|
|
float ndd = Vec3::dot(Vec3(0,1,0),dir);
|
|
|
|
if(ndd == 0)
|
|
|
|
dist=0.0f;
|
|
|
|
float ndo = Vec3::dot(Vec3(0,1,0),nearP);
|
|
|
|
dist= (0 - ndo) / ndd;
|
|
|
|
Vec3 p = nearP + dist * dir;
|
|
|
|
_targetPos=p;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
void onTouchesCancelled(const std::vector<Touch*>& touches, cocos2d::Event *event)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
void Camera3DTestDemo::updateCamera(float fDelta)
|
|
|
|
{
|
|
|
|
if(_sprite3D)
|
|
|
|
{
|
|
|
|
if( _cameraType==CameraType::ThirdCamera)
|
|
|
|
{
|
|
|
|
updateState(fDelta);
|
|
|
|
if(isState(_curState,State_Move))
|
|
|
|
{
|
|
|
|
move3D(fDelta);
|
|
|
|
if(isState(_curState,State_Rotate))
|
|
|
|
{
|
|
|
|
Vec3 curPos = _sprite3D->getPosition3D();
|
|
|
|
|
|
|
|
Vec3 newFaceDir = _targetPos - curPos;
|
|
|
|
newFaceDir.y = 0;
|
|
|
|
newFaceDir.normalize();
|
|
|
|
Vec3 up;
|
|
|
|
_sprite3D->getNodeToWorldTransform().getUpVector(&up);
|
|
|
|
up.normalize();
|
|
|
|
Vec3 right;
|
|
|
|
Vec3::cross(-newFaceDir,up,&right);
|
|
|
|
right.normalize();
|
|
|
|
Vec3 pos = Vec3(0,0,0);
|
|
|
|
Mat4 mat;
|
|
|
|
mat.m[0] = right.x;
|
|
|
|
mat.m[1] = right.y;
|
|
|
|
mat.m[2] = right.z;
|
|
|
|
mat.m[3] = 0.0f;
|
|
|
|
|
|
|
|
mat.m[4] = up.x;
|
|
|
|
mat.m[5] = up.y;
|
|
|
|
mat.m[6] = up.z;
|
|
|
|
mat.m[7] = 0.0f;
|
|
|
|
|
|
|
|
mat.m[8] = newFaceDir.x;
|
|
|
|
mat.m[9] = newFaceDir.y;
|
|
|
|
mat.m[10] = newFaceDir.z;
|
|
|
|
mat.m[11] = 0.0f;
|
|
|
|
|
|
|
|
mat.m[12] = pos.x;
|
|
|
|
mat.m[13] = pos.y;
|
|
|
|
mat.m[14] = pos.z;
|
|
|
|
mat.m[15] = 1.0f;
|
|
|
|
_sprite3D->setAdditionalTransform(&mat);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
bool Camera3DTestDemo::isState(unsigned int state,unsigned int bit) const
|
|
|
|
{
|
|
|
|
return (state & bit) == bit;
|
|
|
|
}
|
|
|
|
void Camera3DTestScene::runThisTest()
|
|
|
|
{
|
|
|
|
auto layer = nextSpriteTestAction();
|
|
|
|
addChild(layer);
|
|
|
|
Director::getInstance()->replaceScene(this);
|
|
|
|
}
|