/** Example 018 Splitscreen, U( a# q, @( S' f1 ]( p作者:Max Winkel." `9 ?( W; K' V1 x8 \译:小时候可靓了(履霜坚冰)2 m! C! N# J2 i6 a2 Z! Q0 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+ Zusing namespace scene;, z+ I& c, e- O- n# `4 n z/*& N' H5 g8 W) U现在我们将定义用于初始化设备分辨率常量,另外我们设置一个全局变量用于决定分屏是否激活*/6 ~% ~+ ]& M. k! o F) Cconst 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 dint main()5 l& p* P3 m+ i+ s* D{ video::E_DRIVER_TYPE driverType;, j! ]2 a8 ?# y; q1 _% 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& O8 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
转载于:https://www.cnblogs.com/qilinzi/archive/2010/04/26/1940506.html
相关资源:各显卡算力对照表!