现在的位置: 首页 > 综合 > 正文

全文:在游戏中使用CEGUI —— 第一章(底层)

2012年10月20日 ⁄ 综合 ⁄ 共 8065字 ⁄ 字号 评论关闭

本文首次刊登于《游戏创造》,现开放与大家共享,转载请注明出处。

 

作者介绍

       唐亮(千里马肝),四年游戏从业经验,曾任职于大宇软星科技(上海)有限公司任程序技术指导,现在ATIEngineer,主要负责XP/Vista下的Display Driver。迄今为止主要个人作品为《阿猫阿狗2》,参与开发《汉朝与罗马》、《阿猫阿狗大作战OLG》和《仙剑奇侠传4》,主要研究方向为C++、图形渲染技术和系统架构。

blog地址:http://oiramario.cnblogs.com/

 

 

简介

       CEGUICrazy Eddie’s GUI http://www.cegui.org.uk/)是一个自由免费的GUI库,基于LGPL协议,使用C++实现,完全面向对象设计。CEGUI开发者的目的是希望能够让游戏开发人员从繁琐的GUI实现细节中抽身出来,以便有更多的开发时间可以放在游戏性上。

CEGUI的渲染需要3D图形API的支持,如OpenGLDirect3D。另外,使用更高级的图形库也是可以的,像是OGREIrrlichtRenderWare,关键需求可以简化为二点:

1.         纹理(Texture)的支持

2.         直接写屏(RHW的顶点格式、正交投影、或者使用shader实现)

