]> git.saurik.com Git - wxWidgets.git/blame - src/unix/evtloopunix.cpp
Support using GetTextExtent() with empty string to get descent in wxOSX.
[wxWidgets.git] / src / unix / evtloopunix.cpp
CommitLineData
b46b1d59
VZ
1/////////////////////////////////////////////////////////////////////////////
2// Name: src/unix/evtloopunix.cpp
3// Purpose: wxEventLoop implementation
4// Author: Lukasz Michalski (lm@zork.pl)
5// Created: 2007-05-07
6// RCS-ID: $Id$
7// Copyright: (c) 2006 Zork Lukasz Michalski
821d856a
VZ
8// (c) 2009, 2013 Vadim Zeitlin <vadim@wxwidgets.org>
9// (c) 2013 Rob Bresalier
b46b1d59
VZ
10// Licence: wxWindows licence
11/////////////////////////////////////////////////////////////////////////////
12
13// ===========================================================================
14// declarations
15// ===========================================================================
16
17// ---------------------------------------------------------------------------
18// headers
19// ---------------------------------------------------------------------------
20
21// For compilers that support precompilation, includes "wx.h".
22#include "wx/wxprec.h"
23
a1873279 24#if wxUSE_CONSOLE_EVENTLOOP
b46b1d59 25
1e04d2bf
PC
26#include "wx/evtloop.h"
27
b46b1d59
VZ
28#ifndef WX_PRECOMP
29 #include "wx/app.h"
30 #include "wx/log.h"
31#endif
32
37ab9399 33#include "wx/apptrait.h"
5cd99866 34#include "wx/scopedptr.h"
b46b1d59
VZ
35#include "wx/thread.h"
36#include "wx/module.h"
ba752031 37#include "wx/unix/private/timer.h"
b46b1d59 38#include "wx/unix/private/epolldispatcher.h"
2ccfebab 39#include "wx/unix/private/wakeuppipe.h"
b46b1d59 40#include "wx/private/selectdispatcher.h"
71e9885b 41#include "wx/private/eventloopsourcesmanager.h"
5e382463 42#include "wx/private/fdioeventloopsourcehandler.h"
71e9885b 43#include "wx/private/eventloopsourcesmanager.h"
b46b1d59 44
5cd99866
VZ
45#if wxUSE_EVENTLOOP_SOURCE
46 #include "wx/evtloopsrc.h"
47#endif // wxUSE_EVENTLOOP_SOURCE
48
b46b1d59
VZ
49// ===========================================================================
50// wxEventLoop implementation
51// ===========================================================================
52
53//-----------------------------------------------------------------------------
54// initialization
55//-----------------------------------------------------------------------------
56
57wxConsoleEventLoop::wxConsoleEventLoop()
58{
39bc0168
VZ
59 // Be pessimistic initially and assume that we failed to initialize.
60 m_dispatcher = NULL;
61 m_wakeupPipe = NULL;
62 m_wakeupSource = NULL;
63
64 // Create the pipe.
65 wxScopedPtr<wxWakeUpPipeMT> wakeupPipe(new wxWakeUpPipeMT);
66 const int pipeFD = wakeupPipe->GetReadFd();
2ccfebab 67 if ( pipeFD == wxPipe::INVALID_FD )
b46b1d59 68 return;
b46b1d59 69
39bc0168
VZ
70 // And start monitoring it in our event loop.
71 m_wakeupSource = wxEventLoopBase::AddSourceForFD
72 (
73 pipeFD,
74 wakeupPipe.get(),
75 wxFDIO_INPUT
76 );
77
78 if ( !m_wakeupSource )
5e1eac14 79 return;
b46b1d59 80
39bc0168
VZ
81 // This is a bit ugly but we know that AddSourceForFD() used the currently
82 // active dispatcher to register this source, so use the same one for our
83 // other operations. Of course, currently the dispatcher returned by
84 // wxFDIODispatcher::Get() is always the same one anyhow so it doesn't
85 // really matter, but if we started returning different things later, it
86 // would.
87 m_dispatcher = wxFDIODispatcher::Get();
88
89 m_wakeupPipe = wakeupPipe.release();
3ef595d5 90}
b46b1d59 91
3f8cdda4
VS
92wxConsoleEventLoop::~wxConsoleEventLoop()
93{
c8299fa8
VZ
94 if ( m_wakeupPipe )
95 {
39bc0168 96 delete m_wakeupSource;
c8299fa8
VZ
97
98 delete m_wakeupPipe;
99 }
3f8cdda4
VS
100}
101
6b8ef0b3
VZ
102//-----------------------------------------------------------------------------
103// adding & removing sources
104//-----------------------------------------------------------------------------
105
106#if wxUSE_EVENTLOOP_SOURCE
107
71e9885b 108class wxConsoleEventLoopSourcesManager : public wxEventLoopSourcesManagerBase
6b8ef0b3 109{
71e9885b
VZ
110public:
111 wxEventLoopSource* AddSourceForFD( int fd,
112 wxEventLoopSourceHandler *handler,
113 int flags)
114 {
115 wxCHECK_MSG( fd != -1, NULL, "can't monitor invalid fd" );
6b8ef0b3 116
71e9885b
VZ
117 wxLogTrace(wxTRACE_EVT_SOURCE,
118 "Adding event loop source for fd=%d", fd);
119
120 // we need a bridge to wxFDIODispatcher
121 //
122 // TODO: refactor the code so that only wxEventLoopSourceHandler is used
123 wxScopedPtr<wxFDIOHandler>
124 fdioHandler(new wxFDIOEventLoopSourceHandler(handler));
6b8ef0b3 125
71e9885b
VZ
126 if ( !wxFDIODispatcher::Get()->RegisterFD(fd, fdioHandler.get(), flags) )
127 return NULL;
6b8ef0b3 128
71e9885b
VZ
129 return new wxUnixEventLoopSource(wxFDIODispatcher::Get(), fdioHandler.release(),
130 fd, handler, flags);
131 }
132};
133
134wxEventLoopSourcesManagerBase* wxAppTraits::GetEventLoopSourcesManager()
135{
136 static wxConsoleEventLoopSourcesManager s_eventLoopSourcesManager;
5cd99866 137
71e9885b 138 return &s_eventLoopSourcesManager;
6b8ef0b3
VZ
139}
140
5cd99866 141wxUnixEventLoopSource::~wxUnixEventLoopSource()
6b8ef0b3 142{
6b8ef0b3 143 wxLogTrace(wxTRACE_EVT_SOURCE,
5cd99866 144 "Removing event loop source for fd=%d", m_fd);
6b8ef0b3 145
5cd99866
VZ
146 m_dispatcher->UnregisterFD(m_fd);
147
148 delete m_fdioHandler;
6b8ef0b3 149}
5cd99866
VZ
150
151#endif // wxUSE_EVENTLOOP_SOURCE
6b8ef0b3 152
b46b1d59
VZ
153//-----------------------------------------------------------------------------
154// events dispatch and loop handling
155//-----------------------------------------------------------------------------
156
157bool wxConsoleEventLoop::Pending() const
158{
a12698ab
VZ
159 if ( m_dispatcher->HasPending() )
160 return true;
161
162#if wxUSE_TIMER
163 wxUsecClock_t nextTimer;
164 if ( wxTimerScheduler::Get().GetNext(&nextTimer) &&
165 !wxMilliClockToLong(nextTimer) )
166 return true;
167#endif // wxUSE_TIMER
168
169 return false;
b46b1d59
VZ
170}
171
172bool wxConsoleEventLoop::Dispatch()
b46b1d59 173{
c22ace4d
VZ
174 DispatchTimeout(static_cast<unsigned long>(
175 wxFDIODispatcher::TIMEOUT_INFINITE));
b46b1d59 176
9af42efd
VZ
177 return true;
178}
179
180int wxConsoleEventLoop::DispatchTimeout(unsigned long timeout)
181{
b46b1d59 182#if wxUSE_TIMER
9af42efd 183 // check if we need to decrease the timeout to account for a timer
b46b1d59
VZ
184 wxUsecClock_t nextTimer;
185 if ( wxTimerScheduler::Get().GetNext(&nextTimer) )
186 {
9af42efd
VZ
187 unsigned long timeUntilNextTimer = wxMilliClockToLong(nextTimer / 1000);
188 if ( timeUntilNextTimer < timeout )
189 timeout = timeUntilNextTimer;
b46b1d59 190 }
b46b1d59 191#endif // wxUSE_TIMER
b46b1d59 192
a12698ab 193 bool hadEvent = m_dispatcher->Dispatch(timeout) > 0;
b46b1d59
VZ
194
195#if wxUSE_TIMER
9af42efd
VZ
196 if ( wxTimerScheduler::Get().NotifyExpired() )
197 hadEvent = true;
198#endif // wxUSE_TIMER
b46b1d59 199
9af42efd 200 return hadEvent ? 1 : -1;
438febca
VZ
201}
202
203void wxConsoleEventLoop::WakeUp()
204{
3f8cdda4 205 m_wakeupPipe->WakeUp();
438febca
VZ
206}
207
208void wxConsoleEventLoop::OnNextIteration()
209{
b46b1d59
VZ
210 // call the signal handlers for any signals we caught recently
211 wxTheApp->CheckSignal();
212}
213
37ab9399
SN
214
215wxEventLoopBase *wxConsoleAppTraits::CreateEventLoop()
216{
217 return new wxEventLoop();
218}
219
a1873279 220#endif // wxUSE_CONSOLE_EVENTLOOP