/** Example 018 Splitscreen, U( a# q, @( S' f1 ]( p
作者:Max Winkel." `9 ?( W; K' V1 x8 \
译:小时候可靓了(履霜坚冰)2 m! C! N# J2 i6 a2 Z! Q
0 U2 D4 U: U% J1 E5 N! a0 L4 q& x
这个例程中我们将学习怎么使用irrlicht中的分屏(比如在赛车类游戏中)& ~$ M; ]8 f. _4 {
我们将创建一个被分为4个部分的视口,有3个固定摄相机和一个用户可以控制的摄相机4 N% L! s9 [( l# n
好,让们从头文件开始吧(我想没有再多说的必要了)3 C9 |* C8 P! X
*/
#include <irrlicht.h>
#include <iostream>

#ifdef _MSC_VER
#pragma comment(lib, "Irrlicht.lib")
#endif


) h- x" R- L6 B4 b6 L
//引擎命名空间! [" B6 k9 Y2 J1 @/ K& {
using namespace irr;
using namespace core;
using namespace video;6 n$ L6 u6 X  ?: {0 M+ Z
using namespace scene;, z+ I& c, e- O- n# `4 n  z

/*& N' H5 g8 W) U
现在我们将定义用于初始化设备分辨率常量,另外我们设置一个全局变量用于决定分屏是否激活
*/6 ~% ~+ ]& M. k! o  F) C
const int ResX=800;
const int ResY=600;3 U- b& o1 j$ c1 G- D! C$ B& ~
const bool fullScreen=false;
@6 A
//是否使用分屏
bool SplitScreen=true;# p: x* k' {* s
% A% x/ d* j5 k2 j* ~& G7 ?8 y
/*
现在我们需要4个点来设定即将要创建摄相机的位置。( s" N  n( |; y
*/
//摄相机
ICameraSceneNode *camera[4]={0,0,0,0};4 L/ N7 C" \1 V. Z& u5 i# r  L


