1 ///////////////////////////////////////////////////////////////////////////////
2 // Name: x11/evtloop.cpp
3 // Purpose: implements wxEventLoop for X11
4 // Author: Julian Smart
8 // Copyright: (c) 2002 Julian Smart
9 // License: wxWindows licence
10 ///////////////////////////////////////////////////////////////////////////////
12 // ============================================================================
14 // ============================================================================
16 // ----------------------------------------------------------------------------
18 // ----------------------------------------------------------------------------
21 #pragma implementation "evtloop.h"
24 #include "wx/window.h"
26 #include "wx/evtloop.h"
27 #include "wx/tooltip.h"
29 #include "wx/x11/private.h"
32 // ----------------------------------------------------------------------------
34 // ----------------------------------------------------------------------------
36 class WXDLLEXPORT wxEventLoopImpl
40 wxEventLoopImpl() { SetExitCode(0); m_keepGoing
= FALSE
; }
43 void ProcessEvent(XEvent
* event
);
45 // generate an idle message, return TRUE if more idle time requested
48 // set/get the exit code
49 void SetExitCode(int exitcode
) { m_exitcode
= exitcode
; }
50 int GetExitCode() const { return m_exitcode
; }
53 // preprocess an event, return TRUE if processed (i.e. no further
54 // dispatching required)
55 bool PreProcessMessage(XEvent
* event
);
57 // the exit code of the event loop
63 // ============================================================================
64 // wxEventLoopImpl implementation
65 // ============================================================================
67 // ----------------------------------------------------------------------------
68 // wxEventLoopImpl message processing
69 // ----------------------------------------------------------------------------
71 void wxEventLoopImpl::ProcessEvent(XEvent
*event
)
73 // give us the chance to preprocess the message first
74 if ( !PreProcessEvent(event
) )
76 // if it wasn't done, dispatch it to the corresponding window
78 wxTheApp
->ProcessXEvent((WXEvent
*) event
);
82 bool wxEventLoopImpl::PreProcessEvent(XEvent
*event
)
86 HWND hWnd
= msg
->hwnd
;
87 wxWindow
*wndThis
= wxGetWindowFromHWND((WXHWND
)hWnd
);
90 // try translations first; find the youngest window with a translation
93 for ( wnd
= wndThis
; wnd
; wnd
= wnd
->GetParent() )
95 if ( wnd
->MSWTranslateMessage((WXMSG
*)msg
) )
99 // Anyone for a non-translation message? Try youngest descendants first.
100 for ( wnd
= wndThis
; wnd
; wnd
= wnd
->GetParent() )
102 if ( wnd
->MSWProcessMessage((WXMSG
*)msg
) )
110 // ----------------------------------------------------------------------------
111 // wxEventLoopImpl idle event processing
112 // ----------------------------------------------------------------------------
114 bool wxEventLoopImpl::SendIdleEvent()
117 event
.SetEventObject(wxTheApp
);
119 return wxTheApp
->ProcessEvent(event
) && event
.MoreRequested();
122 // ============================================================================
123 // wxEventLoop implementation
124 // ============================================================================
126 wxEventLoop
*wxEventLoop::ms_activeLoop
= NULL
;
128 // ----------------------------------------------------------------------------
129 // wxEventLoop running and exiting
130 // ----------------------------------------------------------------------------
132 wxEventLoop::~wxEventLoop()
134 wxASSERT_MSG( !m_impl
, _T("should have been deleted in Run()") );
137 bool wxEventLoop::IsRunning() const
139 return m_impl
!= NULL
;
142 int wxEventLoop::Run()
144 // event loops are not recursive, you need to create another loop!
145 wxCHECK_MSG( !IsRunning(), -1, _T("can't reenter a message loop") );
147 m_impl
= new wxEventLoopImpl
;
149 wxEventLoop
*oldLoop
= ms_activeLoop
;
150 ms_activeLoop
= this;
152 m_impl
->m_keepGoing
= TRUE
;
153 while ( m_impl
->m_keepGoing
)
156 wxMutexGuiLeaveOrEnter();
157 #endif // wxUSE_THREADS
159 // generate and process idle events for as long as we don't have
160 // anything else to do
161 while ( ! Pending() )
163 if (!m_impl
->SendIdleEvent())
166 // leave the main loop to give other threads a chance to
167 // perform their GUI work
172 // Break out of while loop
177 // a message came or no more idle processing to do, sit in Dispatch()
178 // waiting for the next message
185 int exitcode
= m_impl
->GetExitCode();
189 ms_activeLoop
= oldLoop
;
194 void wxEventLoop::Exit(int rc
)
196 wxCHECK_RET( IsRunning(), _T("can't call Exit() if not running") );
198 m_impl
->SetExitCode(rc
);
199 m_impl
->m_keepGoing
= FALSE
;
202 // ----------------------------------------------------------------------------
203 // wxEventLoop message processing dispatching
204 // ----------------------------------------------------------------------------
206 bool wxEventLoop::Pending() const
208 XFlush(wxGetDisplay());
209 return (XPending(wxGetDisplay()) > 0);
212 bool wxEventLoop::Dispatch()
216 // TODO allowing for threads, as per e.g. wxMSW
218 XNextEvent(wxGetDisplay(), & event
);
219 m_impl
->ProcessEvent(& event
);