ogre3D学习基础12 --- 让机器人动起来(移动模型动画)
学了那么长时间,才学会跑起来。My Ogre,动起来。
第一,还是要把框架搭起来,这里我们用到双端队列deque,前面已经简单介绍过,头文件如下:
#include "ExampleApplication.h"
#include "deque"
using namespace std;
第二,是我们的监听器,继承至ExampleFrameListener,构造函数我们使用五个参数,分别为渲染窗口,摄像机,场景节点,实体,行走路线上的位置坐标。
MoveDemoListener(RenderWindow *win,Camera *cam,SceneNode *sn,Entity *ent,std::deque <Vector3> &walk) :ExampleFrameListener(win,cam,false,false), mNode(sn),mEntity(ent),mWalkList(walk)
{ // Set default values for variables
mWalkSpeed = 35.0f;//行走 速度设为每秒 35 个单位
mDirection = Vector3::ZERO;//可用来判断机器人是否正在行走,为零时不走 }
第三,我们接着完善监听器,首先成员函数添加了
Real mDistance;//实体要移动的距离
Vector3 mDirection;//方向
Vector3 mDestination;//目的地 AnimationState *mAnimationState;//物体的目前的动画状态 Entity *mEntity;
SceneNode *mNode;
std::deque<Vector3> mWalkList;//要行走的点
Real mWalkSpeed;//物体移动的速度
然后添加一个函数,bool nextLocation();用来判断是否有下一个方位,确定下一步怎么走。如果返回行走的队列为空,返回false
bool nextLocation()
{
if (mWalkList.empty())
return false;
mDestination = mWalkList.front(); // 取得队列的头部
mWalkList.pop_front(); // 删除已走过的点
mDirection = mDestination - mNode->getPosition();//距离由目的地减去当前位置得到
mDistance = mDirection.normalise();//normalise()作用是将方向向量转换成单位向量,返回向量的原始长度
return ture;
}
最后我们添加在每一帧前渲染的函数frameStarted(),这函数我们见了好多次,以后我们还会见面的。移动、转向操作等都在这里。
bool frameStarted(const FrameEvent &evt)
{
if (mDirection == Vector3::ZERO)
{
if (nextLocation())//如果行走列表不为空
{
// 设置行走的动画
mAnimationState = mEntity->getAnimationState("Walk");//设置动画为走动
mAnimationState->setLoop(true);//循环执行
mAnimationState->setEnabled(true);//激活
}
}
else//开始 移动
{
Real move = mWalkSpeed * evt.timeSinceLastFrame;//速度*时间 = 移动距离
mDistance -= move;//更新距离
if (mDistance <= 0.0f)//小于0,说明已经走过目标地点
{
mNode->setPosition(mDestination);//重新设置为目的地
mDirection = Vector3::ZERO;//方向为0,不走
if (! nextLocation())//如果行走列表为空,没有目的地
{// Set Idle animation
mAnimationState = mEntity->getAnimationState("Idle");//从网格动画的动画集(Idle)中获取当前的状态
mAnimationState->setLoop(true);//设置循环
mAnimationState->setEnabled(true);//激活
}
else //继续走
{
Vector3 src = mNode->getOrientation() * Vector3::UNIT_X;//获取实体的当前朝向,这里有向量的乘积,如果俩个向量方向相反,乘积就是-1
if ((1.0f + src.dotProduct(mDirection)) < 0.0001f)//如果要旋转180度
{
mNode->yaw(Degree());
}
else//如果不是要旋转180度
{
Ogre::Quaternion quat = src.getRotationTo(mDirection);//获取方向
mNode->rotate(quat);//旋转
} // else
}
}
else
{
mNode->translate(mDirection * move);//移动一定距离
} // else
} // if mAnimationState->addTime(evt.timeSinceLastFrame);//更新动画状态
return ExampleFrameListener::frameStarted(evt);
}
我已经把注释写的很详细了。这里说一下第29行Vector3 src = mNode->getOrientation() * Vector3::UNIT_X;这里牵涉到向量乘积的问题,两个向量乘积有以下两种情况
向量的积分为数量积和向量积 数量积就是向量的点乘 向量积就是向量的叉乘
设a(x,y,z) b(m,n,p)
则 a点乘b=xm+yn+zp
或 a点乘b=|a||b|*cos<a,b>
设a=xi+yj+zk b=mi+nj+pk
则叉乘 a×b=(yp-zn)i+(zm-xp)j+(xn-ym)k
Vector3 src = mNode->getOrientation() * Vector3::UNIT_X;//这个是点乘,结果是一个数,大小在-1,1之间。两个向量方向相反时,乘积为-1
还有个老熟人createFrameListener(),我就不解释了,不懂,看前一章。
void createFrameListener(void)
{
mFrameListener = new MoveDemoListener(mWindow,mCamera,mNode,mEntity,mWalkList);
mFrameListener->showDebugOverlay(true);
mRoot->addFrameListener(mFrameListener);
}
第四,创建场景,如下,都是一些简单的东西,不解释
void createScene(void)
{
mSceneMgr->setAmbientLight( ColourValue( 1.0f, 1.0f, 1.0f ) );
mEntity = mSceneMgr->createEntity( "Robot", "robot.mesh" );
mNode = mSceneMgr->getRootSceneNode( )->createChildSceneNode( "RobotNode", Vector3( 0.0f, 0.0f, 25.0f ) );
mNode->attachObject( mEntity ); // 创建走动列表
mWalkList.push_back( Vector3( 550.0f, 0.0f, 50.0f ) );
mWalkList.push_back( Vector3(-100.0f, 0.0f, -200.0f ) ); Entity *ent;
SceneNode *node;
ent = mSceneMgr->createEntity( "Knot1", "knot.mesh" );
node = mSceneMgr->getRootSceneNode( )->createChildSceneNode( "Knot1Node",Vector3( 0.0f, -10.0f, 25.0f ) );
node->attachObject( ent );
node->setScale( 0.1f, 0.1f, 0.1f );
ent = mSceneMgr->createEntity( "Knot2", "knot.mesh" );
node = mSceneMgr->getRootSceneNode( )->createChildSceneNode( "Knot2Node",Vector3( 550.0f, -10.0f, 50.0f ) );
node->attachObject( ent );
node->setScale( 0.1f, 0.1f, 0.1f );
ent = mSceneMgr->createEntity( "Knot3", "knot.mesh" );
node = mSceneMgr->getRootSceneNode( )->createChildSceneNode( "Knot3Node",Vector3(-100.0f, -10.0f,-200.0f ) );
node->attachObject( ent );
node->setScale( 0.1f, 0.1f, 0.1f ); mCamera->setPosition( 90.0f, 280.0f, 535.0f );
mCamera->pitch( Degree(-30.0f) );
mCamera->yaw( Degree(-15.0f) );
}
好了,完整代码如下
#include "ExampleApplication.h"
#include "deque"
using namespace std; class MoveDemoListener : public ExampleFrameListener
{
public:
MoveDemoListener(RenderWindow *win,Camera *cam,SceneNode *sn,Entity *ent,std::deque <Vector3> &walk)
:ExampleFrameListener(win,cam,false,false),mNode(sn),mEntity(ent),mWalkList(walk)
{
mWalkSpeed = 35.0f;//行走 速度设为每秒 35 个单位
mDirection = Vector3::ZERO;//可用来判断机器人是否正在行走
} bool nextLocation()
{
if (mWalkList.empty())
return false;
mDestination = mWalkList.front(); //
mWalkList.pop_front(); //
mDirection = mDestination - mNode->getPosition();
mDistance = mDirection.normalise();
return true;
}
bool frameStarted(const FrameEvent &evt)
{
if (mDirection == Vector3::ZERO)
{
if (nextLocation())//如果行走列表不为空
{
// 设置行走的动画
mAnimationState = mEntity->getAnimationState("Walk");//设置动画为走动
mAnimationState->setLoop(true);//循环执行
mAnimationState->setEnabled(true);//激活
}
}
else//开始 移动
{
Real move = mWalkSpeed * evt.timeSinceLastFrame;//速度*时间 = 移动距离
mDistance -= move;//更新距离
if (mDistance <= 0.0f)//小于0,说明已经走过目标地点
{
mNode->setPosition(mDestination);//重新设置为目的地
mDirection = Vector3::ZERO;//方向为0,不走
if (! nextLocation())//如果行走列表为空,没有目的地
{
mAnimationState = mEntity->getAnimationState("Idle");//从网格动画的动画集(Idle)中获取当前的状态
mAnimationState->setLoop(true);//设置循环
mAnimationState->setEnabled(true);//激活
}
else //继续走
{
Vector3 src = mNode->getOrientation() * Vector3::UNIT_X;//获取实体的当前朝向,这里有向量的乘积,如果俩个向量方向相反,乘积就是-1
if ((1.0f + src.dotProduct(mDirection)) < 0.0001f)//如果要旋转180度
{
mNode->yaw(Degree());
}
else//如果不是要旋转180度
{
Ogre::Quaternion quat = src.getRotationTo(mDirection);//获取方向
mNode->rotate(quat);//旋转
} // else
}
}
else
{
mNode->translate(mDirection * move);//移动一定距离
} // else
} // if mAnimationState->addTime(evt.timeSinceLastFrame);//更新动画状态
return ExampleFrameListener::frameStarted(evt);
}
protected:
Real mDistance;//实体要移动的距离
Vector3 mDirection;//方向
Vector3 mDestination;//目的地 AnimationState *mAnimationState;//物体的目前的动画状态 Entity *mEntity;
SceneNode *mNode;
std::deque<Vector3> mWalkList;//要行走的点
Real mWalkSpeed;//物体移动的速度 }; class MoveDemoApplication:public ExampleApplication
{
public:
MoveDemoApplication()
{ }
~MoveDemoApplication()
{ }
protected:
Entity *mEntity;
SceneNode *mNode;
std::deque<Vector3> mWalkList;
void createScene(void)
{
mSceneMgr->setAmbientLight( ColourValue( 1.0f, 1.0f, 1.0f ) );
mEntity = mSceneMgr->createEntity( "Robot", "robot.mesh" );
mNode = mSceneMgr->getRootSceneNode( )->createChildSceneNode( "RobotNode", Vector3( 0.0f, 0.0f, 25.0f ) );
mNode->attachObject( mEntity ); mWalkList.push_back( Vector3( 550.0f, 0.0f, 50.0f ) );
mWalkList.push_back( Vector3(-100.0f, 0.0f, -200.0f ) ); Entity *ent;
SceneNode *node;
ent = mSceneMgr->createEntity( "Knot1", "knot.mesh" );
node = mSceneMgr->getRootSceneNode( )->createChildSceneNode( "Knot1Node",Vector3( 0.0f, -10.0f, 25.0f ) );
node->attachObject( ent );
node->setScale( 0.1f, 0.1f, 0.1f );
ent = mSceneMgr->createEntity( "Knot2", "knot.mesh" );
node = mSceneMgr->getRootSceneNode( )->createChildSceneNode( "Knot2Node",Vector3( 550.0f, -10.0f, 50.0f ) );
node->attachObject( ent );
node->setScale( 0.1f, 0.1f, 0.1f );
ent = mSceneMgr->createEntity( "Knot3", "knot.mesh" );
node = mSceneMgr->getRootSceneNode( )->createChildSceneNode( "Knot3Node",Vector3(-100.0f, -10.0f,-200.0f ) );
node->attachObject( ent );
node->setScale( 0.1f, 0.1f, 0.1f ); mCamera->setPosition( 90.0f, 280.0f, 535.0f );
mCamera->pitch( Degree(-30.0f) );
mCamera->yaw( Degree(-15.0f) );
} void createFrameListener(void)
{
mFrameListener = new MoveDemoListener(mWindow,mCamera,mNode,mEntity,mWalkList);
mFrameListener->showDebugOverlay(true);
mRoot->addFrameListener(mFrameListener);
} }; #include "windows.h" INT WINAPI WinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance, LPSTR lpCmdLine, INT)
{
MoveDemoApplication app;
app.go();
return ;
}
ogre3D学习基础12 --- 让机器人动起来(移动模型动画)的更多相关文章
- ogre3D学习基础1 -- 核心对象与脚本技术
一.核心对象介绍1.命名空间 Ogre3d使用了C++的特性--命名空间,可以防止命名混淆.使用方法也简单,using namespace Ogre;或者直接在使用时加上“Ogre::”的前缀,如Og ...
- ogre3D学习基础13 -- 键盘控制网格动画mesh
以上一节为蓝本,这里增加一点难度,添加了四个节点,增加键盘控制移动速度,使用bool变量控制是否移动. 第一,要增加键盘控制,那就使用OIS::KeyListener,在监听器里添加一个父类KeyLi ...
- ogre3D学习基础5 -- 阴影与动画
五.阴影 阴影是渲染一个真实场景的重要组成部分,它可以给场景中的物体提供更加真实的感觉,同时还可以帮助用户更好的了解对象间的空间关系. 启用阴影: 缺省情况下,阴影是关闭的,开启方式如下: 1.建立场 ...
- ogre3D学习基础8 --- 资源管理器
资源管理 可管理的资源有: 材质资源:在.material文件中包含的材质脚本定义(技术.通路.纹理单元等数据的定义). 模型资源:经过优化的二进制网格模型文件,扩展名为.mesh.包含几何信息和一些 ...
- ogre3D学习基础19 --- 材质的继承,纹理的滚动与旋转
以上一节为基础,废话不多说. 首先新增一个节点,用于比较显示 //新增一个节点 ent = mSceneMgr->createEntity("Quad"); ent-> ...
- ogre3D学习基础18 -- 材质的使用与脚本的简单书写
这一节以基础16为基础,练习材质的使用. 第一,看看框架 //material #include "ExampleApplication.h" class TutorialAppl ...
- ogre3D学习基础11 -- 交换两个场景管理器
这一节,练习一下前几次学习的内容,功能很简单,就是建立两个不同的场景管理器,当按下键盘上某个键时切换镜头. 基本框架不变,这个监听器继承了两个父类,一个是我们的老朋友ExampleFrameListe ...
- ogre3D学习基础9 -- 光源程序实例
这一章练习一下光源的使用,光源分为三种:点光源,聚光源,有向光.具体内容前面说过,这里就不解释了. 继续在上一章的程序的基础上实现. 1.创建摄像机(Camera) createCamera()函数是 ...
- ogre3D学习基础7---材质详解
物体着色的基础 --- 四种不同光照作用 1.环境反射 近似的模拟了场景中的全局辐射,也就是用来近似模拟所有光在场景中不断散射的结果.材质中有相应的属性来代表这种环境反射颜色. 2.漫反射 这种颜色是 ...
随机推荐
- Android使用文件管理器打开指定文件夹,浏览里面的内容
Android下可以打开一些文件,带有.doc 等后缀的文件网上一般都有解释,这个写一个使用文件管理器打开指定文件夹的 private void openAssignFolder(String pat ...
- Python学习-用户输入和字符串拼接
用户输入和字符串拼接 #用户输入和字符串拼接username=input("username:")age=int(input("Age:")) #转换整数型 ...
- HDU1043 八数码(BFS + 打表)
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1043 , 康托展开 + BFS + 打表. 经典八数码问题,传说此题不做人生不完整,关于八数码的八境界 ...
- pat甲级1013
1013 Battle Over Cities (25)(25 分) It is vitally important to have all the cities connected by highw ...
- Python http
# import httplib # http_client = None # http_client = httplib.HTTPConnection('localhost', 8080, time ...
- java生成excel文档
要做一个后台自动化,要先预先生成一份文档,以下内容生成了文档 首先下载jxl.jar包,下载地址:http://download.csdn.net/detail/prstaxy/4469935 1.生 ...
- 5-15 笔记 jtopo使用
Jtopo的核心对象有6个,分别是Stage(舞台对象),Scene(场景对象),Node(节点对象),Link(连线对象),Container(容器对象),Effect.Animate(动画效果) ...
- Web开发中,用到的4种会话跟踪技术
会话跟踪:主要解决HTTP的无状态问题,即: 当用户发出请求时,服务器就会做出响应,客户端与服务器之间的联系是离散的.非连续的.当用户在同一网站的多个页面之间转换时,根本无法确定是否是同一个客户,会话 ...
- github不能加载css、js解决办法
很奇怪,上午在公司还能正常访问github,晚点访问却有问题,页面样式明显错乱. 在FireFox下用F12开发者工具一看,有2条css和2条js 404 了,猜想应该是github的DNS被GFW污 ...
- 阿里云服务器下安装LAMP环境(CentOS Linux 6.3)(1)
阿里的云服务器准备好以后,我们首先要做的就是把自己购买的磁盘空间挂载到系统里面,我们为服务器选择的是 Linux 系统,确切说的是 CentOS 系统. 默认阿里云服务器带了一个 20G 的空间,一般 ...