Planet
navi homePPSaboutscreenshotsdownloaddevelopmentforum

source: code/branches/ingamemenu/src/libraries/core/GUIManager.cc @ 6003

Last change on this file since 6003 was 6003, checked in by dafrick, 15 years ago

Implemented support for multiple simultaniously showing GUIs. What happens now, in essence, is, that every root-window of a gui, that is to be showed, is attached to a global root window, which is always displayed and therefore needs to be fully transparent (alpha = 0.0). To not inherit the transparency (and thus be fully transparent as well) each root-window of a gui must (or should) set the property 'InheritsAlpha' to false.

  • Property svn:eol-style set to native
File size: 11.8 KB
Line 
1/*
2 *   ORXONOX - the hottest 3D action shooter ever to exist
3 *                    > www.orxonox.net <
4 *
5 *
6 *   License notice:
7 *
8 *   This program is free software; you can redistribute it and/or
9 *   modify it under the terms of the GNU General Public License
10 *   as published by the Free Software Foundation; either version 2
11 *   of the License, or (at your option) any later version.
12 *
13 *   This program is distributed in the hope that it will be useful,
14 *   but WITHOUT ANY WARRANTY; without even the implied warranty of
15 *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
16 *   GNU General Public License for more details.
17 *
18 *   You should have received a copy of the GNU General Public License
19 *   along with this program; if not, write to the Free Software
20 *   Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301, USA.
21 *
22 *   Author:
23 *      Reto Grieder
24 *      Benjamin Knecht
25 *   Co-authors:
26 *      ...
27 *
28 */
29
30#include "GUIManager.h"
31
32#include <memory>
33extern "C" {
34#include <lua.h>
35}
36#include <CEGUIDefaultLogger.h>
37#include <CEGUIExceptions.h>
38#include <CEGUIInputEvent.h>
39#include <CEGUIMouseCursor.h>
40#include <CEGUIResourceProvider.h>
41#include <CEGUISystem.h>
42#include <ogreceguirenderer/OgreCEGUIRenderer.h>
43
44#include "SpecialConfig.h" // Configures the macro below
45#ifdef CEGUILUA_USE_INTERNAL_LIBRARY
46#   include <ceguilua/CEGUILua.h>
47#else
48#   include <CEGUILua.h>
49#endif
50
51#include "util/Clock.h"
52#include "util/Debug.h"
53#include "util/Exception.h"
54#include "util/OrxAssert.h"
55#include "ConsoleCommand.h"
56#include "Core.h"
57#include "input/InputManager.h"
58#include "LuaState.h"
59#include "PathConfig.h"
60#include "Resource.h"
61
62namespace orxonox
63{
64    class CEGUILogger : public CEGUI::DefaultLogger
65    {
66    public:
67        void logEvent(const CEGUI::String& message, CEGUI::LoggingLevel level = CEGUI::Standard)
68        {
69            int orxonoxLevel = CEGUI::Standard;
70            switch (level)
71            {
72                case CEGUI::Errors:      orxonoxLevel = 1; break;
73                case CEGUI::Warnings:    orxonoxLevel = 2; break;
74                case CEGUI::Standard:    orxonoxLevel = 4; break;
75                case CEGUI::Informative: orxonoxLevel = 5; break;
76                case CEGUI::Insane:      orxonoxLevel = 6; break;
77                default: OrxAssert(false, "CEGUI log level out of range, inpect immediately!");
78            }
79            OutputHandler::getOutStream().setOutputLevel(orxonoxLevel)
80                << "CEGUI: " << message << std::endl;
81
82            CEGUI::DefaultLogger::logEvent(message, level);
83        }
84    };
85
86    static CEGUI::MouseButton convertButton(MouseButtonCode::ByEnum button);
87
88    GUIManager* GUIManager::singletonPtr_s = 0;
89
90    SetConsoleCommandShortcut(GUIManager, showGUI).accessLevel(AccessLevel::User);
91    SetConsoleCommandShortcut(GUIManager, hideGUI).accessLevel(AccessLevel::User);
92
93    /**
94    @brief
95        Constructs the GUIManager by starting up CEGUI
96
97        Creates the interface to Ogre, sets up the CEGUI renderer and the Lua script module together with the Lua engine.
98        The log is set up and connected to the CEGUILogger.
99        After Lua setup tolua++-elements are linked to Lua-state to give Lua access to C++-code.
100        Finally initial Lua code is executed (maybe we can do this with the CEGUI startup script automatically).
101    @param renderWindow
102        Ogre's render window. Without this, the GUI cannot be displayed.
103    @return true if success, otherwise false
104    */
105    GUIManager::GUIManager(Ogre::RenderWindow* renderWindow, const std::pair<int, int>& mousePosition, bool bFullScreen)
106        : renderWindow_(renderWindow)
107        , resourceProvider_(0)
108        , camera_(NULL)
109    {
110        using namespace CEGUI;
111
112        COUT(3) << "Initialising CEGUI." << std::endl;
113
114        // Note: No SceneManager specified yet
115        guiRenderer_.reset(new OgreCEGUIRenderer(renderWindow_, Ogre::RENDER_QUEUE_OVERLAY, false, 3000));
116        resourceProvider_ = guiRenderer_->createResourceProvider();
117        resourceProvider_->setDefaultResourceGroup("GUI");
118
119        // setup scripting
120        luaState_.reset(new LuaState());
121        scriptModule_.reset(new LuaScriptModule(luaState_->getInternalLuaState()));
122
123        // Create our own logger to specify the filepath
124        std::auto_ptr<CEGUILogger> ceguiLogger(new CEGUILogger());
125        ceguiLogger->setLogFilename(PathConfig::getLogPathString() + "cegui.log");
126        // set the log level according to ours (translate by subtracting 1)
127        ceguiLogger->setLoggingLevel(
128            static_cast<LoggingLevel>(Core::getSoftDebugLevel(OutputHandler::LD_Logfile) - 1));
129        this->ceguiLogger_ = ceguiLogger.release();
130
131        // create the CEGUI system singleton
132        guiSystem_.reset(new System(guiRenderer_.get(), resourceProvider_, 0, scriptModule_.get()));
133
134        // Initialise the basic lua code
135        rootFileInfo_ = Resource::getInfo("InitialiseGUI.lua", "GUI");
136        this->luaState_->doFile("InitialiseGUI.lua", "GUI", false);
137
138        // Align CEGUI mouse with OIS mouse
139        guiSystem_->injectMousePosition(mousePosition.first, mousePosition.second);
140
141        // Hide the mouse cursor unless playing in fullscreen mode
142        if (!bFullScreen)
143            CEGUI::MouseCursor::getSingleton().hide();
144    }
145
146    /**
147    @brief
148        Basically shuts down CEGUI (member smart pointers) but first unloads our Tolua modules.
149    */
150    GUIManager::~GUIManager()
151    {
152    }
153
154    /**
155    @brief
156        used to tick the GUI
157    @param time
158        clock which provides time value for the GUI System
159
160        Ticking the GUI means updating it with a certain regularity.
161        The elapsed time since the last call is given in the time value provided by the clock.
162        This time value is then used to provide a fluent animation of the GUI.
163    */
164    void GUIManager::update(const Clock& time)
165    {
166        assert(guiSystem_);
167        guiSystem_->injectTimePulse(time.getDeltaTime());
168    }
169
170    /**
171    @brief
172        Tells the GUIManager which SceneManager to use
173    @param camera
174        The current camera on which the GUI should be displayed on.
175
176        In fact the GUIManager needs the SceneManager and not the Camera to display the GUI.
177        This means the GUI is not bound to a camera but rather to the SceneManager.
178        Hiding the GUI when needed can therefore not be resolved by just NOT setting the current camera.
179    */
180    void GUIManager::setCamera(Ogre::Camera* camera)
181    {
182        this->camera_ = camera;
183        if (camera == NULL)
184            this->guiRenderer_->setTargetSceneManager(0);
185        else
186            this->guiRenderer_->setTargetSceneManager(camera->getSceneManager());
187    }
188
189    /**
190    @brief
191        Executes Lua code
192    @param str
193        reference to string object holding the Lua code which is to be executed
194
195        This function gives total access to the GUI. You can execute ANY Lua code here.
196    */
197    void GUIManager::executeCode(const std::string& str)
198    {
199        this->luaState_->doString(str, rootFileInfo_);
200    }
201
202    /**
203    @brief
204        Displays specified GUI on screen
205    @param name
206        The name of the GUI
207
208        The function executes the Lua function with the same name in case the GUIManager is ready.
209        For more details check out loadGUI_2.lua where the function presides.
210    */
211    /*static*/ void GUIManager::showGUI(const std::string& name)
212    {
213        std::pair<std::set<std::string>::iterator,bool> result = GUIManager::getInstance().showingGUIs_.insert(name);
214        if(result.second == false) //!< GUI already showing.
215            return;
216        if(GUIManager::getInstance().showingGUIs_.size() == 1 && result.second == true) //!< If it's the first GUI.
217        {
218            GUIManager::getInstance().executeCode("showCursor()");
219            InputManager::getInstance().enterState("guiMouseOnly");
220        }
221        GUIManager::getInstance().executeCode("showGUI(\"" + name + "\")");
222    }
223
224    /**
225    @brief
226        Hack-ish. Needed for GUIOverlay.
227    */
228    void GUIManager::showGUIExtra(const std::string& name, const std::string& ptr)
229    {
230        std::pair<std::set<std::string>::iterator,bool> result = this->showingGUIs_.insert(name);
231        if(result.second == false) //!< GUI already showing.
232            return;
233        if(this->showingGUIs_.size() == 1 && result.second == true) //!< If it's the first GUI.
234        {
235            this->executeCode("showCursor()");
236            InputManager::getInstance().enterState("guiMouseOnly");
237        }
238        this->executeCode("showGUI(\"" + name + "\", " + ptr + ")");
239    }
240
241    /**
242    @brief
243        Hides specified GUI.
244    @param name
245        The name of the GUI.
246    */
247    /*static*/ void GUIManager::hideGUI(const std::string& name)
248    {
249        bool present = GUIManager::getInstance().showingGUIs_.erase(name);
250        if(!present) //!< If there was nothing to erase.
251            return;
252        GUIManager::getInstance().executeCode("hideGUI(\"" + name + "\")");
253        if(GUIManager::getInstance().showingGUIs_.size() == 0)
254        {
255            GUIManager::getInstance().executeCode("hideCursor()");
256            InputManager::getInstance().leaveState("guiMouseOnly");
257        }
258    }
259
260    void GUIManager::keyPressed(const KeyEvent& evt)
261    {
262        guiSystem_->injectKeyDown(evt.getKeyCode());
263        guiSystem_->injectChar(evt.getText());
264    }
265    void GUIManager::keyReleased(const KeyEvent& evt)
266    {
267        guiSystem_->injectKeyUp(evt.getKeyCode());
268    }
269
270    /**
271    @brief
272        Function receiving a mouse button pressed event.
273    @param id
274        ID of the mouse button which got pressed
275
276        This function is inherited by MouseHandler and injects the event into CEGUI.
277        It is for CEGUI to process the event.
278    */
279    void GUIManager::buttonPressed(MouseButtonCode::ByEnum id)
280    {
281        try
282        {
283            guiSystem_->injectMouseButtonDown(convertButton(id));
284        }
285        catch (CEGUI::ScriptException& ex)
286        {
287            // We simply ignore the exception and proceed
288            COUT(1) << ex.getMessage() << std::endl;
289        }
290    }
291
292    /**
293    @brief
294        Function receiving a mouse button released event.
295    @param id
296        ID of the mouse button which got released
297
298        This function is inherited by MouseHandler and injects the event into CEGUI.
299        It is for CEGUI to process the event.
300    */
301    void GUIManager::buttonReleased(MouseButtonCode::ByEnum id)
302    {
303        try
304        {
305            guiSystem_->injectMouseButtonUp(convertButton(id));
306        }
307        catch (CEGUI::ScriptException& ex)
308        {
309            // We simply ignore the exception and proceed
310            COUT(1) << ex.getMessage() << std::endl;
311        }
312    }
313
314    void GUIManager::mouseMoved(IntVector2 abs, IntVector2 rel, IntVector2 clippingSize)
315    {
316        guiSystem_->injectMousePosition(static_cast<float>(abs.x), static_cast<float>(abs.y));
317    }
318    void GUIManager::mouseScrolled(int abs, int rel)
319    {
320        guiSystem_->injectMouseWheelChange(static_cast<float>(rel));
321    }
322
323    /**
324    @brief
325        converts mouse event code to CEGUI event code
326    @param button
327        code of the mouse button as we use it in Orxonox
328    @return
329        code of the mouse button as it is used by CEGUI
330
331        Simple convertion from mouse event code in Orxonox to the one used in CEGUI.
332     */
333    static inline CEGUI::MouseButton convertButton(MouseButtonCode::ByEnum button)
334    {
335        switch (button)
336        {
337        case MouseButtonCode::Left:
338            return CEGUI::LeftButton;
339
340        case MouseButtonCode::Right:
341            return CEGUI::RightButton;
342
343        case MouseButtonCode::Middle:
344            return CEGUI::MiddleButton;
345
346        case MouseButtonCode::Button3:
347            return CEGUI::X1Button;
348
349        case MouseButtonCode::Button4:
350            return CEGUI::X2Button;
351
352        default:
353            return CEGUI::NoButton;
354        }
355    }
356}
Note: See TracBrowser for help on using the repository browser.