= Graphical User Interface =
[[TracNav(TracNav/TOC_Development)]]
The GUI in Orxonox is administered with CEGUI. We use the GUI for interactions with the user in shops, [wiki:Questsystem] or parts of the HUD and of course the main menu.
== CEGUI ==
CEGUI (Crazy Edi's GUI) is available for various graphic engines like [http://www.ogre3d.org Ogre] or [http://irrlicht.sourceforge.net/ Irrlicht]. In Orxonox we use the [http://www.orxonox.net/doxygen/classCEGUI_1_1OgreCEGUIRenderer.html OgreCEGUIRenderer] to display our GUI.
== GUI Framework ==
The framework consists of various elements, putting all necessary elements together for a functioning GUI. Those elements are basically the GUIManager, Lua-scripts, layouts and the corresponding graphics.
=== GUIManager ===
Written in C++ the GUIManager manages all the GUIs in Orxonox and represents the interface to CEGUI. Connected to Lua via a tolua++ interface it is able to load Lua-scripts and connect C++-functions to Lua and vice versa.
To use the GUIManager in C++-code you can call it easily as it is a singleton:
{{{
#include "gui/GUIManager.h"
...
GUIManager guiMan = GUIManager::getInstance();
GUIManager* guiManPtr = GUIManager::getInstancePtr();
}}}
Find out more about the GUIManager's methods in the [http://www.orxonox.net/doxygen/classorxonox_1_1GUIManager.html Doxygen-Documentation]
To access C++-code in Lua you have to mark them. You can see how this is done on [wiki:Scripting this page]. Afterwards calling a method of a C++-class you can proceed as below (''--'' denote commentaries in Lua)
{{{
-- using namespace orxonox and calling method "getInstance()" of GUIManager
guiman = orxonox.GUIManager:getInstance()
}}}
=== Lua-scripts ===
Lua-scripts are written in Lua and implement the functionalities of CEGUI like interactivity. Each GUI needs a separate Lua-script which will be loaded by the main script. Our implementation ensures that a GUI cannot be loaded more than once.
If you want to create a new GUI you need to create a lua file with the following header. Of course you would change ''mainmenu'' to the name of your GUI. The name of the GUI and the filename should be the same.
{{{
gui = require("gui")
local P = gui:new() --inherit everything from the gui package
mainmenu = P
P.filename = "mainmenu"
P.layoutString = "MainMenu.layout"
...
return mainmenu
}}}
Specify events in your GUI as follows:
{{{
-- events for mainmenu
function P.button_quit_clicked(e)
... -- do something
end
}}}
If you want to execute code in Lua just when the GUI is loading (for instance manipulating the loaded elements) you can write an ''init'' function which will automatically executed when the GUI is loading.
To load a GUI from inside Orxonox, you can just call the showGUI()-function of the GUIManager with the file name of your Lua-script:
{{{
// shows the GUI called "mainmenu" and loads it beforehand if necessary
GUIManager::getInstance().showGUI("mainmenu");
}}}
=== Imagesets ===
Images are loaded into so called imagesets. As the name suggests, you can group several images into one single imageset. This can be used for instance when defining "active", "hover", "clicked" and "inactive" state of a button. You'd load just one image file into the imageset, but define various images from that imageset.
Imageset files are written in XML and very easy to use. To be able to use the specified images though, you need to include the imageset into a scheme file. Schemes are explicated in the next section.
To add an image file to a imageset and defining one or more images from this imageset this example code should help:
{{{
}}}
=== Schemes ===
Schemes define a certain style and provide the Lua-script with the necessary resources. It is written in XML. At the moment we just use the standard-scheme for the window look and for our own images ([http://svn.orxonox.net/data/media/gui/schemes/OrxonoxGUIScheme.scheme OrxonoxGUIScheme.scheme]). Since this is included in the main script, you just have to add your new imageset to the scheme and can forget about it.
=== Layouts ===
Layouts define the structure of the GUI. It defines the positions of the various elements and their size and behavior. You can also add events to the elements of the GUI and process the event in the Lua-script. Everything is written in XML, but you can access every parameter of each element in the layout. The layout however reaches its limits soon, but it is advised to define as much as possible in the layout.
=== GUI-Elements ===
Every GUI-element is a window. The underlying class-structure inherits everything from a basic window element. Buttons and text input elements implement various new parameters and functions. They also need different kinds of images. At the moment we advise to use a predefined look for our GUI, namely the !TaharezLook. However this may change in the future, when we want to define our own looks. Check out the parameters and functions of each element in the layouts already in the media directory or in the [http://www.cegui.org.uk/api_reference/ CEGUI-API]. The API is written for C++, but you can access the same function in a similar fashion, using Lua or XML.
==== Window ====
Every element in CEGUI is a window. So all elements have the following properties. You may want to disable or enable certain elements to render them unresponsive to user input. Some elements like buttons do change their look, when you change their state. Every element is enabled by default. To disable an element from the beginning you can add this line to your layout:
{{{
}}}
Please note, that there is no property called ''Enabled''. So trying to set it to ''false'' will not work. In Lua however you can either enable or disable a window simply like this:
{{{
win = windowManager:createWindow("TaharezLook/Button", "test")
if b_disableNewButtons == false then
win:enable()
else
win:disable()
end
}}}
==== Listbox ====
Listbox is a box where you have a list of ListboxItems which you can choose from. See this example on how to create a Listbox in Lua:
{{{
list = windowManager:createWindow("TaharezLook/Listbox", "listbox")
item = CEGUI.createListboxTextItem("Text")
CEGUI.toListbox(list):addItem(item)
}}}
To be able to choose an item in the Listbox use
{{{
item:setSelectionBrushImage("TaharezLook", "MultiListSelectionBrush")
}}}
The selected item is received when calling
{{{
list:getFirstSelectedItem()
}}}
If nothing is chosen, the function returns ''nil''.
To react on events use
* ''ListItemsChanged'' - For changed content of the list
* ''ItemSelectionChanged'' - If the selection of an item changed
Other events are easy to figure out from their documented event name. Just drop the ''Event'' and you're good to go.