/*
在我们的事件接收器中,当用户按下S键的时候,我们切换分屏变量,所有的其它事件则交给FPS摄相机。1 j3 Y/ v. l' T- e; L+ o! |, @
*/
class MyEventReceiver : public IEventReceiver
{6 K/ S* |" k8 x( ~% D2 M
        public:- n7 V0 Y8 b! x  @8 m6 _
                virtual bool OnEvent(const SEvent& event)- w$ y# T. E8 T3 Q( {  k& H, W
                {! I" S% Z$ C( o, R8 \7 a
                        
   //S键   启用/禁用分屏
   if (event.EventType == irr::EET_KEY_INPUT_EVENT &&
  event.KeyInput.Key == KEY_KEY_S && event.KeyInput.PressedDown)6 I- U6 C) b+ g- W9 X5 J
                        {
                          SplitScreen = !SplitScreen;
                               return true;) {8 R7 A+ e. t4 W: B* [% _4 ?  O
                        }                    5 X) j" _: i) a& h# [% L7 j
                        //发送其它所有的事件给第4个摄相机! F; ?# r8 W; f! F
                        if (camera[3])# p& Q7 j  h+ z+ Y% D2 N' c
                                return camera[3]->OnEvent(event);
                  return false;
            }2 z6 G- w+ `$ ?3 _& z( H4 y
};


F8 H
/*
好,现在看Main函数
首先,我们初始化设备,取得资源管理器和图形驱动。从md2中加载一个动画模型,; K6 F* ^9 F% a7 K5 `  z5 [
从pk3中加载一个地图。因为这是先前使用过的东西,我就不逐步解释了。
只关心地图的位置
*// u0 X6 N; b) L8 ?& T8 q  d
int main()5 l& p* P3 m+ i+ s* D
{
    video::E_DRIVER_TYPE driverType;, j! ]2 a8 ?# y; q
1 _% O( A/ n- F
        printf("Please select the driver you want for this example:\n"\6 T  a* Y. ~& B! P+ w8 x  X
                " (a) Direct3D 9.0c\n (b) Direct3D 8.1\n (c) OpenGL 1.5\n"\
           " (d) Software Renderer\n (e) Burning's Software Renderer\n"\
         " (f) NullDevice\n (otherKey) exit\n\n");% m5 @2 @) }! [& a! e

        char i;+ w- b/ X6 l, q/ J
        std::cin >> i;$ P' `* _; @$ y% h$ G  M  {
  h: H0 y/ f' _! Y/ L8 T
        switch(i)2 Q8 ^& t5 h) x, {, e
        {* U5 `: B1 y" h/ K* O1 a# Y) }
                case 'a': driverType = video::EDT_DIRECT3D9;break;
                case 'b': driverType = video::EDT_DIRECT3D8;break;4 Q, {' \5 B1 a9 d3 U0 l4 u* q
                case 'c': driverType = video::EDT_OPENGL;   break;9 D8 t9 P2 U; _/ S6 ~1 Z7 e
                case 'd': driverType = video::EDT_SOFTWARE; break;
               case 'e': driverType = video::EDT_BURNINGSVIDEO;break;" ^7 W# ^, u( p
                case 'f': driverType = video::EDT_NULL;     break;
            default: return 1;
         }
m0 I: K5 O" ]8 w
        

     //实例化事件接收器1 i, i# a# j/ Y! f3 ?
        MyEventReceiver receiver;$ y/ I7 ]* Z! R9 _8 x

     //初始化引擎
    IrrlichtDevice *device = createDevice(driverType,1 O- Y# K% x- s6 u
                        dimension2du(ResX,ResY), 32, fullScreen,
                      false, false, &receiver);
    if (!device)
          return 1;
- i- c
        ISceneManager *smgr = device->getSceneManager();
     IVideoDriver *driver = device->getVideoDriver();
     //加载模型
      IAnimatedMesh *model = smgr->getMesh("../../media/sydney.md2");
     if (!model)
              return 1;: }' k2 L5 J, c6 F7 O% ]8 i0 ]
        IAnimatedMeshSceneNode *model_node = smgr->addAnimatedMeshSceneNode(model);
       //加载纹理
     if (model_node): H0 m7 `2 H" V
        {
            ITexture *texture = driver->getTexture("../../media/sydney.bmp");2 _2 O% I7 [$ R  k  V3 P2 w1 f
                model_node->setMaterialTexture(0,texture);' G" R+ T6 c" o1 m+ i! w+ v
                model_node->setMD2Animation(scene::EMAT_RUN);
            //禁用光照. W/ B- {; T$ @
                model_node->setMaterialFlag(EMF_LIGHTING,false);
       }
+ A  I5 H
        //加载地图/ n. Y1 n. e  f& e+ F
        device->getFileSystem()->addZipFileArchive("../../media/map-20kdm2.pk3");# R* @8 U1 U" t' ]' r$ `
        IAnimatedMesh *map = smgr->getMesh("20kdm2.bsp");; N8 o* Q  l# `$ s/ [
        if (map)
      {
            ISceneNode *map_node = smgr->addOctTreeSceneNode(map->getMesh(0));
             //设置位置
            map_node->setPosition(vector3df(-850,-220,-850));
       }+ o7 A! v% `, w% |9 S5 t, i- C& O
8 a; x/ ?7 u1 ~/ s& D+ d
/*  D: g' p+ ?2 O4 A) Y9 y
现在我们建仓我们的4个摄相机,一个用于从前面看模型,
一个用于从上往下看模型,另一个从侧面看。! l* S: q3 c- }0 `" f! `* O* T+ O
另外,我们还创建一个FPS摄相机,用来让用户操作# a. B1 p  @2 V5 V
*/7 ]* h# T% z" {- ?
        //创建3个固定的摄相机
      //前% T' P) p7 n! E( z6 O5 v3 V" Q
        camera[0] = smgr->addCameraSceneNode(0, vector3df(50,0,0), vector3df(0,0,0));
       //上
       camera[1] = smgr->addCameraSceneNode(0, vector3df(0,50,0), vector3df(0,0,0));
       //左0 p7 W# @9 D% _+ z9 g
        camera[2] = smgr->addCameraSceneNode(0, vector3df(0,0,50), vector3df(0,0,0));8 \% N* b4 a* Z$ ~# b! b
        //用户控制的
        camera[3] = smgr->addCameraSceneNodeFPS();$ M) P* {* q7 ?/ `$ M0 \' M
        //不要在sydney的位置开始' {* T" B9 d8 _) `
        if (camera[3])
               camera[3]->setPosition(core::vector3df(-50,0,-50));( d' z% ]# M$ `) T8 n: w+ \
! Z- G" d! p  u, ?0 Q) X
/*7 {* g9 ]6 F% `/ n
//创建用于记录帧数的变量和隐藏鼠标4 |9 _. ?. |4 G  Y" h
*/% O, d( U+ j! w7 G0 i' F
        //隐藏鼠标
        device->getCursorControl()->setVisible(false);
       //帧数记录
        int lastFPS = -1;
/*! R2 c4 u6 B2 [! @
到现在还没出现什么新的内容
仅是定义我们的4个摄相机,那游戏画面不会出现分屏效果。
为了实现这个目标,你需要以下几步; r' S: p: j4 i, a/ A6 w
       设置整个屏幕的视口
         开始一个新的场景(清除屏幕)
         以下3步是分屏中的每个视口都要重复的7 r* N# Y( k* _& h7 q% b. J. F
              设置视口到你想看的区域
               激活摄相机,使它和视口“连接”& P8 i8 j; D; l( I/ P
                  渲染所有物体2 Z5 [3 X( `9 D6 C

   如果你有一个GUI. R# h+ N$ U# ]0 m* y8 m
      设置视口到整个屏幕5 F7 |4 G$ h2 C; `% }
          显示GUI& ]6 J9 N8 Z- d9 E6 V
  结束场景
 z" E0 v+ R5 g9 q8 k
听起来好像有点复杂,但你看了就明白了
*/
       while(device->run())
       {
               //设置整个场景的视口并开始绘制
            driver->setViewPort(rect<s32>(0,0,ResX,ResY));
              driver->beginScene(true,true,SColor(255,100,100,100));& J- i$ o; T. w: m' \# [
                //如果分屏被启动
            if (SplitScreen) //还记得这个变量吗?前面定义过的; K/ y2 c& y' C( }/ Y, U9 }
                {) h7 P* c( C2 }) y
                        //Activate camera1+ w: V3 }3 Q* ?! `: v
                        //激活1号摄相机2 e8 V' M' f8 k6 l
                        smgr->setActiveCamera(camera[0]);
                     //Set viewpoint to the first quarter (left top)

                       //设置视口到左上角
                    driver->setViewPort(rect<s32>(0,0,ResX/2,ResY/2));) K: ?. W# x+ ~9 i! ^
                        //Draw scene
                   //绘制场景
                      smgr->drawAll();2 l% L, Z% V1 b' W+ x
                        //Activate camera22 V  \; o: @7 M: g* X
                        //激活2号摄相机
                  smgr->setActiveCamera(camera[1]);
                     //Set viewpoint to the second quarter (right top)
                     //设置视口到右上角% g' N1 d! x3 N$ f7 o+ C- }
                        driver->setViewPort(rect<s32>(ResX/2,0,ResX,ResY/2));
                     //Draw scene
                    //绘制场景5 x7 L: `6 k& E; x, ?' k" u
                        smgr->drawAll();
                      //Activate camera3
                      //激活3号摄相机' x% S( E( B$ f5 H# Y4 o& l
                        smgr->setActiveCamera(camera[2]);! c# c0 u& v% O  I1 ^& F
                        //Set viewpoint to the third quarter (left bottom)' O3 M, t5 u. O3 ^8 g" O8 G% z
                        //设置视口到左下角
                      driver->setViewPort(rect<s32>(0,ResY/2,ResX/2,ResY));
                    //Draw scene/ q+ B3 [7 Y+ M
                        //绘制场景' l; ~$ w' b7 t. x
                        smgr->drawAll();6 y( k3 G% B1 M5 K
                        //Set viewport the last quarter (right bottom)
                      //设置视口到右下角6 W% h/ |" l' Q( Z: l
                        driver->setViewPort(rect<s32>(ResX/2,ResY/2,ResX,ResY));
               }
              //Activate camera46 ~3 I5 O7 o! A0 k, R
                //激活4号摄相机$ g: O+ R2 j% |' w. y5 g
                smgr->setActiveCamera(camera[3]);
              //Draw scene1 L( a4 x: I  C- A; q8 K9 R2 d
                //绘制场景: j. f0 H" B! Z2 J0 S+ Q
                smgr->drawAll();! G- \$ Y7 p% `: B# p# P
                driver->endScene();
/*3 A3 l  a+ C5 N/ q: r( [& T
                你大概已经看到了,这幅图片被每个视口分别渲染,那意味着你要损失许多性能(毕竟同样的场景绘了四次)
               好,如果你问;“我这样做有必要吗?”不要慌,它真的很简单:在区域函数中,我们定义了4个坐标: R  m7 T% B% \# t0 o
                (0,0)左上角坐标,(X,Y)右下角坐标* U; L4 T! u$ f; V2 k: p

              这意味着,如果你想要把屏幕分割成两部分,你需要给出如下的坐标
              1号视口: 0,0,ResX/2,ResY
             2号视口:ResX/2,0,ResX,ResY
 M5 V- ~3 n0 Q9 i
                如果你没有完全明白,只需要修改示例程序的值,看发生了什么。
G* l3 ?! M) q4 s
                现在我们只需要显示FPS和在用户要离开的时候销毁引擎即可. _; ~+ H' ~6 y, M: ]+ E
*/
               //取得并显示帧率8 x  W: e( ?7 |# g9 s7 T3 Z/ d
                if (driver->getFPS() != lastFPS)
            {' G7 _0 E) E: l" m+ `& |% \
                        lastFPS = driver->getFPS();* V- ~+ D& x  R3 }4 c
                        core::stringw tmp = L"Irrlicht SplitScreen-Example (FPS: ";6 G5 h9 x# a! G0 w, S# V
                        tmp += lastFPS;) Q- h" x# j0 g5 T
                        tmp += ")";
                      device->setWindowCaption(tmp.c_str());% S1 `3 f6 {% R0 `
                }
l' X' R
        //销毁设备# T  t% w$ o' ?# [
        device->drop();
       return 0;
}
/*( P: F4 J, T6 l$ c8 u1 P! H* i: ]
全部结束,只需要编译就可以玩了
 注意:用S键你可以控制是否分屏
*/" f9 |% Q# F- U9 E8 \7 Q1 F
发表于: irrlicht3d中文官方论坛(3D网游开发论坛) http://www.irrlicht3d.cn

作者:麒麟子