]>
Commit | Line | Data |
---|---|---|
1 | ///////////////////////////////////////////////////////////////////////////// | |
2 | // Name: timer.cpp | |
3 | // Purpose: wxTimer implementation | |
4 | // Author: David Webster | |
5 | // Modified by: | |
6 | // Created: 10/17/99 | |
7 | // RCS-ID: $Id$ | |
8 | // Copyright: (c) David Webster | |
9 | // Licence: wxWindows licence | |
10 | ///////////////////////////////////////////////////////////////////////////// | |
11 | ||
12 | #ifdef __GNUG__ | |
13 | #pragma implementation "timer.h" | |
14 | #endif | |
15 | ||
16 | // For compilers that support precompilation, includes "wx.h". | |
17 | #include "wx/wxprec.h" | |
18 | ||
19 | #include "wx/window.h" | |
20 | #include "wx/os2/private.h" | |
21 | ||
22 | #ifndef WX_PRECOMP | |
23 | #include "wx/setup.h" | |
24 | #include "wx/list.h" | |
25 | #include "wx/event.h" | |
26 | #include "wx/app.h" | |
27 | #endif | |
28 | ||
29 | #include "wx/intl.h" | |
30 | #include "wx/log.h" | |
31 | ||
32 | #include "wx/timer.h" | |
33 | ||
34 | #include <time.h> | |
35 | #include <sys/types.h> | |
36 | ||
37 | #include <sys/timeb.h> | |
38 | ||
39 | // ---------------------------------------------------------------------------- | |
40 | // private globals | |
41 | // ---------------------------------------------------------------------------- | |
42 | ||
43 | // define a hash containing all the timers: it is indexed by timer id and | |
44 | // contains the corresponding timer | |
45 | WX_DECLARE_HASH_MAP(unsigned long, wxTimer *, wxIntegerHash, wxIntegerEqual, | |
46 | wxTimerMap); | |
47 | ||
48 | // instead of using a global here, wrap it in a static function as otherwise it | |
49 | // could have been used before being initialized if a timer object were created | |
50 | // globally | |
51 | static wxTimerMap& TimerMap() | |
52 | { | |
53 | static wxTimerMap s_timerMap; | |
54 | ||
55 | return s_timerMap; | |
56 | } | |
57 | ||
58 | // ---------------------------------------------------------------------------- | |
59 | // private functions | |
60 | // ---------------------------------------------------------------------------- | |
61 | ||
62 | // timer callback used for all timers | |
63 | ULONG wxTimerProc(HWND hwnd, ULONG, int nIdTimer, ULONG); | |
64 | ||
65 | // ---------------------------------------------------------------------------- | |
66 | // macros | |
67 | // ---------------------------------------------------------------------------- | |
68 | ||
69 | IMPLEMENT_ABSTRACT_CLASS(wxTimer, wxEvtHandler) | |
70 | ||
71 | // ============================================================================ | |
72 | // implementation | |
73 | // ============================================================================ | |
74 | ||
75 | // ---------------------------------------------------------------------------- | |
76 | // wxTimer class | |
77 | // ---------------------------------------------------------------------------- | |
78 | ||
79 | void wxTimer::Init() | |
80 | { | |
81 | m_ulId = 0; | |
82 | } | |
83 | ||
84 | wxTimer::~wxTimer() | |
85 | { | |
86 | wxTimer::Stop(); | |
87 | } | |
88 | ||
89 | void wxTimer::Notify() | |
90 | { | |
91 | // | |
92 | // The base class version generates an event if it has owner - which it | |
93 | // should because otherwise nobody can process timer events, but it does | |
94 | // not use the OS's ID, which OS/2 must have to figure out which timer fired | |
95 | // | |
96 | wxCHECK_RET( m_owner, _T("wxTimer::Notify() should be overridden.") ); | |
97 | ||
98 | wxTimerEvent vEvent( m_idTimer | |
99 | ,m_milli | |
100 | ); | |
101 | ||
102 | (void)m_owner->ProcessEvent(vEvent); | |
103 | } // end of wxTimer::Notify | |
104 | ||
105 | bool wxTimer::Start( | |
106 | int nMilliseconds | |
107 | , bool bOneShot | |
108 | ) | |
109 | { | |
110 | (void)wxTimerBase::Start( nMilliseconds | |
111 | ,bOneShot | |
112 | ); | |
113 | ||
114 | wxCHECK_MSG( m_milli > 0L, FALSE, wxT("invalid value for timer") ); | |
115 | ||
116 | wxWindow* pWin = NULL; | |
117 | ||
118 | if (m_owner) | |
119 | { | |
120 | pWin = (wxWindow*)m_owner; | |
121 | m_ulId = ::WinStartTimer( m_Hab | |
122 | ,pWin->GetHWND() | |
123 | ,m_idTimer | |
124 | ,(ULONG)nMilliseconds | |
125 | ); | |
126 | } | |
127 | else | |
128 | m_ulId = ::WinStartTimer( m_Hab | |
129 | ,NULLHANDLE | |
130 | ,0 | |
131 | ,(ULONG)nMilliseconds | |
132 | ); | |
133 | if (m_ulId > 0L) | |
134 | { | |
135 | // check that SetTimer() didn't reuse an existing id: according to | |
136 | // the MSDN this can happen and this would be catastrophic to us as | |
137 | // we rely on ids uniquely identifying the timers because we use | |
138 | // them as keys in the hash | |
139 | if ( TimerMap().find(m_ulId) != TimerMap().end() ) | |
140 | { | |
141 | wxLogError(_("Timer creation failed.")); | |
142 | ||
143 | ::WinStopTimer(m_Hab, pWin?(pWin->GetHWND()):NULL, m_ulId); | |
144 | m_ulId = 0; | |
145 | ||
146 | return false; | |
147 | } | |
148 | ||
149 | TimerMap()[m_ulId] = this; | |
150 | ||
151 | return true; | |
152 | } | |
153 | else | |
154 | { | |
155 | wxLogSysError(_("Couldn't create a timer")); | |
156 | ||
157 | return(FALSE); | |
158 | } | |
159 | } | |
160 | ||
161 | void wxTimer::Stop() | |
162 | { | |
163 | if ( m_ulId ) | |
164 | { | |
165 | if (m_owner) | |
166 | { | |
167 | wxWindow* pWin = (wxWindow*)m_owner; | |
168 | ||
169 | ::WinStopTimer(m_Hab, pWin->GetHWND(), m_ulId); | |
170 | } | |
171 | else | |
172 | ::WinStopTimer(m_Hab, NULLHANDLE, m_ulId); | |
173 | ||
174 | TimerMap().erase(m_ulId); | |
175 | } | |
176 | m_ulId = 0L; | |
177 | } | |
178 | ||
179 | // ---------------------------------------------------------------------------- | |
180 | // private functions | |
181 | // ---------------------------------------------------------------------------- | |
182 | ||
183 | void wxProcessTimer( | |
184 | wxTimer& rTimer | |
185 | ) | |
186 | { | |
187 | // | |
188 | // Avoid to process spurious timer events | |
189 | // | |
190 | if (rTimer.m_ulId == 0L) | |
191 | return; | |
192 | ||
193 | if (rTimer.IsOneShot()) | |
194 | rTimer.Stop(); | |
195 | ||
196 | rTimer.Notify(); | |
197 | } | |
198 | ||
199 | ULONG wxTimerProc( | |
200 | HWND WXUNUSED(hwnd) | |
201 | , ULONG | |
202 | , int nIdTimer | |
203 | , ULONG | |
204 | ) | |
205 | { | |
206 | wxTimerMap::iterator node = TimerMap().find((ULONG)nIdTimer); | |
207 | ||
208 | wxCHECK_MSG(node != TimerMap().end(), 0, | |
209 | wxT("bogus timer id in wxTimerProc") ); | |
210 | wxProcessTimer(*(node->second)); | |
211 | return 0; | |
212 | } | |
213 |