• ogre3D学习基础13 -- 键盘控制网格动画mesh


    以上一节为蓝本,这里增加一点难度,添加了四个节点,增加键盘控制移动速度,使用bool变量控制是否移动。

      第一,要增加键盘控制,那就使用OIS::KeyListener,在监听器里添加一个父类KeyListener,添加成员变量并初始化

      其中构造函数的ExampleFrameListener(win,cam,true,false);第三个参数改为true,表示使用带缓冲的键盘输入。

     1     MoveDemoListener(RenderWindow *win,Camera *cam,SceneNode *sn,Entity *ent,std::deque <Vector3> &walk)
     2         :ExampleFrameListener(win,cam,true,false),mNode(sn),mEntity(ent),mWalkList(walk)
     3     {    
     4         mWalkSpeed = 0.0f;//行走 速度设为每秒 35 个单位
     5         mDirection = Vector3::ZERO;//可用来判断机器人是否正在行走
     6         mMove = 10;
     7         mContinue = true;
     8         mKeyboard->setEventCallback(this);
     9         mWalking = false;
    10     }    
    11 
    12    bool mWalking;//是否在移动
    13     Real mMove;// + - 移动常量
    14     bool mContinue;//是否继续渲染    

      第二,添加键盘相应事件,很简单,这两个都是虚函数,必须实现,在构造函数里添加 mKeyboard->setEventCallback(this);相应键盘事件

     1 bool keyPressed(const OIS::KeyEvent &e)//
     2     {
     3         switch(e.key)
     4         {
     5         case OIS::KC_ESCAPE:
     6             mContinue = false;
     7             break;
     8         case OIS::KC_ADD://移动速度+10
     9             mWalkSpeed += mMove;
    10             break;
    11         case  OIS::KC_MINUS://移动速度-10
    12             mWalkSpeed -= mMove;
    13             break;
    14         default:
    15             break;
    16         }
    17         return true;
    18     }
    19     bool keyReleased(const OIS::KeyEvent &e)//
    20     {
    21         return true;
    22     }

      第三,是否移动的结果有mWalklist是否为空决定,不空为ture,空为false,代码操作如下:

     1     bool nextLocation()
     2     {
     3         if (mWalkList.empty())
     4         {
     5             mWalking = false;//不动
     6             return false;
     7         }
     8         mDestination = mWalkList.front(); //获取第一个元素
     9         mWalkList.pop_front(); //弹出一个元素 
    10         mDirection = mDestination - mNode->getPosition();//计算距离
    11         mDistance = mDirection.normalise();//转换为单位向量
    12         mWalking = true;//开始移动
    13         return true;
    14     }

      第四,在frameStarted()里捕获键盘,还有一些其他的更改,关键操作啊,看仔细了

     1     bool frameStarted(const FrameEvent &evt)
     2     {    
     3         mKeyboard->capture();    
     4 
     5 //      if (mDirection == Vector3::ZERO)
     6 //      {
     7         if (!mWalking)//第一次判断是否在走动,如果没有walk,就激活
     8         {
     9             if (nextLocation())//如果行走列表不为空
    10             {
    11                 // 设置行走的动画            
    12                 mAnimationState = mEntity->getAnimationState("Walk");//设置动画为走动
    13                 mAnimationState->setLoop(true);//循环执行
    14                 mAnimationState->setEnabled(true);//激活
    15             }
    16         }
    17 //      }
    18         else if (mWalking)//如果正在移动,接着判断
    19         //开始 移动
    20         {    
    21             Real move = mWalkSpeed * evt.timeSinceLastFrame;//速度*时间 = 移动距离
    22             mDistance -= move;//更新距离
    23             if (mDistance <= 0.0f)//小于0,说明已经走过目标地点.
    24             {
    25                 mNode->setPosition(mDestination);//重新设置为目的地
    26                 //mDirection = Vector3::ZERO;//方向为0,不走            
    27                 if (! nextLocation())//如果行走列表为空,没有目的地,结束
    28                 {                    
    29                     mAnimationState = mEntity->getAnimationState("Die");//死亡动画
    30                     mAnimationState->setLoop(false);//设置循环
    31                     mAnimationState->setEnabled(true);//激活
    32                 }
    33                 else    //继续走
    34                 {            
    35                     Vector3 src = mNode->getOrientation() * Vector3::UNIT_X;//获取实体的当前朝向,这里有向量的乘积,如果俩个向量方向相反,乘积就是-1
    36                     if ((1.0f + src.dotProduct(mDirection)) < 0.0001f)//如果要旋转180度
    37                     {
    38                         mNode->yaw(Degree(180));
    39                     }
    40                     else//如果不是要旋转180度
    41                     {
    42                         Ogre::Quaternion quat = src.getRotationTo(mDirection);//获取方向
    43                         mNode->rotate(quat,Node::TS_LOCAL);//旋转
    44                     } // else
    45                 }
    46             }
    47             else
    48             {        
    49                 mNode->translate(mDirection * move);//移动一定距离
    50             } // else
    51         } // if
    52 
    53         mAnimationState->addTime(evt.timeSinceLastFrame);//更新动画状态
    54         //return ExampleFrameListener::frameStarted(evt);
    55         return mContinue;//这里接受的是键盘传过来的数据,为false时,退出程序
    56     }

      第五,就是创建场景,这里增加了一个节点,组成一个正方形,,从一角出发,走一圈回来,然后就挂了(Die)--死亡动画。

     1     void createScene(void)
     2     {
     3         mSceneMgr->setAmbientLight( ColourValue( 1.0f, 1.0f, 1.0f ) );
     4         mEntity = mSceneMgr->createEntity( "robot", "robot.mesh" );
     5         mNode = mSceneMgr->getRootSceneNode( )->createChildSceneNode( "NinjaNode", Vector3( 0.0f, 0.0f, 0.0f ) );
     6         mNode->attachObject( mEntity );
     7 
     8         //添加行走节点,这是一个正方形
     9         mWalkList.push_back( Vector3( 500.0f, 0.0f, 0.0f ));
    10         mWalkList.push_back( Vector3(500.0f, 0.0f,-500.0f ));
    11         mWalkList.push_back( Vector3(0.0f, 0.0f,-500.0f ));
    12         mWalkList.push_back( Vector3(0,0,0));
    13 
    14         Entity *ent;
    15         SceneNode *node;
    16         ent = mSceneMgr->createEntity( "Knot1", "ogrehead.mesh" );
    17         node = mSceneMgr->getRootSceneNode( )->createChildSceneNode( "Knot1Node",Vector3( 0.0f, 0.0f, 0.0f ) );
    18         node->attachObject( ent );
    19         node->setScale( 0.51f, 0.51f, 0.51f );//缩小
    20 
    21         ent = mSceneMgr->createEntity( "Knot2", "ogrehead.mesh" );
    22         node = mSceneMgr->getRootSceneNode( )->createChildSceneNode( "Knot2Node",Vector3( 500.0f, 0.0f, 0.0f ) );
    23         node->attachObject( ent );
    24         node->setScale( 0.51f, 0.51f, 0.51f );//缩小
    25 
    26         ent = mSceneMgr->createEntity( "Knot3", "ogrehead.mesh" );
    27         node = mSceneMgr->getRootSceneNode( )->createChildSceneNode( "Knot3Node",Vector3(500.0f, 0.0f,-500.0f ) );
    28         node->attachObject( ent );
    29         node->setScale( 0.51f, 0.51f, 0.51f );//缩小
    30 
    31         ent = mSceneMgr->createEntity( "Knot4", "ogrehead.mesh" );
    32         node = mSceneMgr->getRootSceneNode( )->createChildSceneNode( "Knot4Node",Vector3(0.0f,0.0f,-500.0f ) );
    33         node->attachObject( ent );
    34         node->setScale( 0.51f, 0.51f, 0.51f );//缩小
    35 
    36         mCamera->setPosition(150.0f, 450.0f,450.0f );
    37         mCamera->pitch( Degree(-45.0f) );
    38         mCamera->yaw( Degree(-15.0f) );
    39     
    40     }

      好了,看截图了

      

       简单就这样了,好好学习,天天向上。

    2013-9-22

    升级版:

       这里再进一步学习,使用list(循环链表)可以使机器人不停的移动

      更改如下,将所有的deque替换成list

      然后在类MoveDemoListener中添加一行代码,如下:

     1 bool nextLocation()
     2     {
     3         if (mWalkList.empty())
     4         {
     5             mWalking = false;//不动
     6             return false;
     7         }
     8         mDestination = mWalkList.front(); // 
     9         mWalkList.pop_front(); // 
    10         mWalkList.push_back(mDestination);//这一行是新增加的,每次取出一个点后,就将其添加到链表末尾
    11         mDirection = mDestination - mNode->getPosition();
    12         mDistance = mDirection.normalise();
    13         mWalking = true;//开始移动
    14         return true;
    15     }

      这样,robot就可以绕着方块一直走了。其他的代码保持原样。

  • 相关阅读:
    异步加载JS
    解决Vue刷新一瞬间出现样式未加载完或者出现Vue代码问题
    什么是NaN?它的类型是什么?如何可靠的测试一个值是否等于NaN?
    JS快速获取图片宽高的方法
    为什么操作dom会消耗性能
    localstorage的浏览器支持情况
    mongodb Windows系统下安装卡死问题
    正则表达式
    CSS Hack
    css中cursor(光标类型)
  • 原文地址:https://www.cnblogs.com/songliquan/p/3319835.html
Copyright © 2020-2023  润新知