]>
Commit | Line | Data |
---|---|---|
8000ae7f | 1 | /////////////////////////////////////////////////////////////////////////////// |
670f9935 | 2 | // Name: src/gtk1/evtloop.cpp |
8000ae7f VZ |
3 | // Purpose: implements wxEventLoop for GTK+ |
4 | // Author: Vadim Zeitlin | |
5 | // Modified by: | |
6 | // Created: 10.07.01 | |
8000ae7f | 7 | // Copyright: (c) 2001 Vadim Zeitlin <zeitlin@dptmaths.ens-cachan.fr> |
526954c5 | 8 | // Licence: wxWindows licence |
8000ae7f VZ |
9 | /////////////////////////////////////////////////////////////////////////////// |
10 | ||
11 | // ============================================================================ | |
12 | // declarations | |
13 | // ============================================================================ | |
14 | ||
15 | // ---------------------------------------------------------------------------- | |
16 | // headers | |
17 | // ---------------------------------------------------------------------------- | |
18 | ||
8000ae7f VZ |
19 | // For compilers that support precompilation, includes "wx.h". |
20 | #include "wx/wxprec.h" | |
21 | ||
22 | #ifdef __BORLANDC__ | |
23 | #pragma hdrstop | |
24 | #endif | |
25 | ||
8000ae7f | 26 | #include "wx/evtloop.h" |
670f9935 WS |
27 | |
28 | #ifndef WX_PRECOMP | |
29 | #include "wx/app.h" | |
6918257d | 30 | #include "wx/log.h" |
670f9935 | 31 | #endif // WX_PRECOMP |
8000ae7f VZ |
32 | |
33 | #include <gtk/gtk.h> | |
34 | ||
35 | // ---------------------------------------------------------------------------- | |
36 | // wxEventLoopImpl | |
37 | // ---------------------------------------------------------------------------- | |
38 | ||
39 | class WXDLLEXPORT wxEventLoopImpl | |
40 | { | |
41 | public: | |
42 | // ctor | |
43 | wxEventLoopImpl() { SetExitCode(0); } | |
44 | ||
45 | // set/get the exit code | |
46 | void SetExitCode(int exitcode) { m_exitcode = exitcode; } | |
47 | int GetExitCode() const { return m_exitcode; } | |
48 | ||
49 | private: | |
50 | // the exit code of the event loop | |
51 | int m_exitcode; | |
52 | }; | |
53 | ||
54 | // ============================================================================ | |
b46b1d59 | 55 | // wxGUIEventLoop implementation |
8000ae7f VZ |
56 | // ============================================================================ |
57 | ||
58 | // ---------------------------------------------------------------------------- | |
b46b1d59 | 59 | // wxGUIEventLoop running and exiting |
8000ae7f VZ |
60 | // ---------------------------------------------------------------------------- |
61 | ||
b46b1d59 | 62 | wxGUIEventLoop::~wxGUIEventLoop() |
8000ae7f | 63 | { |
9a83f860 | 64 | wxASSERT_MSG( !m_impl, wxT("should have been deleted in Run()") ); |
8000ae7f VZ |
65 | } |
66 | ||
c738d187 | 67 | int wxGUIEventLoop::DoRun() |
8000ae7f | 68 | { |
8000ae7f VZ |
69 | m_impl = new wxEventLoopImpl; |
70 | ||
d3ad22bd VZ |
71 | guint loopLevel = gtk_main_level(); |
72 | ||
73 | // This is placed inside of a loop to take into account nested | |
74 | // event loops. For example, inside this event loop, we may recieve | |
75 | // Exit() for a different event loop (which we are currently inside of) | |
76 | // That Exit() will cause this gtk_main() to exit so we need to re-enter it. | |
67afffcd JJ |
77 | #if 0 |
78 | // changed by JJ | |
79 | // this code was intended to support nested event loops. However, | |
80 | // exiting a dialog will result in a application hang (because | |
81 | // gtk_main_quit is called when closing the dialog????) | |
82 | // So for the moment this code is disabled and nested event loops | |
83 | // probably fail for wxGTK1 | |
84 | while ( !m_shouldExit ) | |
d3ad22bd | 85 | { |
67afffcd JJ |
86 | #endif |
87 | gtk_main(); | |
88 | #if 0 | |
d3ad22bd VZ |
89 | } |
90 | ||
91 | // Force the enclosing event loop to also exit to see if it is done | |
92 | // in case that event loop ended inside of this one. If it is not time | |
93 | // yet for that event loop to exit, it will be executed again due to | |
94 | // the while() loop on m_shouldExit(). | |
95 | // | |
96 | // This is unnecessary if we are the top level loop, i.e. loop of level 0. | |
97 | if ( loopLevel ) | |
98 | { | |
99 | gtk_main_quit(); | |
100 | } | |
67afffcd | 101 | #endif |
8000ae7f | 102 | |
16d17da6 VZ |
103 | OnExit(); |
104 | ||
8000ae7f | 105 | int exitcode = m_impl->GetExitCode(); |
5276b0a5 | 106 | wxDELETE(m_impl); |
8000ae7f VZ |
107 | |
108 | return exitcode; | |
109 | } | |
110 | ||
d3ad22bd | 111 | void wxGUIEventLoop::ScheduleExit(int rc) |
8000ae7f | 112 | { |
d3ad22bd | 113 | wxCHECK_RET( IsInsideRun(), wxT("can't call ScheduleExit() if not started") ); |
8000ae7f VZ |
114 | |
115 | m_impl->SetExitCode(rc); | |
116 | ||
d3ad22bd VZ |
117 | m_shouldExit = true; |
118 | ||
8000ae7f VZ |
119 | gtk_main_quit(); |
120 | } | |
121 | ||
122 | // ---------------------------------------------------------------------------- | |
123 | // wxEventLoop message processing dispatching | |
124 | // ---------------------------------------------------------------------------- | |
125 | ||
b46b1d59 | 126 | bool wxGUIEventLoop::Pending() const |
8000ae7f | 127 | { |
c263eb03 | 128 | if (wxTheApp) |
9213ca5d RD |
129 | { |
130 | // We need to remove idle callbacks or gtk_events_pending will | |
131 | // never return false. | |
fe593cc5 | 132 | wxTheApp->RemoveIdleTag(); |
9213ca5d RD |
133 | } |
134 | ||
135 | return gtk_events_pending(); | |
8000ae7f VZ |
136 | } |
137 | ||
b46b1d59 | 138 | bool wxGUIEventLoop::Dispatch() |
8000ae7f | 139 | { |
9a83f860 | 140 | wxCHECK_MSG( IsRunning(), false, wxT("can't call Dispatch() if not running") ); |
8000ae7f VZ |
141 | |
142 | gtk_main_iteration(); | |
143 | ||
670f9935 | 144 | return true; |
8000ae7f | 145 | } |
dde19c21 FM |
146 | |
147 | //----------------------------------------------------------------------------- | |
148 | // wxYield | |
149 | //----------------------------------------------------------------------------- | |
150 | ||
151 | bool wxGUIEventLoop::YieldFor(long eventsToProcess) | |
152 | { | |
153 | #if wxUSE_THREADS | |
154 | if ( !wxThread::IsMain() ) | |
155 | { | |
156 | // can't call gtk_main_iteration() from other threads like this | |
157 | return true; | |
158 | } | |
159 | #endif // wxUSE_THREADS | |
160 | ||
161 | m_isInsideYield = true; | |
162 | m_eventsToProcessInsideYield = eventsToProcess; | |
163 | ||
164 | // We need to remove idle callbacks or the loop will | |
165 | // never finish. | |
166 | wxTheApp->RemoveIdleTag(); | |
167 | ||
168 | #if wxUSE_LOG | |
169 | // disable log flushing from here because a call to wxYield() shouldn't | |
170 | // normally result in message boxes popping up &c | |
171 | wxLog::Suspend(); | |
172 | #endif | |
173 | ||
174 | // TODO: implement event filtering using the eventsToProcess mask | |
175 | while (gtk_events_pending()) | |
176 | gtk_main_iteration(); | |
177 | ||
178 | // It's necessary to call ProcessIdle() to update the frames sizes which | |
179 | // might have been changed (it also will update other things set from | |
180 | // OnUpdateUI() which is a nice (and desired) side effect). But we | |
181 | // call ProcessIdle() only once since this is not meant for longish | |
182 | // background jobs (controlled by wxIdleEvent::RequestMore() and the | |
183 | // return value of Processidle(). | |
184 | ProcessIdle(); | |
185 | ||
186 | #if wxUSE_LOG | |
187 | // let the logs be flashed again | |
188 | wxLog::Resume(); | |
189 | #endif | |
190 | ||
191 | m_isInsideYield = false; | |
192 | ||
193 | return true; | |
194 | } | |
67afffcd JJ |
195 | |
196 | class wxGUIEventLoopSourcesManager : public wxEventLoopSourcesManagerBase | |
197 | { | |
198 | public: | |
199 | wxEventLoopSource * | |
200 | AddSourceForFD(int WXUNUSED(fd), | |
201 | wxEventLoopSourceHandler* WXUNUSED(handler), | |
202 | int WXUNUSED(flags)) | |
203 | { | |
204 | wxFAIL_MSG("Monitoring FDs in the main loop is not implemented in wxGTK1"); | |
205 | ||
206 | return NULL; | |
207 | } | |
208 | }; | |
209 | ||
210 | wxEventLoopSourcesManagerBase* wxGUIAppTraits::GetEventLoopSourcesManager() | |
211 | { | |
212 | static wxGUIEventLoopSourcesManager s_eventLoopSourcesManager; | |
213 | ||
214 | return &s_eventLoopSourcesManager; | |
215 | } |