| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465 |
- /*
- -----------------------------------------------------------------------------
- This source file is part of OGRE
- (Object-oriented Graphics Rendering Engine)
- For the latest info, see http://www.ogre3d.org/
- Copyright (c) 2000-2011 Torus Knot Software Ltd
- Permission is hereby granted, free of charge, to any person obtaining a copy
- of this software and associated documentation files (the "Software"), to deal
- in the Software without restriction, including without limitation the rights
- to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
- copies of the Software, and to permit persons to whom the Software is
- furnished to do so, subject to the following conditions:
- The above copyright notice and this permission notice shall be included in
- all copies or substantial portions of the Software.
- THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
- IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
- FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
- AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
- LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
- OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
- THE SOFTWARE.
- -----------------------------------------------------------------------------
- */
- #include "CmWindowEventUtilities.h"
- #include "CmRenderWindow.h"
- #include "CmApplication.h"
- #include "CmException.h"
- #include "CmCursor.h"
- #include "CmInput.h"
- #if CM_PLATFORM == CM_PLATFORM_LINUX
- #include <X11/Xlib.h>
- void GLXProc( CamelotFramework::RenderWindow *win, const XEvent &event );
- #endif
- using namespace CamelotFramework;
- WindowEventUtilities::WindowEventListeners WindowEventUtilities::_msListeners;
- WindowEventUtilities::Windows WindowEventUtilities::_msWindows;
- //--------------------------------------------------------------------------------//
- void WindowEventUtilities::messagePump()
- {
- #if CM_PLATFORM == CM_PLATFORM_WIN32
- // Windows Message Loop (NULL means check all HWNDs belonging to this context)
- MSG msg;
- while( PeekMessage( &msg, NULL, 0U, 0U, PM_REMOVE ) )
- {
- TranslateMessage( &msg );
- DispatchMessage( &msg );
- }
- #elif CM_PLATFORM == CM_PLATFORM_LINUX
- //GLX Message Pump
- Windows::iterator win = _msWindows.begin();
- Windows::iterator end = _msWindows.end();
- Display* xDisplay = 0; // same for all windows
-
- for (; win != end; win++)
- {
- XID xid;
- XEvent event;
- if (!xDisplay)
- (*win)->getCustomAttribute("XDISPLAY", &xDisplay);
- (*win)->getCustomAttribute("WINDOW", &xid);
- while (XCheckWindowEvent (xDisplay, xid, StructureNotifyMask | VisibilityChangeMask | FocusChangeMask, &event))
- {
- GLXProc(*win, event);
- }
- // The ClientMessage event does not appear under any Event Mask
- while (XCheckTypedWindowEvent (xDisplay, xid, ClientMessage, &event))
- {
- GLXProc(*win, event);
- }
- }
- #elif CM_PLATFORM == CM_PLATFORM_APPLE && !defined __OBJC__ && !defined __LP64__
- // OSX Message Pump
- EventRef event = NULL;
- EventTargetRef targetWindow;
- targetWindow = GetEventDispatcherTarget();
-
- // If we are unable to get the target then we no longer care about events.
- if( !targetWindow ) return;
-
- // Grab the next event, process it if it is a window event
- if( ReceiveNextEvent( 0, NULL, kEventDurationNoWait, true, &event ) == noErr )
- {
- // Dispatch the event
- SendEventToEventTarget( event, targetWindow );
- ReleaseEvent( event );
- }
- #endif
- }
- //--------------------------------------------------------------------------------//
- void WindowEventUtilities::addWindowEventListener( RenderWindow* window, WindowEventListener* listener )
- {
- _msListeners.insert(std::make_pair(window, listener));
- }
- //--------------------------------------------------------------------------------//
- void WindowEventUtilities::removeWindowEventListener( RenderWindow* window, WindowEventListener* listener )
- {
- WindowEventListeners::iterator i = _msListeners.begin(), e = _msListeners.end();
- for( ; i != e; ++i )
- {
- if( i->first == window && i->second == listener )
- {
- _msListeners.erase(i);
- break;
- }
- }
- }
- //--------------------------------------------------------------------------------//
- void WindowEventUtilities::_addRenderWindow(RenderWindow* window)
- {
- _msWindows.push_back(window);
- }
- //--------------------------------------------------------------------------------//
- void WindowEventUtilities::_removeRenderWindow(RenderWindow* window)
- {
- Windows::iterator i = std::find(_msWindows.begin(), _msWindows.end(), window);
- if( i != _msWindows.end() )
- _msWindows.erase( i );
- }
- #if CM_PLATFORM == CM_PLATFORM_WIN32
- //--------------------------------------------------------------------------------//
- LRESULT CALLBACK WindowEventUtilities::_WndProc(HWND hWnd, UINT uMsg, WPARAM wParam, LPARAM lParam)
- {
- if (uMsg == WM_CREATE)
- { // Store pointer to Win32Window in user data area
- SetWindowLongPtr(hWnd, GWLP_USERDATA, (LONG_PTR)(((LPCREATESTRUCT)lParam)->lpCreateParams));
- return 0;
- }
- // look up window instance
- // note: it is possible to get a WM_SIZE before WM_CREATE
- RenderWindow* win = (RenderWindow*)GetWindowLongPtr(hWnd, GWLP_USERDATA);
- if (!win)
- return DefWindowProc(hWnd, uMsg, wParam, lParam);
- //LogManager* log = LogManager::getSingletonPtr();
- //Iterator of all listeners registered to this RenderWindow
- WindowEventListeners::iterator index,
- start = _msListeners.lower_bound(win),
- end = _msListeners.upper_bound(win);
- switch( uMsg )
- {
- case WM_ACTIVATE:
- {
- bool active = (LOWORD(wParam) != WA_INACTIVE);
- if( active )
- {
- win->setActive( true );
- if(!win->hasFocus())
- win->setHasFocus(true);
- gInput().inputWindowChanged(*win);
- }
- else
- {
- if( win->isDeactivatedOnFocusChange() )
- {
- win->setActive( false );
- }
- if(win->hasFocus())
- win->setHasFocus(false);
- }
- for( ; start != end; ++start )
- (start->second)->windowFocusChange(win);
- break;
- }
- case WM_SYSKEYDOWN:
- switch( wParam )
- {
- case VK_CONTROL:
- case VK_SHIFT:
- case VK_MENU: //ALT
- //return zero to bypass defProc and signal we processed the message
- return 0;
- }
- break;
- case WM_SYSKEYUP:
- switch( wParam )
- {
- case VK_CONTROL:
- case VK_SHIFT:
- case VK_MENU: //ALT
- case VK_F10:
- //return zero to bypass defProc and signal we processed the message
- return 0;
- }
- break;
- case WM_SYSCHAR:
- // return zero to bypass defProc and signal we processed the message, unless it's an ALT-space
- if (wParam != VK_SPACE)
- return 0;
- break;
- case WM_ENTERSIZEMOVE:
- //log->logMessage("WM_ENTERSIZEMOVE");
- break;
- case WM_EXITSIZEMOVE:
- //log->logMessage("WM_EXITSIZEMOVE");
- break;
- case WM_MOVE:
- //log->logMessage("WM_MOVE");
- win->windowMovedOrResized();
- for(index = start; index != end; ++index)
- (index->second)->windowMoved(win);
- break;
- case WM_DISPLAYCHANGE:
- win->windowMovedOrResized();
- for(index = start; index != end; ++index)
- (index->second)->windowResized(win);
- break;
- case WM_SIZE:
- //log->logMessage("WM_SIZE");
- win->windowMovedOrResized();
- for(index = start; index != end; ++index)
- (index->second)->windowResized(win);
- break;
- case WM_SETCURSOR:
- if(Cursor::isHidden())
- Cursor::_win32HideCursor();
- else
- Cursor::_win32ShowCursor();
- return true;
- case WM_GETMINMAXINFO:
- // Prevent the window from going smaller than some minimu size
- ((MINMAXINFO*)lParam)->ptMinTrackSize.x = 100;
- ((MINMAXINFO*)lParam)->ptMinTrackSize.y = 100;
- break;
- case WM_CLOSE:
- {
- //log->logMessage("WM_CLOSE");
- bool close = true;
- for(index = start; index != end; ++index)
- {
- if (!(index->second)->windowClosing(win))
- close = false;
- }
- if (!close) return 0;
- for(index = _msListeners.lower_bound(win); index != end; ++index)
- (index->second)->windowClosed(win);
- gApplication().stopMainLoop();
- return 0;
- }
- }
- return DefWindowProc( hWnd, uMsg, wParam, lParam );
- }
- #elif CM_PLATFORM == CM_PLATFORM_LINUX
- //--------------------------------------------------------------------------------//
- void GLXProc( RenderWindow *win, const XEvent &event )
- {
- //An iterator for the window listeners
- WindowEventUtilities::WindowEventListeners::iterator index,
- start = WindowEventUtilities::_msListeners.lower_bound(win),
- end = WindowEventUtilities::_msListeners.upper_bound(win);
- switch(event.type)
- {
- case ClientMessage:
- {
- ::Atom atom;
- win->getCustomAttribute("ATOM", &atom);
- if(event.xclient.format == 32 && event.xclient.data.l[0] == (long)atom)
- { //Window closed by window manager
- //Send message first, to allow app chance to unregister things that need done before
- //window is shutdown
- bool close = true;
- for(index = start ; index != end; ++index)
- {
- if (!(index->second)->windowClosing(win))
- close = false;
- }
- if (!close) return;
- for(index = start ; index != end; ++index)
- (index->second)->windowClosed(win);
- }
- break;
- }
- case DestroyNotify:
- {
- if (!win->isClosed())
- {
- // Window closed without window manager warning.
- for(index = start ; index != end; ++index)
- (index->second)->windowClosed(win);
- }
- break;
- }
- case ConfigureNotify:
- {
- // This could be slightly more efficient if windowMovedOrResized took arguments:
- unsigned int oldWidth, oldHeight, oldDepth;
- int oldLeft, oldTop;
- win->getMetrics(oldWidth, oldHeight, oldDepth, oldLeft, oldTop);
- win->windowMovedOrResized();
- unsigned int newWidth, newHeight, newDepth;
- int newLeft, newTop;
- win->getMetrics(newWidth, newHeight, newDepth, newLeft, newTop);
- if (newLeft != oldLeft || newTop != oldTop)
- {
- for(index = start ; index != end; ++index)
- (index->second)->windowMoved(win);
- }
- if (newWidth != oldWidth || newHeight != oldHeight)
- {
- for(index = start ; index != end; ++index)
- (index->second)->windowResized(win);
- }
- break;
- }
- case FocusIn: // Gained keyboard focus
- case FocusOut: // Lost keyboard focus
- for(index = start ; index != end; ++index)
- (index->second)->windowFocusChange(win);
- break;
- case MapNotify: //Restored
- win->setActive( true );
- for(index = start ; index != end; ++index)
- (index->second)->windowFocusChange(win);
- break;
- case UnmapNotify: //Minimised
- win->setActive( false );
- win->setVisible( false );
- for(index = start ; index != end; ++index)
- (index->second)->windowFocusChange(win);
- break;
- case VisibilityNotify:
- switch(event.xvisibility.state)
- {
- case VisibilityUnobscured:
- win->setActive( true );
- win->setVisible( true );
- break;
- case VisibilityPartiallyObscured:
- win->setActive( true );
- win->setVisible( true );
- break;
- case VisibilityFullyObscured:
- win->setActive( false );
- win->setVisible( false );
- break;
- }
- for(index = start ; index != end; ++index)
- (index->second)->windowFocusChange(win);
- break;
- default:
- break;
- } //End switch event.type
- }
- #elif CM_PLATFORM == CM_PLATFORM_APPLE && !defined __OBJC__ && !defined __LP64__
- //--------------------------------------------------------------------------------//
- OSStatus WindowEventUtilities::_CarbonWindowHandler(EventHandlerCallRef nextHandler, EventRef event, void* wnd)
- {
- OSStatus status = noErr;
-
- // Only events from our window should make it here
- // This ensures that our user data is our WindowRef
- RenderWindow* curWindow = (RenderWindow*)wnd;
- if(!curWindow) return eventNotHandledErr;
-
- //Iterator of all listeners registered to this RenderWindow
- WindowEventListeners::iterator index,
- start = _msListeners.lower_bound(curWindow),
- end = _msListeners.upper_bound(curWindow);
-
- // We only get called if a window event happens
- UInt32 eventKind = GetEventKind( event );
- switch( eventKind )
- {
- case kEventWindowActivated:
- curWindow->setActive( true );
- for( ; start != end; ++start )
- (start->second)->windowFocusChange(curWindow);
- break;
- case kEventWindowDeactivated:
- if( curWindow->isDeactivatedOnFocusChange() )
- {
- curWindow->setActive( false );
- }
- for( ; start != end; ++start )
- (start->second)->windowFocusChange(curWindow);
- break;
- case kEventWindowShown:
- case kEventWindowExpanded:
- curWindow->setActive( true );
- curWindow->setVisible( true );
- for( ; start != end; ++start )
- (start->second)->windowFocusChange(curWindow);
- break;
- case kEventWindowHidden:
- case kEventWindowCollapsed:
- curWindow->setActive( false );
- curWindow->setVisible( false );
- for( ; start != end; ++start )
- (start->second)->windowFocusChange(curWindow);
- break;
- case kEventWindowDragCompleted:
- curWindow->windowMovedOrResized();
- for( ; start != end; ++start )
- (start->second)->windowMoved(curWindow);
- break;
- case kEventWindowBoundsChanged:
- curWindow->windowMovedOrResized();
- for( ; start != end; ++start )
- (start->second)->windowResized(curWindow);
- break;
- case kEventWindowClose:
- {
- bool close = true;
- for( ; start != end; ++start )
- {
- if (!(start->second)->windowClosing(curWindow))
- close = false;
- }
- if (close)
- // This will cause event handling to continue on to the standard handler, which calls
- // DisposeWindow(), which leads to the 'kEventWindowClosed' event
- status = eventNotHandledErr;
- break;
- }
- case kEventWindowClosed:
- for( ; start != end; ++start )
- (start->second)->windowClosed(curWindow);
- break;
- default:
- status = eventNotHandledErr;
- break;
- }
-
- return status;
- }
- #endif
|