X-Git-Url: https://git.saurik.com/wxWidgets.git/blobdiff_plain/1b0fb34be895a9596131233edd1bf68b10c052b4..cd5e9298159e58f57e05f3b76c9d4a45e1eefe12:/src/x11/evtloop.cpp diff --git a/src/x11/evtloop.cpp b/src/x11/evtloop.cpp index da6b8f6d82..15e344b6f0 100644 --- a/src/x11/evtloop.cpp +++ b/src/x11/evtloop.cpp @@ -25,10 +25,16 @@ #include "wx/app.h" #include "wx/evtloop.h" #include "wx/tooltip.h" - +#if wxUSE_THREADS +#include "wx/thread.h" +#endif +#include "wx/timer.h" #include "wx/x11/private.h" #include "X11/Xlib.h" +#include +#include + // ---------------------------------------------------------------------------- // wxEventLoopImpl // ---------------------------------------------------------------------------- @@ -39,8 +45,8 @@ public: // ctor wxEventLoopImpl() { SetExitCode(0); m_keepGoing = FALSE; } - // process an XEvent - void ProcessEvent(XEvent* event); + // process an XEvent, return TRUE if it was processed + bool ProcessEvent(XEvent* event); // generate an idle message, return TRUE if more idle time requested bool SendIdleEvent(); @@ -52,7 +58,7 @@ public: public: // preprocess an event, return TRUE if processed (i.e. no further // dispatching required) - bool PreProcessMessage(XEvent* event); + bool PreProcessEvent(XEvent* event); // the exit code of the event loop int m_exitcode; @@ -68,15 +74,17 @@ public: // wxEventLoopImpl message processing // ---------------------------------------------------------------------------- -void wxEventLoopImpl::ProcessEvent(XEvent *event) +bool wxEventLoopImpl::ProcessEvent(XEvent *event) { // give us the chance to preprocess the message first - if ( !PreProcessEvent(event) ) - { - // if it wasn't done, dispatch it to the corresponding window - if (wxTheApp) - wxTheApp->ProcessXEvent((WXEvent*) event); - } + if ( PreProcessEvent(event) ) + return TRUE; + + // if it wasn't done, dispatch it to the corresponding window + if (wxTheApp) + return wxTheApp->ProcessXEvent((WXEvent*) event); + + return FALSE; } bool wxEventLoopImpl::PreProcessEvent(XEvent *event) @@ -152,14 +160,17 @@ int wxEventLoop::Run() m_impl->m_keepGoing = TRUE; while ( m_impl->m_keepGoing ) { -#if wxUSE_THREADS - wxMutexGuiLeaveOrEnter(); +#if 0 // wxUSE_THREADS + wxMutexGuiLeaveOrEnter(); #endif // wxUSE_THREADS // generate and process idle events for as long as we don't have // anything else to do while ( ! Pending() ) { +#if wxUSE_TIMER + wxTimer::NotifyTimers(); // TODO: is this the correct place for it? +#endif if (!m_impl->SendIdleEvent()) { #if wxUSE_THREADS @@ -205,8 +216,8 @@ void wxEventLoop::Exit(int rc) bool wxEventLoop::Pending() const { - XFlush(wxGetDisplay()); - return (XPending(wxGetDisplay()) > 0); + XFlush((Display*) wxGetDisplay()); + return (XPending((Display*) wxGetDisplay()) > 0); } bool wxEventLoop::Dispatch() @@ -215,8 +226,53 @@ bool wxEventLoop::Dispatch() // TODO allowing for threads, as per e.g. wxMSW - XNextEvent(wxGetDisplay(), & event); - m_impl->ProcessEvent(& event); +#if 0 + XNextEvent((Display*) wxGetDisplay(), & event); +#endif + + // This now waits until either an X event is received, + // or the select times out. So we should now process + // wxTimers in a reasonably timely fashion. However it + // does also mean that idle processing will happen more + // often, so we should probably limit idle processing to + // not be repeated more than every N milliseconds. + + if (XPending((Display*) wxGetDisplay()) == 0) + { +#if wxUSE_NANOX + GR_TIMEOUT timeout = 10; // Milliseconds + // Wait for next event, or timeout + GrGetNextEventTimeout(& event, timeout); + + // Fall through to ProcessEvent. + // we'll assume that ProcessEvent will just ignore + // the event if there was a timeout and no event. + +#else + struct timeval tv; + tv.tv_sec=0; + tv.tv_usec=10000; // TODO make this configurable + int fd = ConnectionNumber((Display*) wxGetDisplay()); + fd_set readset; + FD_ZERO(&readset); + FD_SET(fd, &readset); + if (select(fd+1, &readset, NULL, NULL, & tv) == 0) + { + // Timed out, so no event to process + return TRUE; + } + else + { + // An event was pending, so get it + XNextEvent((Display*) wxGetDisplay(), & event); + } +#endif + } else + { + XNextEvent((Display*) wxGetDisplay(), & event); + } + + (void) m_impl->ProcessEvent(& event); return TRUE; }