本文截止日时,CEGUI的最新版本是0.4.1(本文的讨论也是基于此版本),提供了SDK和全部源码的下载,同时为了适应不同的使用需求,还根据STL的使用区分为NativeVC自带的P.J. STL)和STLport(基于SGI STL实现的跨编译器版本,详细见http://www.stlport.org/),以及VC6.0VC7.0VC7.1VC8.0几种。

除此之外,CEGUI还同步提供了官方界面编辑器LayoutEditor,以方便UI的制作,下载地址:http://www.2dgame-tutorial.com/downloads/CELayoutEditorSetup_0.4.1.exe。作为界面编辑器,它需要系统级界面以提供编辑器操作,在此之前的0.3.0版是基于MFC实现的;而在0.4.1版本中,改为基于wxWidgets(跨平台的本地UI框架,这里的UIWindow操作系统底层,如:WindowsUnixMac,详见http://www.wxwidgets.org/)实现。

OGRE作为目前最活跃的开源3D引擎,许多公司开始使用它进行游戏开发,原因也是其功能非常得全面和强大。在最初,OGRE曾经实现过一版UI,但是最后却放弃自己的实现而选择了CEGUI

 

 

 

Why

很多人可能会觉得UI这种东西很简单,自己写就好了。我想这首先要看标准是什么了,如果只是简单的按钮、图片什么的控件,那当然不必要去负担如此大的一个库。但是,如果是以Windows 9x这样为标准,那么就不是一般得复杂了,M$也不是白混的,还要继续坳的话,那么就请自己试实现一次吧,就会发现其实事情不像是看上去那么容易。

另外,CEGUI也是由人设计出来的,我坚信会有其他的大牛可以做得到。但是,这样做真的有必要吗,有可能你在De一个别人2年前早已修掉的Bug,而别人这时正在做下一代框架,干麻不花这个时间一起去完善它呢?

最后,我想就是开源的力量。凡事不去尝试,是不会了解到其真象的。为什么会有所谓头脑风暴,这就是集体的力量,广大人民群众齐心协力,会让人感到个人力量的有限。

那么,让我们放下成见,卸掉包袱,开始这一次CEGUI之旅。

 

 

设计思想

WidgetSets

CEGUI的设计思想是以窗口为单位的WidgetSets,它称作这些WidgetSetsxxxLook,例如自带的两个TaharezLookWindowsLook,也就是说在同一个Look里,所有的同类型的控件都长一个模样(这个可能无法满足我们通常游戏中的需要,所以要对其进行一些改造),感觉上比较像Windows98Theme(主题),只不是Theme的概念更大,包括了桌面、音效和鼠标等。

 

TaharezLook WindowsLook

 

       如上左右二图,可以看到,所有该Look所支持的Control类型所需要的图素都被一张图片所包含,假设需要更改样式和外观,可以设计多张拥有同样结构和相同元素的图片,然后换图即可。

 

 

 

 

体系结构

       CEGUI的窗口体系结构,跟以往我们所了解的一样,它底层的基类是Window,如下图:

 

       以上便是CEGUI提供给我们的控件集合,其他不在此范畴内的复合控件,也可以使用这些基本控件组合而成。

Window

可以看到,中间黑块中的Window,它继承于PropertySetEventSet。从这里开始,需要说明一个CEGUI中常见的概念:CEGUI中,如果存在某对象为xxx,通常会有一个xxxSet与之对应,而xxxSet的任务是对其进行管理或是分发的工作。因此,对于PropertySet而言,同时存在有Property,而Property的概念是:构建一个物件所必须的属性或组件。

举例来说,WindowProperties::AbsoluteHeight是一个在namespace WindowProperties中的一个Window属性AbsoluteHeight,用作描述Window的高度。同理,EventSet是全部Window事件的集合,其中就有像EventSized用作描述Window大小改变的事件(理解同“消息”)。

       Window拥有了PropertySetEventSet的特征后,在初始化的时候,它自己便会往里面“填入” 许多的属性和事件,丰富一番后,它也会定义一些接口,供子类继承或是供外部操作使用,像是会有接口virtual   void    drawSelf(float z) = 0;(供子类实现绘制),当然也会有一些公共的操作接口,如void setYPosition(float y);(设置坐标)。

       在上图的右边,有一长串由Window派生出来的子控件,也就是由这些控件构成了整个CEGUI,其中包括有基本的控件:按钮、文字、图片、编辑框等;也有较复杂的复合控件:列表框、表格、多行编辑框等,它们由多个基本控件组合而成。另外,作为一种附属窗体Tooltip,它就是当鼠标在某控件上悬停一会儿后出现的说明框。

下图中,描述了整个Window所拥有的信息,所有的事件响应,所有的基本属性:

 

       显而易见,这的确十分庞大,以致于我无法在不浪费页面的情况下,同时让这个体系图能够清晰得显示。

Property

       作为“属性”的描述,需要注意的是,所有的Property都是一个独立的class,哪怕只是一个简单的AbsoluteHeight,那为什么要把一个int变量搞得如此神秘和复杂呢?

原因有二个:

1.         操作接口化,使用Interface来隔离各模块,当功能发生变动,只需要修改实现,而接口不变

2.         序列化,便于Window在从文件中读取时存取和初始化各属性

而实现一个Property,基本上简单到只需要实现两个接口:

 

virtual String get(const PropertyReceiver* receiver) const = 0;

virtual void    set(PropertyReceiver* receiver, const String& value) = 0;

 

相同之处在于参数PropertyReceiver* receiver,其中receiver在不同控件中的Property代表着不同的含义,对于WindowProperties::AbsoluteHeight而言,receiver就等同于Window的实例,所以我们可以直接static_cast<Window*>(receiver)。因为每个Property都代表了不同的属性含义,在存取时也就需要不同的处理方式,所以传入一个宿主实例的指针,由Property自己决定应该做的事情。下面以WindowProperties::AbsoluteHeight的实现为例,相信只要看完之后,就会非常清楚Property的工作原理了。

 

String AbsoluteHeight::get(const PropertyReceiver* receiver) const

{

    return PropertyHelper::floatToString(static_cast<const Window*>(receiver)->getAbsoluteHeight());

}

 

void AbsoluteHeight::set(PropertyReceiver* receiver, const String& value)

{

    static_cast<Window*>(receiver)->setHeight(Absolute, PropertyHelper::stringToFloat(value));

}

   

    对,它仅仅只是再次调用了Window的接口去设置了一下,这也就是封装的概念和意义。

       出现了一个新面孔PropertyHelper,为了方便属性的存取,它提供了一些类似std::itoastd::atoi这样的函数来简化字符串操作;对于复杂的PropertyPropertyHelper通过定义一些规范的格式来操作,像是

 

Stringfloat的转换:

float PropertyHelper::stringToFloat(const String& str)

{

    using namespace std;

   

    float val = 0;

    sscanf(str.c_str(), " %f", &val);

   

    return val;

}

 

 

StringImage的转换:

const Image* PropertyHelper::stringToImage(const String& str)

{

    using namespace std;

   

    char imageSet[128] = {0};

    char imageName[128] = {0};

   

    sscanf(str.c_str(), " set:%127s image:%127s", imageSet, imageName);

   

    const Image* image;

   

    try

    {

        image = &ImagesetManager::getSingleton().getImageset((utf8*)imageSet)->getImage((utf8*)imageName);

    }

    catch (UnknownObjectException)

    {

        image = NULL;

    }

   

    return image;

}

 

 

 

 

Event

作为“事件”的描述,与Property不同的是,Event是以String实现的,它只是一段文字描述,当不同的事件发生时,CEGUI便会发送对应的Event来通知窗口。

一个Window会有很多像是EventMouseMoveEventKeyDownEventSized等等这样的事件。从名字上,就可以很容易得区分它们各自所代表的意义,以EventMouseMove为例 ,它的真身是const String Window::EventMouseMove( (utf8*)"MouseMove" );,是的,它就只是一个字符串而已。以EventMouseMove为例,当CEGUI底层在处理消息时,会判断鼠标是否在该窗体的区域范围中移动时,如果是,则通过接口

virtual void    fireEvent(const String& name, EventArgs& args, const String& eventNamespace = "");

来发送事件给该窗口。其中,name是消息字符串名称,args中存放着该消息对应的一些信息以供函数处理,像是EventMouseMove就对应MouseEventArgs来传递数据,以下是实现:

      

class CEGUIEXPORT MouseEventArgs : public WindowEventArgs

{

public:

    MouseEventArgs(Window* wnd) : WindowEventArgs(wnd) {}

   

    Point           position;           //!< holds current mouse position.

    Vector2     moveDelta;      //!< holds variation of mouse position from last mouse input

    MouseButton button;         //!< one of the MouseButton enumerated values describing the mouse button causing the event (for button inputs only)

    uint            sysKeys;            //!< current state of the system keys and mouse buttons.

    float           wheelChange;        //!< Holds the amount the scroll wheel has changed.

    uint            clickCount;     //!< Holds number of mouse button down events currently counted in a multi-click sequence (for button inputs only).

};

      

因为WindowEventArgs是从EventArgs派生过来的,那么Window就可以通过成员函数

virtual void    onMouseMove(MouseEventArgs& e);来响应该事件了。

                      

 

哦,我不会忘记这里还有一个参数eventNamespace,还是举例说明一下吧,在Window中,它就是const String Window::EventNamespace("Window");,用来区分在不同控件中可能会出现的同名事件。

小结

上面只是简单扼要得介绍了一些CEGUI的基础概念,对于一个熟悉Window的人而言,可能会觉得“不过如此”,但是,事情往往说起来容易做起来难。从整个设计体系来看,固然一个Window like的系统怎么也逃不出这些个概念,然而在控件的细节实现上,还是有很多复杂繁琐的东西需要去实现。

 

 

 

 

 

 

 

 

 

 

 

 

 

渲染器

前面说了那么多逻辑层的底层机制,接下来想要将CEGUI的界面显示出来,则必须要实现两个类:TextureRenderer。它们算作是“渲染底层”;而CEGUI会在此基础上再完成一些“中间层”(像是Image之类);最上面才是控件类,共三层构成了整个CEGUI

Texture

       实现Texture需要重载几个接口,依次是:

      

virtual ushort getWidth(void) const = 0;

virtual ushort getHeight(void) const = 0;

virtual void    loadFromFile(const String& filename, const String& resourceGroup) = 0;

virtual void    loadFromMemory(const void* buffPtr, uint buffWidth, uint buffHeight) = 0;

   

    CEGUI需要通过这些接口操作纹理对象:得到纹理的宽度和高度、二种不同的载入方式。这里唯一需要解释的部分就是const String& resourceGroup,通过使用不同的“组” 前缀名,以区分可能相同名称的资源名,保证资源唯一ID的存取。

 

       Texture虽然很简单,但它却是Renderer实现所必须的一个重要组成部件。

Renderer

       实现Renderer需要重载更多的接口,因为数量比较多,且不像Texture的接口那么容易从字面上理解,所以我在下面会分别作解释:

 

virtual void    addQuad(const Rect& dest_rect, float z, const Texture* tex, const Rect& texture_rect, const ColourRect& colours, QuadSplitMode quad_split_mode) = 0;

增加一个Quad到渲染缓冲中。因为对象是Quad,所以一些参数都是以Rect4个顶点)为单位在描述,这可能会和以往的了解有些许不同:

