/**************************************************************************** Copyright (c) 2012 cocos2d-x.org Copyright (c) 2013-2016 Chukong Technologies Inc. Copyright (c) 2017-2018 Xiamen Yaji Software Co., Ltd. https://adxeproject.github.io/ 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 "testResource.h" #include "ui/UISlider.h" #include "platform/CCFileUtils.h" #include "renderer/backend/Device.h" USING_NS_CC; #define SET_UNIFORM(name, addr, size) \ do \ { \ auto _loc_ = _programState1->getUniformLocation(name); \ _programState1->setUniform(_loc_, (addr), (size)); \ _loc_ = _programState2->getUniformLocation(name); \ _programState2->setUniform(_loc_, (addr), (size)); \ } while (false) enum { IDC_NEXT = 100, IDC_BACK, IDC_RESTART }; Camera3DTests::Camera3DTests() { ADD_TEST_CASE(CameraRotationTest); ADD_TEST_CASE(Camera3DTestDemo); ADD_TEST_CASE(CameraCullingDemo); ADD_TEST_CASE(FogTestDemo); ADD_TEST_CASE(CameraArcBallDemo); // ADD_TEST_CASE(CameraFrameBufferTest); //TODO render target ADD_TEST_CASE(BackgroundColorBrushTest); } //------------------------------------------------------------------ // // Camera Rotation Test // //------------------------------------------------------------------ CameraRotationTest::CameraRotationTest() { auto s = Director::getInstance()->getWinSize(); _camControlNode = Node::create(); _camControlNode->setPositionNormalized(Vec2(0.5f, 0.5f)); addChild(_camControlNode); _camNode = Node::create(); _camNode->setPositionZ(Camera::getDefaultCamera()->getPosition3D().z); _camControlNode->addChild(_camNode); auto mesh = MeshRenderer::create(); mesh->setPosition(s.width / 2, s.height / 2); addChild(mesh); auto lship = Label::create(); lship->setString("Ship"); lship->setPosition(0, 20); mesh->addChild(lship); // Billboards // Yellow is at the back bill1 = BillBoard::create("Images/Icon.png"); bill1->setPosition3D(Vec3(50.0f, 10.0f, -10.0f)); bill1->setColor(Color3B::YELLOW); bill1->setScale(0.6f); mesh->addChild(bill1); l1 = Label::create(); l1->setPosition(Vec2(0.0f, -10.0f)); l1->setString("Billboard1"); l1->setColor(Color3B::WHITE); l1->setScale(3); bill1->addChild(l1); auto p1 = ParticleSystemQuad::create("Particles/SmallSun.plist"); p1->setPosition(30.0f, 80.0f); bill1->addChild(p1); bill2 = BillBoard::create("Images/Icon.png"); bill2->setPosition3D(Vec3(-50.0f, -10.0f, 10.0f)); bill2->setScale(0.6f); mesh->addChild(bill2); l2 = Label::create(); l2->setString("Billboard2"); l2->setPosition(Vec2(0.0f, -10.0f)); l2->setColor(Color3B::WHITE); l2->setScale(3); bill2->addChild(l2); auto p2 = ParticleSystemQuad::create("Particles/SmallSun.plist"); p2->setPosition(30, 80); bill2->addChild(p2); // 3D models auto model = MeshRenderer::create("MeshRendererTest/boss1.obj"); model->setScale(4); model->setTexture("MeshRendererTest/boss.png"); model->setPosition3D(Vec3(s.width / 2, s.height / 2, 0)); addChild(model); // Listener _lis = EventListenerTouchOneByOne::create(); _lis->onTouchBegan = [](Touch* t, Event* e) { return true; }; _lis->onTouchMoved = [this](Touch* t, Event* e) { float dx = t->getDelta().x; Vec3 rot = _camControlNode->getRotation3D(); rot.y += dx; _camControlNode->setRotation3D(rot); Vec3 worldPos; _camNode->getNodeToWorldTransform().getTranslation(&worldPos); Camera::getDefaultCamera()->setPosition3D(worldPos); Camera::getDefaultCamera()->lookAt(_camControlNode->getPosition3D()); }; Director::getInstance()->getEventDispatcher()->addEventListenerWithSceneGraphPriority(_lis, this); schedule(CC_SCHEDULE_SELECTOR(CameraRotationTest::update)); } CameraRotationTest::~CameraRotationTest() { Director::getInstance()->getEventDispatcher()->removeEventListener(_lis); } std::string CameraRotationTest::title() const { return "Camera Rotation Test"; } std::string CameraRotationTest::subtitle() const { return "Slide to rotate"; } void CameraRotationTest::onEnter() { CameraBaseTest::onEnter(); } void CameraRotationTest::onExit() { CameraBaseTest::onExit(); } void CameraRotationTest::update(float dt) {} //------------------------------------------------------------------ // // Camera3DTestDemo // //------------------------------------------------------------------ Camera3DTestDemo::Camera3DTestDemo() : _cameraType(CameraType::Free) , _incRot(nullptr) , _decRot(nullptr) , _camera(nullptr) , _bZoomOut(false) , _bZoomIn(false) , _bRotateLeft(false) , _bRotateRight(false) {} Camera3DTestDemo::~Camera3DTestDemo() {} void Camera3DTestDemo::reachEndCallBack() {} std::string Camera3DTestDemo::title() const { return "Testing Camera"; } void Camera3DTestDemo::scaleCameraCallback(Ref* sender, float value) { if (_camera && _cameraType != CameraType::FirstPerson) { Vec3 cameraPos = _camera->getPosition3D(); cameraPos += cameraPos.getNormalized() * value; _camera->setPosition3D(cameraPos); } } void Camera3DTestDemo::rotateCameraCallback(Ref* sender, float value) { if (_cameraType == CameraType::Free || _cameraType == CameraType::FirstPerson) { 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::Free) { _camera->setPosition3D(Vec3(0, 130, 130) + _mesh->getPosition3D()); _camera->lookAt(_mesh->getPosition3D()); _RotateRightlabel->setColor(Color3B::WHITE); _RotateLeftlabel->setColor(Color3B::WHITE); _ZoomInlabel->setColor(Color3B::WHITE); _ZoomOutlabel->setColor(Color3B::WHITE); } else if (_cameraType == CameraType::FirstPerson) { Vec3 newFaceDir; _mesh->getWorldToNodeTransform().getForwardVector(&newFaceDir); newFaceDir.normalize(); _camera->setPosition3D(Vec3(0, 35, 0) + _mesh->getPosition3D()); _camera->lookAt(_mesh->getPosition3D() + newFaceDir * 50); _RotateRightlabel->setColor(Color3B::WHITE); _RotateLeftlabel->setColor(Color3B::WHITE); _ZoomInlabel->setColor(Color3B::GRAY); _ZoomOutlabel->setColor(Color3B::GRAY); } else if (_cameraType == CameraType::ThirdPerson) { _camera->setPosition3D(Vec3(0, 130, 130) + _mesh->getPosition3D()); _camera->lookAt(_mesh->getPosition3D()); _RotateRightlabel->setColor(Color3B::GRAY); _RotateLeftlabel->setColor(Color3B::GRAY); _ZoomInlabel->setColor(Color3B::WHITE); _ZoomOutlabel->setColor(Color3B::WHITE); } } void Camera3DTestDemo::onEnter() { CameraBaseTest::onEnter(); _mesh = 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), "MeshRendererTest/girl.c3b", true, 0.2f, true); TTFConfig ttfConfig("fonts/arial.ttf", 20); auto containerForLabel1 = Node::create(); _ZoomOutlabel = Label::createWithTTF(ttfConfig, "zoom out"); _ZoomOutlabel->setPosition(s.width - 50, VisibleRect::top().y - 20); containerForLabel1->addChild(_ZoomOutlabel); addChild(containerForLabel1, 10); auto listener1 = EventListenerTouchOneByOne::create(); listener1->setSwallowTouches(true); listener1->onTouchBegan = CC_CALLBACK_2(Camera3DTestDemo::onTouchesZoomOut, this); listener1->onTouchEnded = CC_CALLBACK_2(Camera3DTestDemo::onTouchesZoomOutEnd, this); _eventDispatcher->addEventListenerWithSceneGraphPriority(listener1, _ZoomOutlabel); auto containerForLabel2 = Node::create(); _ZoomInlabel = Label::createWithTTF(ttfConfig, "zoom in"); _ZoomInlabel->setPosition(s.width - 50, VisibleRect::top().y - 70); containerForLabel2->addChild(_ZoomInlabel); addChild(containerForLabel2, 10); auto listener2 = EventListenerTouchOneByOne::create(); listener2->setSwallowTouches(true); listener2->onTouchBegan = CC_CALLBACK_2(Camera3DTestDemo::onTouchesZoomIn, this); listener2->onTouchEnded = CC_CALLBACK_2(Camera3DTestDemo::onTouchesZoomInEnd, this); _eventDispatcher->addEventListenerWithSceneGraphPriority(listener2, _ZoomInlabel); auto containerForLabel3 = Node::create(); _RotateLeftlabel = Label::createWithTTF(ttfConfig, "rotate left"); _RotateLeftlabel->setPosition(s.width - 50, VisibleRect::top().y - 120); containerForLabel3->addChild(_RotateLeftlabel); addChild(containerForLabel3, 10); auto listener3 = EventListenerTouchOneByOne::create(); listener3->setSwallowTouches(true); listener3->onTouchBegan = CC_CALLBACK_2(Camera3DTestDemo::onTouchesRotateLeft, this); listener3->onTouchEnded = CC_CALLBACK_2(Camera3DTestDemo::onTouchesRotateLeftEnd, this); _eventDispatcher->addEventListenerWithSceneGraphPriority(listener3, _RotateLeftlabel); auto containerForLabel4 = Node::create(); _RotateRightlabel = Label::createWithTTF(ttfConfig, "rotate right"); _RotateRightlabel->setPosition(s.width - 50, VisibleRect::top().y - 170); containerForLabel4->addChild(_RotateRightlabel); addChild(containerForLabel4, 10); auto listener4 = EventListenerTouchOneByOne::create(); listener4->setSwallowTouches(true); listener4->onTouchBegan = CC_CALLBACK_2(Camera3DTestDemo::onTouchesRotateRight, this); listener4->onTouchEnded = CC_CALLBACK_2(Camera3DTestDemo::onTouchesRotateRightEnd, this); _eventDispatcher->addEventListenerWithSceneGraphPriority(listener4, _RotateRightlabel); auto label1 = Label::createWithTTF(ttfConfig, "free "); auto menuItem1 = MenuItemLabel::create(label1, CC_CALLBACK_1(Camera3DTestDemo::SwitchViewCallback, this, CameraType::Free)); auto label2 = Label::createWithTTF(ttfConfig, "third person"); auto menuItem2 = MenuItemLabel::create( label2, CC_CALLBACK_1(Camera3DTestDemo::SwitchViewCallback, this, CameraType::ThirdPerson)); auto label3 = Label::createWithTTF(ttfConfig, "first person"); auto menuItem3 = MenuItemLabel::create( label3, CC_CALLBACK_1(Camera3DTestDemo::SwitchViewCallback, this, CameraType::FirstPerson)); auto menu = Menu::create(menuItem1, menuItem2, menuItem3, nullptr); menu->setPosition(Vec2::ZERO); menuItem1->setPosition(VisibleRect::left().x + 100, VisibleRect::top().y - 50); menuItem2->setPosition(VisibleRect::left().x + 100, VisibleRect::top().y - 100); menuItem3->setPosition(VisibleRect::left().x + 100, VisibleRect::top().y - 150); addChild(menu, 0); schedule(CC_SCHEDULE_SELECTOR(Camera3DTestDemo::updateCamera), 0.0f); if (_camera == nullptr) { _camera = Camera::createPerspective(60, (float)s.width / s.height, 1, 1000); _camera->setCameraFlag(CameraFlag::USER1); _layer3D->addChild(_camera); } SwitchViewCallback(this, CameraType::ThirdPerson); DrawNode3D* line = DrawNode3D::create(); // draw x for (int j = -20; j <= 20; j++) { line->drawLine(Vec3(-100.0f, 0.0f, 5.0f * j), Vec3(100.0f, 0.0f, 5.0f * j), Color4F(1, 0, 0, 1)); } // draw z for (int j = -20; j <= 20; j++) { line->drawLine(Vec3(5.0f * j, 0.0f, -100.0f), Vec3(5.0f * j, 0.0f, 100.0f), Color4F(0, 0, 1, 1)); } // draw y line->drawLine(Vec3(0.0f, -50.0f, 0.0f), Vec3(0, 0, 0), Color4F(0, 0.5, 0, 1)); line->drawLine(Vec3(0, 0, 0), Vec3(0, 50.0f, 0), Color4F(0, 1, 0, 1)); _layer3D->addChild(line); _layer3D->setCameraMask(2); } void Camera3DTestDemo::onExit() { CameraBaseTest::onExit(); if (_camera) { _camera = nullptr; } } void Camera3DTestDemo::addNewSpriteWithCoords(Vec3 p, std::string fileName, bool playAnimation, float scale, bool bindCamera) { auto mesh = MeshRenderer::create(fileName); _layer3D->addChild(mesh); float globalZOrder = mesh->getGlobalZOrder(); mesh->setPosition3D(Vec3(p.x, p.y, p.z)); mesh->setGlobalZOrder(globalZOrder); if (playAnimation) { auto animation = Animation3D::create(fileName, "Take 001"); if (animation) { auto animate = Animate3D::create(animation); mesh->runAction(RepeatForever::create(animate)); } } if (bindCamera) { _mesh = mesh; } mesh->setScale(scale); } void Camera3DTestDemo::onTouchesBegan(const std::vector& touches, cocos2d::Event* event) {} void Camera3DTestDemo::onTouchesMoved(const std::vector& touches, cocos2d::Event* event) { if (touches.size() == 1) { auto touch = touches[0]; auto location = touch->getLocation(); Point newPos = touch->getPreviousLocation() - location; if (_cameraType == CameraType::Free || _cameraType == CameraType::FirstPerson) { 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.1f; cameraPos += cameraRightDir * newPos.x * 0.1f; _camera->setPosition3D(cameraPos); if (_mesh && _cameraType == CameraType::FirstPerson) { _mesh->setPosition3D(Vec3(_camera->getPositionX(), 0, _camera->getPositionZ())); _targetPos = _mesh->getPosition3D(); } } } } void Camera3DTestDemo::move3D(float elapsedTime) { if (_mesh) { Vec3 curPos = _mesh->getPosition3D(); Vec3 newFaceDir = _targetPos - curPos; newFaceDir.y = 0.0f; newFaceDir.normalize(); Vec3 offset = newFaceDir * 25.0f * elapsedTime; curPos += offset; _mesh->setPosition3D(curPos); if (_cameraType == CameraType::ThirdPerson) { Vec3 cameraPos = _camera->getPosition3D(); cameraPos.x += offset.x; cameraPos.z += offset.z; _camera->setPosition3D(cameraPos); } } } void Camera3DTestDemo::updateState(float elapsedTime) { if (_mesh) { Vec3 curPos = _mesh->getPosition3D(); Vec3 curFaceDir; _mesh->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& touches, cocos2d::Event* event) { for (auto& item : touches) { auto touch = item; auto location = touch->getLocationInView(); if (_camera) { if (_mesh && _cameraType == CameraType::ThirdPerson && _bZoomOut == false && _bZoomIn == false && _bRotateLeft == false && _bRotateRight == false) { Vec3 nearP(location.x, location.y, -1.0f), farP(location.x, location.y, 1.0f); auto size = Director::getInstance()->getWinSize(); nearP = _camera->unproject(nearP); farP = _camera->unproject(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; if (p.x > 100) p.x = 100; if (p.x < -100) p.x = -100; if (p.z > 100) p.z = 100; if (p.z < -100) p.z = -100; _targetPos = p; } } } } void onTouchesCancelled(const std::vector& touches, cocos2d::Event* event) {} void Camera3DTestDemo::updateCamera(float fDelta) { if (_mesh) { if (_cameraType == CameraType::ThirdPerson) { updateState(fDelta); if (isState(_curState, State_Move)) { move3D(fDelta); if (isState(_curState, State_Rotate)) { Vec3 curPos = _mesh->getPosition3D(); Vec3 newFaceDir = _targetPos - curPos; newFaceDir.y = 0; newFaceDir.normalize(); Vec3 up; _mesh->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; _mesh->setAdditionalTransform(&mat); } } } if (_bZoomOut == true) { if (_camera) { if (_cameraType == CameraType::ThirdPerson) { Vec3 lookDir = _camera->getPosition3D() - _mesh->getPosition3D(); Vec3 cameraPos = _camera->getPosition3D(); if (lookDir.length() <= 300) { cameraPos += lookDir.getNormalized(); _camera->setPosition3D(cameraPos); } } else if (_cameraType == CameraType::Free) { Vec3 cameraPos = _camera->getPosition3D(); if (cameraPos.length() <= 300) { cameraPos += cameraPos.getNormalized(); _camera->setPosition3D(cameraPos); } } } } if (_bZoomIn == true) { if (_camera) { if (_cameraType == CameraType::ThirdPerson) { Vec3 lookDir = _camera->getPosition3D() - _mesh->getPosition3D(); Vec3 cameraPos = _camera->getPosition3D(); if (lookDir.length() >= 50) { cameraPos -= lookDir.getNormalized(); _camera->setPosition3D(cameraPos); } } else if (_cameraType == CameraType::Free) { Vec3 cameraPos = _camera->getPosition3D(); if (cameraPos.length() >= 50) { cameraPos -= cameraPos.getNormalized(); _camera->setPosition3D(cameraPos); } } } } if (_bRotateLeft == true) { if (_cameraType == CameraType::Free || _cameraType == CameraType::FirstPerson) { Vec3 rotation3D = _camera->getRotation3D(); rotation3D.y += 1; _camera->setRotation3D(rotation3D); } } if (_bRotateRight == true) { if (_cameraType == CameraType::Free || _cameraType == CameraType::FirstPerson) { Vec3 rotation3D = _camera->getRotation3D(); rotation3D.y -= 1; _camera->setRotation3D(rotation3D); } } } } bool Camera3DTestDemo::onTouchesCommon(Touch* touch, Event* event, bool* touchProperty) { auto target = static_cast(event->getCurrentTarget()); Vec2 locationInNode = target->convertToNodeSpace(touch->getLocation()); Size s = target->getContentSize(); Rect rect = Rect(0, 0, s.width, s.height); if (rect.containsPoint(locationInNode)) { *touchProperty = true; return true; } return false; } bool Camera3DTestDemo::isState(unsigned int state, unsigned int bit) const { return (state & bit) == bit; } bool Camera3DTestDemo::onTouchesZoomOut(Touch* touch, Event* event) { return Camera3DTestDemo::onTouchesCommon(touch, event, &_bZoomOut); } void Camera3DTestDemo::onTouchesZoomOutEnd(Touch* touch, Event* event) { _bZoomOut = false; } bool Camera3DTestDemo::onTouchesZoomIn(Touch* touch, Event* event) { return Camera3DTestDemo::onTouchesCommon(touch, event, &_bZoomIn); } void Camera3DTestDemo::onTouchesZoomInEnd(Touch* touch, Event* event) { _bZoomIn = false; } bool Camera3DTestDemo::onTouchesRotateLeft(Touch* touch, Event* event) { return Camera3DTestDemo::onTouchesCommon(touch, event, &_bRotateLeft); } void Camera3DTestDemo::onTouchesRotateLeftEnd(Touch* touch, Event* event) { _bRotateLeft = false; } bool Camera3DTestDemo::onTouchesRotateRight(Touch* touch, Event* event) { return Camera3DTestDemo::onTouchesCommon(touch, event, &_bRotateRight); } void Camera3DTestDemo::onTouchesRotateRightEnd(Touch* touch, Event* event) { _bRotateRight = false; } //////////////////////////////////////////////////////////// // CameraCullingDemo CameraCullingDemo::CameraCullingDemo() : _layer3D(nullptr) , _cameraType(CameraType::FirstPerson) , _cameraFirst(nullptr) , _cameraThird(nullptr) , _moveAction(nullptr) , _drawAABB(nullptr) , _drawFrustum(nullptr) , _row(3) {} CameraCullingDemo::~CameraCullingDemo() {} std::string CameraCullingDemo::title() const { return "Camera Frustum Clipping"; } void CameraCullingDemo::onEnter() { CameraBaseTest::onEnter(); schedule(CC_SCHEDULE_SELECTOR(CameraCullingDemo::update), 0.0f); 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; // switch camera MenuItemFont::setFontName("fonts/arial.ttf"); MenuItemFont::setFontSize(20); auto menuItem1 = MenuItemFont::create("Switch Camera", CC_CALLBACK_1(CameraCullingDemo::switchViewCallback, this)); menuItem1->setColor(Color3B(0, 200, 20)); auto menu = Menu::create(menuItem1, NULL); menu->setPosition(Vec2::ZERO); menuItem1->setPosition(VisibleRect::left().x + 80, VisibleRect::top().y - 70); addChild(menu, 1); // + - MenuItemFont::setFontSize(40); auto decrease = MenuItemFont::create(" - ", CC_CALLBACK_1(CameraCullingDemo::delMeshCallback, this)); decrease->setColor(Color3B(0, 200, 20)); auto increase = MenuItemFont::create(" + ", CC_CALLBACK_1(CameraCullingDemo::addMeshCallback, this)); increase->setColor(Color3B(0, 200, 20)); menu = Menu::create(decrease, increase, nullptr); menu->alignItemsHorizontally(); menu->setPosition(Vec2(s.width - 60, VisibleRect::top().y - 70)); addChild(menu, 1); TTFConfig ttfCount("fonts/Marker Felt.ttf", 30); _labelMeshCount = Label::createWithTTF(ttfCount, "0 sprits"); _labelMeshCount->setColor(Color3B(0, 200, 20)); _labelMeshCount->setPosition(Vec2(s.width / 2, VisibleRect::top().y - 70)); addChild(_labelMeshCount); // aabb drawNode3D _drawAABB = DrawNode3D::create(); _drawAABB->setCameraMask((unsigned short)CameraFlag::USER1); addChild(_drawAABB); // frustum drawNode3D _drawFrustum = DrawNode3D::create(); _drawFrustum->setCameraMask((unsigned short)CameraFlag::USER1); addChild(_drawFrustum); // set camera switchViewCallback(this); // add sprite addMeshCallback(nullptr); } void CameraCullingDemo::onExit() { CameraBaseTest::onExit(); if (_cameraFirst) { _cameraFirst = nullptr; } if (_cameraThird) { _cameraThird = nullptr; } } void CameraCullingDemo::update(float dt) { _drawAABB->clear(); if (_cameraType == CameraType::ThirdPerson) drawCameraFrustum(); Vector& children = _layer3D->getChildren(); Vec3 corners[8]; for (const auto& iter : children) { const AABB& aabb = static_cast(iter)->getAABB(); if (_cameraFirst->isVisibleInFrustum(&aabb)) { aabb.getCorners(corners); _drawAABB->drawCube(corners, Color4F(0, 1, 0, 1)); } } } void CameraCullingDemo::reachEndCallBack() { _cameraFirst->stopActionByTag(100); auto inverse = MoveTo::create(4.f, Vec2(-_cameraFirst->getPositionX(), 0.0f)); inverse->retain(); _moveAction->release(); _moveAction = inverse; auto rot = RotateBy::create(1.f, Vec3(0.f, 180.f, 0.f)); auto seq = Sequence::create(rot, _moveAction, CallFunc::create(CC_CALLBACK_0(CameraCullingDemo::reachEndCallBack, this)), nullptr); seq->setTag(100); _cameraFirst->runAction(seq); } void CameraCullingDemo::switchViewCallback(Ref* sender) { auto s = Director::getInstance()->getWinSize(); if (_cameraFirst == nullptr) { _cameraFirst = Camera::createPerspective(30.0f, (float)s.width / s.height, 10.0f, 200.0f); _cameraFirst->setCameraFlag(CameraFlag::USER8); _cameraFirst->setPosition3D(Vec3(-100.0f, 0.0f, 0.0f)); _cameraFirst->lookAt(Vec3(1000.0f, 0.0f, 0.0f)); _moveAction = MoveTo::create(4.f, Vec2(-_cameraFirst->getPositionX(), 0.0f)); _moveAction->retain(); auto seq = Sequence::create( _moveAction, CallFunc::create(CC_CALLBACK_0(CameraCullingDemo::reachEndCallBack, this)), nullptr); seq->setTag(100); _cameraFirst->runAction(seq); addChild(_cameraFirst); } if (_cameraThird == nullptr) { _cameraThird = Camera::createPerspective(60, (float)s.width / s.height, 1, 1000); _cameraThird->setCameraFlag(CameraFlag::USER8); _cameraThird->setPosition3D(Vec3(0.0f, 130.0f, 130.0f)); _cameraThird->lookAt(Vec3(0, 0, 0)); addChild(_cameraThird); } if (_cameraType == CameraType::FirstPerson) { _cameraType = CameraType::ThirdPerson; _cameraThird->setCameraFlag(CameraFlag::USER1); _cameraFirst->setCameraFlag(CameraFlag::USER8); } else if (_cameraType == CameraType::ThirdPerson) { _cameraType = CameraType::FirstPerson; _cameraFirst->setCameraFlag(CameraFlag::USER1); _cameraThird->setCameraFlag(CameraFlag::USER8); _drawFrustum->clear(); } } void CameraCullingDemo::addMeshCallback(Ref* sender) { _layer3D->removeAllChildren(); _objects.clear(); _drawAABB->clear(); ++_row; for (int x = -_row; x < _row; x++) { for (int z = -_row; z < _row; z++) { auto sprite = MeshRenderer::create("MeshRendererTest/orc.c3b"); sprite->setPosition3D(Vec3(x * 30.0f, 0.0f, z * 30.0f)); sprite->setRotation3D(Vec3(0.0f, 180.0f, 0.0f)); _objects.push_back(sprite); _layer3D->addChild(sprite); } } // set layer mask. _layer3D->setCameraMask((unsigned short)CameraFlag::USER1); // update sprite number char szText[16]; sprintf(szText, "%d sprits", static_cast(_layer3D->getChildrenCount())); _labelMeshCount->setString(szText); } void CameraCullingDemo::delMeshCallback(Ref* sender) { if (_row == 0) return; _layer3D->removeAllChildren(); _objects.clear(); --_row; for (int x = -_row; x < _row; x++) { for (int z = -_row; z < _row; z++) { auto sprite = MeshRenderer::create("MeshRendererTest/orc.c3b"); sprite->setPosition3D(Vec3(x * 30.0f, 0.0f, z * 30.0f)); _objects.push_back(sprite); _layer3D->addChild(sprite); } } // set layer mask. _layer3D->setCameraMask((unsigned short)CameraFlag::USER1); // update sprite number char szText[16]; sprintf(szText, "%l sprits", static_cast(_layer3D->getChildrenCount())); _labelMeshCount->setString(szText); } void CameraCullingDemo::drawCameraFrustum() { _drawFrustum->clear(); auto size = Director::getInstance()->getWinSize(); Color4F color(1.f, 1.f, 0.f, 1); // top-left Vec3 tl_0, tl_1; Vec3 src(0, 0, 0); tl_0 = _cameraFirst->unproject(src); src = Vec3(0, 0, 1); tl_1 = _cameraFirst->unproject(src); // top-right Vec3 tr_0, tr_1; src = Vec3(size.width, 0, 0); tr_0 = _cameraFirst->unproject(src); src = Vec3(size.width, 0, 1); tr_1 = _cameraFirst->unproject(src); // bottom-left Vec3 bl_0, bl_1; src = Vec3(0, size.height, 0); bl_0 = _cameraFirst->unproject(src); src = Vec3(0, size.height, 1); bl_1 = _cameraFirst->unproject(src); // bottom-right Vec3 br_0, br_1; src = Vec3(size.width, size.height, 0); br_0 = _cameraFirst->unproject(src); src = Vec3(size.width, size.height, 1); br_1 = _cameraFirst->unproject(src); _drawFrustum->drawLine(tl_0, tl_1, color); _drawFrustum->drawLine(tr_0, tr_1, color); _drawFrustum->drawLine(bl_0, bl_1, color); _drawFrustum->drawLine(br_0, br_1, color); _drawFrustum->drawLine(tl_0, tr_0, color); _drawFrustum->drawLine(tr_0, br_0, color); _drawFrustum->drawLine(br_0, bl_0, color); _drawFrustum->drawLine(bl_0, tl_0, color); _drawFrustum->drawLine(tl_1, tr_1, color); _drawFrustum->drawLine(tr_1, br_1, color); _drawFrustum->drawLine(br_1, bl_1, color); _drawFrustum->drawLine(bl_1, tl_1, color); } //////////////////////////////////////////////////////////// // CameraArcBallDemo CameraArcBallDemo::CameraArcBallDemo() : CameraBaseTest() , _layer3D(nullptr) , _cameraType(CameraType::Free) , _camera(nullptr) , _drawGrid(nullptr) , _radius(1.0f) , _distanceZ(50.0f) , _operate(OperateCamType::RotateCamera) , _center(Vec3(0, 0, 0)) , _target(0) , _mesh1(nullptr) , _mesh2(nullptr) {} CameraArcBallDemo::~CameraArcBallDemo() {} std::string CameraArcBallDemo::title() const { return "Camera ArcBall Moving"; } void CameraArcBallDemo::onEnter() { CameraBaseTest::onEnter(); _rotationQuat.set(0.0f, 0.0f, 0.0f, 1.0f); schedule(CC_SCHEDULE_SELECTOR(CameraArcBallDemo::update), 0.0f); auto s = Director::getInstance()->getWinSize(); auto listener = EventListenerTouchAllAtOnce::create(); listener->onTouchesMoved = CC_CALLBACK_2(CameraArcBallDemo::onTouchsMoved, this); _eventDispatcher->addEventListenerWithSceneGraphPriority(listener, this); // switch camera MenuItemFont::setFontName("fonts/arial.ttf"); MenuItemFont::setFontSize(20); auto menuItem1 = MenuItemFont::create("Switch Operation", CC_CALLBACK_1(CameraArcBallDemo::switchOperateCallback, this)); menuItem1->setColor(Color3B(0, 200, 20)); auto menuItem2 = MenuItemFont::create("Switch Target", CC_CALLBACK_1(CameraArcBallDemo::switchTargetCallback, this)); menuItem2->setColor(Color3B(0, 200, 20)); auto menu = Menu::create(menuItem1, menuItem2, NULL); menu->setPosition(Vec2::ZERO); menuItem1->setPosition(VisibleRect::left().x + 80, VisibleRect::top().y - 70); menuItem2->setPosition(VisibleRect::left().x + 80, VisibleRect::top().y - 100); addChild(menu, 1); auto layer3D = Layer::create(); addChild(layer3D, 0); _layer3D = layer3D; if (_camera == nullptr) { _camera = Camera::createPerspective(60, (float)s.width / s.height, 1, 1000); _camera->setCameraFlag(CameraFlag::USER1); _camera->setPosition3D(Vec3(0.0f, 10.0f, 50.0f)); _camera->lookAt(Vec3(0, 0, 0), Vec3(0.0f, 1.0f, 0.0f)); _camera->retain(); _layer3D->addChild(_camera); } _mesh1 = MeshRenderer::create("MeshRendererTest/orc.c3b"); _mesh1->setScale(0.5); _mesh1->setRotation3D(Vec3(0.0f, 180.0f, 0.0f)); _mesh1->setPosition3D(Vec3(0, 0, 0)); _layer3D->addChild(_mesh1); _mesh2 = MeshRenderer::create("MeshRendererTest/boss.c3b"); _mesh2->setScale(0.6f); _mesh2->setRotation3D(Vec3(-90.0f, 0.0f, 0.0f)); _mesh2->setPosition3D(Vec3(20.0f, 0.0f, 0.0f)); _layer3D->addChild(_mesh2); _drawGrid = DrawNode3D::create(); // draw x for (int j = -20; j <= 20; j++) { _drawGrid->drawLine(Vec3(-100.0f, 0, 5.0f * j), Vec3(100.0f, 0, 5.0f * j), Color4F(1, 0, 0, 1)); } // draw z for (int j = -20; j <= 20; j++) { _drawGrid->drawLine(Vec3(5.0f * j, 0, -100.0f), Vec3(5.0f * j, 0, 100.0f), Color4F(0, 0, 1, 1)); } // draw y _drawGrid->drawLine(Vec3(0, 0, 0), Vec3(0, 50.0f, 0), Color4F(0, 1, 0, 1)); _layer3D->addChild(_drawGrid); _layer3D->setCameraMask(2); updateCameraTransform(); } void CameraArcBallDemo::onExit() { CameraBaseTest::onExit(); if (_camera) { _camera = nullptr; } } void CameraArcBallDemo::onTouchsMoved(const std::vector& touchs, Event* event) { if (!touchs.empty()) { if (_operate == OperateCamType::RotateCamera) // arc ball rotate { Size visibleSize = Director::getInstance()->getVisibleSize(); Vec2 prelocation = touchs[0]->getPreviousLocationInView(); Vec2 location = touchs[0]->getLocationInView(); location.x = 2.0f * (location.x) / (visibleSize.width) - 1.0f; location.y = 2.0f * (visibleSize.height - location.y) / (visibleSize.height) - 1.0f; prelocation.x = 2.0f * (prelocation.x) / (visibleSize.width) - 1.0f; prelocation.y = 2.0f * (visibleSize.height - prelocation.y) / (visibleSize.height) - 1.0f; Vec3 axes; float angle; calculateArcBall(axes, angle, prelocation.x, prelocation.y, location.x, location.y); // calculate rotation quaternion parameters Quaternion quat(axes, angle); // get rotation quaternion _rotationQuat = quat * _rotationQuat; updateCameraTransform(); // update camera Transform } else if (_operate == OperateCamType::MoveCamera) // camera zoom { Point newPos = touchs[0]->getPreviousLocation() - touchs[0]->getLocation(); _distanceZ -= newPos.y * 0.1f; updateCameraTransform(); } } } void CameraArcBallDemo::calculateArcBall(cocos2d::Vec3& axis, float& angle, float p1x, float p1y, float p2x, float p2y) { Mat4 rotation_matrix; Mat4::createRotation(_rotationQuat, &rotation_matrix); Vec3 uv = rotation_matrix * Vec3(0.0f, 1.0f, 0.0f); // rotation y Vec3 sv = rotation_matrix * Vec3(1.0f, 0.0f, 0.0f); // rotation x Vec3 lv = rotation_matrix * Vec3(0.0f, 0.0f, -1.0f); // rotation z Vec3 p1 = sv * p1x + uv * p1y - lv * projectToSphere(_radius, p1x, p1y); // start point screen transform to 3d Vec3 p2 = sv * p2x + uv * p2y - lv * projectToSphere(_radius, p2x, p2y); // end point screen transform to 3d Vec3::cross(p2, p1, &axis); // calculate rotation axis axis.normalize(); float t = (p2 - p1).length() / (2.0f * _radius); // clamp -1 to 1 if (t > 1.0) t = 1.0; if (t < -1.0) t = -1.0; angle = asin(t); // rotation angle } /* project an x,y pair onto a sphere of radius r or a hyperbolic sheet if we are away from the center of the sphere. */ float CameraArcBallDemo::projectToSphere(float r, float x, float y) { float d, t, z; d = sqrt(x * x + y * y); if (d < r * 0.70710678118654752440) // inside sphere { z = sqrt(r * r - d * d); } else // on hyperbola { t = r / 1.41421356237309504880f; z = t * t / d; } return z; } void CameraArcBallDemo::updateCameraTransform() { Mat4 trans, rot, center; Mat4::createTranslation(Vec3(0.0f, 10.0f, _distanceZ), &trans); Mat4::createRotation(_rotationQuat, &rot); Mat4::createTranslation(_center, ¢er); Mat4 result = center * rot * trans; _camera->setNodeToParentTransform(result); } void CameraArcBallDemo::switchOperateCallback(Ref* sender) { if (_operate == OperateCamType::MoveCamera) { _operate = OperateCamType::RotateCamera; } else if (_operate == OperateCamType::RotateCamera) { _operate = OperateCamType::MoveCamera; } } void CameraArcBallDemo::switchTargetCallback(Ref* sender) { if (_target == 0) { _target = 1; _center = _mesh2->getPosition3D(); updateCameraTransform(); } else if (_target == 1) { _target = 0; _center = _mesh1->getPosition3D(); updateCameraTransform(); } } void CameraArcBallDemo::update(float dt) { // updateCameraTransform(); } //////////////////////////////////////////////////////////// // FogTestDemo FogTestDemo::FogTestDemo() : CameraBaseTest() {} FogTestDemo::~FogTestDemo() { CC_SAFE_RELEASE_NULL(_programState1); CC_SAFE_RELEASE_NULL(_programState2); } std::string FogTestDemo::title() const { return "Fog Test Demo"; } void FogTestDemo::onEnter() { CameraBaseTest::onEnter(); schedule(CC_SCHEDULE_SELECTOR(FogTestDemo::update), 0.0f); Director::getInstance()->setClearColor(Color4F(0.5, 0.5, 0.5, 1)); auto s = Director::getInstance()->getWinSize(); auto listener = EventListenerTouchAllAtOnce::create(); listener->onTouchesMoved = CC_CALLBACK_2(FogTestDemo::onTouchesMoved, this); _eventDispatcher->addEventListenerWithSceneGraphPriority(listener, this); // switch fog type TTFConfig ttfConfig("fonts/arial.ttf", 20); auto label1 = Label::createWithTTF(ttfConfig, "Linear "); auto menuItem1 = MenuItemLabel::create(label1, CC_CALLBACK_1(FogTestDemo::switchTypeCallback, this, 0)); auto label2 = Label::createWithTTF(ttfConfig, "Exp"); auto menuItem2 = MenuItemLabel::create(label2, CC_CALLBACK_1(FogTestDemo::switchTypeCallback, this, 1)); auto label3 = Label::createWithTTF(ttfConfig, "Exp2"); auto menuItem3 = MenuItemLabel::create(label3, CC_CALLBACK_1(FogTestDemo::switchTypeCallback, this, 2)); auto menu = Menu::create(menuItem1, menuItem2, menuItem3, nullptr); menu->setPosition(Vec2::ZERO); menuItem1->setPosition(VisibleRect::left().x + 60, VisibleRect::top().y - 50); menuItem2->setPosition(VisibleRect::left().x + 60, VisibleRect::top().y - 100); menuItem3->setPosition(VisibleRect::left().x + 60, VisibleRect::top().y - 150); addChild(menu, 0); auto layer3D = Layer::create(); addChild(layer3D, 0); _layer3D = layer3D; CC_SAFE_RELEASE_NULL(_programState1); CC_SAFE_RELEASE_NULL(_programState2); auto vertexSource = FileUtils::getInstance()->getStringFromFile("MeshRendererTest/fog.vert"); auto fragSource = FileUtils::getInstance()->getStringFromFile("MeshRendererTest/fog.frag"); auto program = backend::Device::getInstance()->newProgram(vertexSource, fragSource); _programState1 = new backend::ProgramState(program); _programState2 = new backend::ProgramState(program); CC_SAFE_RELEASE(program); _mesh1 = MeshRenderer::create("MeshRendererTest/teapot.c3b"); _mesh2 = MeshRenderer::create("MeshRendererTest/teapot.c3b"); _mesh1->setProgramState(_programState1); _mesh2->setProgramState(_programState2); auto fogColor = Vec4(0.5, 0.5, 0.5, 1.0); float fogStart = 10; float fogEnd = 60; int fogEquation = 0; SET_UNIFORM("u_fogColor", &fogColor, sizeof(fogColor)); SET_UNIFORM("u_fogStart", &fogStart, sizeof(fogStart)); SET_UNIFORM("u_fogEnd", &fogEnd, sizeof(fogEnd)); SET_UNIFORM("u_fogEquation", &fogEquation, sizeof(fogEquation)); _layer3D->addChild(_mesh1); _mesh1->setPosition3D(Vec3(0, 0, 0)); _mesh1->setScale(2.0f); _mesh1->setRotation3D(Vec3(-90.0f, 180.0f, 0.0f)); _layer3D->addChild(_mesh2); _mesh2->setPosition3D(Vec3(0.0f, 0.0f, -20.0f)); _mesh2->setScale(2.0f); _mesh2->setRotation3D(Vec3(-90.0f, 180.0f, 0.0f)); if (_camera == nullptr) { _camera = Camera::createPerspective(60, (float)s.width / s.height, 1, 1000); _camera->setCameraFlag(CameraFlag::USER1); _camera->setPosition3D(Vec3(0.0f, 30.0f, 40.0f)); _camera->lookAt(Vec3(0, 0, 0), Vec3(0.0f, 1.0f, 0.0f)); _layer3D->addChild(_camera); } _layer3D->setCameraMask(2); #if (CC_TARGET_PLATFORM == CC_PLATFORM_ANDROID) _backToForegroundListener = EventListenerCustom::create(EVENT_RENDERER_RECREATED, [this](EventCustom*) { Director::getInstance()->setClearColor(Color4F(0.5, 0.5, 0.5, 1)); CC_SAFE_RELEASE_NULL(_programState1); CC_SAFE_RELEASE_NULL(_programState2); auto vertexSource = FileUtils::getInstance()->getStringFromFile("MeshRendererTest/fog.vert"); auto fragSource = FileUtils::getInstance()->getStringFromFile("MeshRendererTest/fog.frag"); auto program = backend::Device::getInstance()->newProgram(vertexSource, fragSource); _programState1 = new backend::ProgramState(program); _programState2 = new backend::ProgramState(program); _mesh1->setProgramState(_programState1); _mesh2->setProgramState(_programState2); CC_SAFE_RELEASE(program); auto fogColor = Vec4(0.5, 0.5, 0.5, 1.0); float fogStart = 10; float fogEnd = 60; int fogEquation = 0; SET_UNIFORM("u_fogColor", &fogColor, sizeof(fogColor)); SET_UNIFORM("u_fogStart", &fogStart, sizeof(fogStart)); SET_UNIFORM("u_fogEnd", &fogEnd, sizeof(fogEnd)); SET_UNIFORM("u_fogEquation", &fogEquation, sizeof(fogEquation)); }); Director::getInstance()->getEventDispatcher()->addEventListenerWithFixedPriority(_backToForegroundListener, -1); #endif } void FogTestDemo::switchTypeCallback(Ref* sender, int type) { if (type == 0) { auto fogColor = Vec4(0.5, 0.5, 0.5, 1.0); float fogStart = 10; float fogEnd = 60; int fogEquation = 0; SET_UNIFORM("u_fogColor", &fogColor, sizeof(fogColor)); SET_UNIFORM("u_fogStart", &fogStart, sizeof(fogStart)); SET_UNIFORM("u_fogEnd", &fogEnd, sizeof(fogEnd)); SET_UNIFORM("u_fogEquation", &fogEquation, sizeof(fogEquation)); } else if (type == 1) { auto fogColor = Vec4(0.5, 0.5, 0.5, 1.0); float fogDensity = 0.03f; int fogEquation = 1; SET_UNIFORM("u_fogColor", &fogColor, sizeof(fogColor)); SET_UNIFORM("u_fogDensity", &fogDensity, sizeof(fogDensity)); SET_UNIFORM("u_fogEquation", &fogEquation, sizeof(fogEquation)); } else if (type == 2) { auto fogColor = Vec4(0.5, 0.5, 0.5, 1.0); float fogDensity = 0.03f; int fogEquation = 2; SET_UNIFORM("u_fogColor", &fogColor, sizeof(fogColor)); SET_UNIFORM("u_fogDensity", &fogDensity, sizeof(fogDensity)); SET_UNIFORM("u_fogEquation", &fogEquation, sizeof(fogEquation)); } } void FogTestDemo::onExit() { CameraBaseTest::onExit(); Director::getInstance()->setClearColor(Color4F(0, 0, 0, 1)); if (_camera) { _camera = nullptr; } #if (CC_TARGET_PLATFORM == CC_PLATFORM_ANDROID) Director::getInstance()->getEventDispatcher()->removeEventListener(_backToForegroundListener); #endif } void FogTestDemo::update(float dt) {} void FogTestDemo::onTouchesMoved(const std::vector& touches, cocos2d::Event* event) { if (touches.size() == 1) { Vec2 prelocation = touches[0]->getPreviousLocationInView(); Vec2 location = touches[0]->getLocationInView(); Vec2 newPos = prelocation - location; if (_cameraType == CameraType::Free) { 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.1f; cameraPos += cameraRightDir * newPos.x * 0.1f; _camera->setPosition3D(cameraPos); } } } // CameraFrameBufferTest::CameraFrameBufferTest() //{ // // } // // CameraFrameBufferTest::~CameraFrameBufferTest() //{ // // } // // std::string CameraFrameBufferTest::title() const //{ // return "Camera FrameBuffer Object Test"; // } // // void CameraFrameBufferTest::onEnter() //{ // auto sizeInpixels = Director::getInstance()->getWinSizeInPixels(); // auto size = Director::getInstance()->getWinSize(); // auto fboSize = Size(sizeInpixels.width * 1, sizeInpixels.height * 1.5); // auto fbo = experimental::FrameBuffer::create(1, fboSize.width, fboSize.height); // // CameraBaseTest::onEnter(); // //auto sprite = Sprite::createWithTexture(fbo); // //sprite->setPosition(Vec2(100,100)); // //std::string filename = "MeshRendererTest/girl.c3b"; // //auto sprite = MeshRenderer::create(filename); // //sprite->setScale(1.0); // //auto animation = Animation3D::create(filename); // //if (animation) // //{ // // auto animate = Animate3D::create(animation); // // // sprite->runAction(RepeatForever::create(animate)); // //} // //sprite->setPosition(Vec2(100,100)); // auto rt = experimental::RenderTarget::create(fboSize.width, fboSize.height); // auto rtDS = experimental::RenderTargetDepthStencil::create(fboSize.width, fboSize.height); // fbo->attachRenderTarget(rt); // fbo->attachDepthStencilTarget(rtDS); // auto sprite = Sprite::createWithTexture(fbo->getRenderTarget()->getTexture()); // sprite->setScale(0.3f); // sprite->runAction(RepeatForever::create(RotateBy::create(1, 90))); // sprite->setPosition(size.width/2, size.height/2); // addChild(sprite); // // auto sprite2 = Sprite::create(s_pathGrossini); // sprite2->setPosition(Vec2(size.width/5,size.height/5)); // addChild(sprite2); // sprite2->setCameraMask((unsigned short)CameraFlag::USER1); // auto move = MoveBy::create(1.0, Vec2(100,100)); // sprite2->runAction( // RepeatForever::create( // Sequence::createWithTwoActions( // move, move->reverse()) // ) // ); // // auto camera = Camera::create(); // camera->setCameraFlag(CameraFlag::USER1); // camera->setDepth(-1); // camera->setFrameBufferObject(fbo); // fbo->setClearColor(Color4F(1,1,1,1)); // addChild(camera); // } BackgroundColorBrushTest::BackgroundColorBrushTest() {} BackgroundColorBrushTest::~BackgroundColorBrushTest() {} std::string BackgroundColorBrushTest::title() const { return "CameraBackgroundColorBrush Test"; } std::string BackgroundColorBrushTest::subtitle() const { return "right side object colored by CameraBG"; } void BackgroundColorBrushTest::onEnter() { CameraBaseTest::onEnter(); auto s = Director::getInstance()->getWinSize(); { // 1st Camera auto camera = Camera::createPerspective(60.0f, (float)s.width / s.height, 1.0f, 1000.0f); camera->setPosition3D(Vec3(0.0f, 0.0f, 200.0f)); camera->lookAt(Vec3::ZERO); camera->setDepth(-2); camera->setCameraFlag(CameraFlag::USER1); addChild(camera); // 3D model auto model = MeshRenderer::create("MeshRendererTest/boss1.obj"); model->setScale(4); model->setPosition3D(Vec3(20.0f, 0.0f, 0.0f)); model->setTexture("MeshRendererTest/boss.png"); model->setCameraMask(static_cast(CameraFlag::USER1)); addChild(model); model->runAction(RepeatForever::create(RotateBy::create(1.f, Vec3(10.0f, 20.0f, 30.0f)))); } { auto base = Node::create(); base->setContentSize(s); base->setCameraMask(static_cast(CameraFlag::USER2)); addChild(base); // 2nd Camera auto camera = Camera::createPerspective(60, (float)s.width / s.height, 1, 1000); auto colorBrush = CameraBackgroundBrush::createColorBrush(Color4F(.1f, .1f, 1.f, .5f), 1.f); camera->setBackgroundBrush(colorBrush); camera->setPosition3D(Vec3(0.0f, 0.0f, 200.0f)); camera->lookAt(Vec3::ZERO); camera->setDepth(-1); camera->setCameraFlag(CameraFlag::USER2); base->addChild(camera); // for alpha setting auto slider = ui::Slider::create(); slider->loadBarTexture("cocosui/sliderTrack.png"); slider->loadSlidBallTextures("cocosui/sliderThumb.png", "cocosui/sliderThumb.png", ""); slider->loadProgressBarTexture("cocosui/sliderProgress.png"); slider->setPosition(Vec2(s.width / 2, s.height / 4)); slider->setPercent(50); slider->addEventListener([slider, colorBrush](Ref*, ui::Slider::EventType) { colorBrush->setColor(Color4F(.1f, .1f, 1.f, (float)slider->getPercent() / 100.f)); }); addChild(slider); // 3D model for 2nd camera auto model = MeshRenderer::create("MeshRendererTest/boss1.obj"); model->setScale(4); model->setPosition3D(Vec3(-20.0f, 0.0f, 0.0f)); model->setTexture("MeshRendererTest/boss.png"); model->setCameraMask(static_cast(CameraFlag::USER2)); base->addChild(model); model->runAction(RepeatForever::create(RotateBy::create(1.f, Vec3(10.0f, 20.0f, 30.0f)))); } }