dest_rect,              目标位置

z                       前后层次关系

tex                   纹理指针

texture_rect,         纹理坐标

colours                   顶点颜色

quad_split_mode     4个顶点的顺序(顺时针、逆时针)

 

virtual void    doRender(void) = 0;

渲染全部UI(整个Quad缓冲)

 

virtual void    clearRenderList(void) = 0;

清空全部渲染缓冲

 

virtual void    setQueueingEnabled(bool setting) = 0;

对于Quad的渲染分为“立即模式”和“缓冲模式”,这里是两种模式的切换开关

 

virtual Texture*    createTexture(void) = 0;

描述Renderer如何创建一个Texture,通常就是new一个Texture后返回指针

 

virtual Texture*    createTexture(const String& filename, const String& resourceGroup) = 0;

描述Renderer如何从文件中创建一个Texture,通常是调用上面的函数后得到新建的Texture,然后调用TextureloadFromFile

 

virtual Texture*    createTexture(float size) = 0;

描述Renderer如何根据指定的大小来创建一个Texture,通常是调用上面的函数后得到新建的Texture,然后根据size创建一块临时的内存,最后调用TextureloadFromMemory

 

virtual void    destroyTexture(Texture* texture) = 0;

销毁指定的Texture,通常Renderer都会保存一份Texture的列表便于管理,这里除了会delete传入的指针外,还会从管理列表中删除它

 

virtual void    destroyAllTextures(void) = 0;

销毁纹理列表中的全部纹理

 

virtual bool    isQueueingEnabled(void) const = 0;

查询缓冲渲染模式是否打开

 

virtual float   getWidth(void) const    = 0;

得到渲染设备的宽度,通常就是Viewport的宽度

 

virtual float   getHeight(void) const   = 0;

得到

抱歉!评论已关闭.