]> git.saurik.com Git - wxWidgets.git/blame_incremental - src/msw/timer.cpp
using on all OSX variants
[wxWidgets.git] / src / msw / timer.cpp
... / ...
CommitLineData
1/////////////////////////////////////////////////////////////////////////////
2// Name: src/msw/timer.cpp
3// Purpose: wxTimer implementation
4// Author: Julian Smart
5// Modified by: Vadim Zeitlin (use hash map instead of list, global rewrite)
6// Created: 04/01/98
7// RCS-ID: $Id$
8// Copyright: (c) Julian Smart
9// Licence: wxWindows licence
10/////////////////////////////////////////////////////////////////////////////
11
12// For compilers that support precompilation, includes "wx.h".
13#include "wx/wxprec.h"
14
15#ifdef __BORLANDC__
16 #pragma hdrstop
17#endif
18
19#if wxUSE_TIMER
20
21#include "wx/msw/private/timer.h"
22
23#ifndef WX_PRECOMP
24 #include "wx/list.h"
25 #include "wx/event.h"
26 #include "wx/app.h"
27 #include "wx/intl.h"
28 #include "wx/log.h"
29 #include "wx/hashmap.h"
30 #include "wx/module.h"
31#endif
32
33#include "wx/msw/private.h"
34
35// ----------------------------------------------------------------------------
36// private globals
37// ----------------------------------------------------------------------------
38
39// define a hash containing all the timers: it is indexed by timer id and
40// contains the corresponding timer
41WX_DECLARE_HASH_MAP(WPARAM, wxMSWTimerImpl *, wxIntegerHash, wxIntegerEqual,
42 wxTimerMap);
43
44// instead of using a global here, wrap it in a static function as otherwise it
45// could have been used before being initialized if a timer object were created
46// globally
47static wxTimerMap& TimerMap()
48{
49 static wxTimerMap s_timerMap;
50
51 return s_timerMap;
52}
53
54// ----------------------------------------------------------------------------
55// private functions
56// ----------------------------------------------------------------------------
57
58LRESULT APIENTRY _EXPORT wxTimerWndProc(HWND hWnd, UINT message,
59 WPARAM wParam, LPARAM lParam);
60
61// implemented in utils.cpp
62extern "C" WXDLLIMPEXP_BASE HWND
63wxCreateHiddenWindow(LPCTSTR *pclassname, LPCTSTR classname, WNDPROC wndproc);
64
65
66// ----------------------------------------------------------------------------
67// wxTimerHiddenWindowModule: used to manage the hidden window used for
68// catching timer messages (we need a module to ensure that the window is
69// always deleted)
70// ----------------------------------------------------------------------------
71
72class wxTimerHiddenWindowModule : public wxModule
73{
74public:
75 // module init/finalize
76 virtual bool OnInit();
77 virtual void OnExit();
78
79 // get the hidden window (creates on demand)
80 static HWND GetHWND();
81
82private:
83 // the HWND of the hidden window
84 static HWND ms_hwnd;
85
86 // the class used to create it
87 static const wxChar *ms_className;
88
89 DECLARE_DYNAMIC_CLASS(wxTimerHiddenWindowModule)
90};
91
92IMPLEMENT_DYNAMIC_CLASS(wxTimerHiddenWindowModule, wxModule)
93
94// ============================================================================
95// implementation
96// ============================================================================
97
98
99// ----------------------------------------------------------------------------
100// wxMSWTimerImpl class
101// ----------------------------------------------------------------------------
102
103bool wxMSWTimerImpl::Start(int milliseconds, bool oneShot)
104{
105 if ( !wxTimerImpl::Start(milliseconds, oneShot) )
106 return false;
107
108 m_id = ::SetTimer(
109 wxTimerHiddenWindowModule::GetHWND(), // window to send the messages to
110 GetId(), // timer ID
111 (UINT)m_milli, // delay
112 NULL // timer proc. Not used since we pass hwnd
113 );
114
115 if ( !m_id )
116 {
117 wxLogSysError(_("Couldn't create a timer"));
118
119 return false;
120 }
121
122 // check that SetTimer() didn't reuse an existing id: according to the MSDN
123 // this can happen and this would be catastrophic to us as we rely on ids
124 // uniquely identifying the timers because we use them as keys in the hash
125 //
126 // notice that this also happens if the same id is reused for multiple
127 // timers: this used to work in previous versions but was never supported
128 // and absolutely shouldn't be done, use wxID_ANY to assign an id to the
129 // timer automatically or ensure that all your timers have unique ids
130 if ( TimerMap().find(m_id) != TimerMap().end() )
131 {
132 wxLogError(_("Timer creation failed."));
133
134 ::KillTimer(wxTimerHiddenWindowModule::GetHWND(), m_id);
135 m_id = 0;
136
137 return false;
138 }
139
140 TimerMap()[m_id] = this;
141
142 return true;
143}
144
145void wxMSWTimerImpl::Stop()
146{
147 wxASSERT_MSG( m_id, wxT("should be running") );
148
149 ::KillTimer(wxTimerHiddenWindowModule::GetHWND(), m_id);
150
151 TimerMap().erase(m_id);
152
153 m_id = 0;
154}
155
156// ----------------------------------------------------------------------------
157// private functions
158// ----------------------------------------------------------------------------
159
160void wxProcessTimer(wxMSWTimerImpl& timer)
161{
162 wxASSERT_MSG( timer.IsRunning(), wxT("bogus timer id") );
163
164 if ( timer.IsOneShot() )
165 timer.Stop();
166
167 timer.Notify();
168}
169
170
171LRESULT APIENTRY _EXPORT wxTimerWndProc(HWND hWnd, UINT message,
172 WPARAM wParam, LPARAM lParam)
173{
174 if ( message == WM_TIMER )
175 {
176 wxTimerMap::iterator node = TimerMap().find(wParam);
177
178 wxCHECK_MSG( node != TimerMap().end(), 0, wxT("bogus timer id in wxTimerProc") );
179
180 wxProcessTimer(*(node->second));
181 }
182 else
183 {
184 return ::DefWindowProc(hWnd, message, wParam, lParam);
185 }
186 return 0;
187}
188
189// ----------------------------------------------------------------------------
190// wxTimerHiddenWindowModule functions
191// ----------------------------------------------------------------------------
192
193
194HWND wxTimerHiddenWindowModule::ms_hwnd = NULL;
195
196const wxChar *wxTimerHiddenWindowModule::ms_className = NULL;
197
198bool wxTimerHiddenWindowModule::OnInit()
199{
200 // do not initialize ms_hwnd to ms_className to NULL here: it may happen
201 // that our GetHWND() is called before the modules are initialized if a
202 // timer is created from wxApp-derived class ctor and in this case we
203 // shouldn't overwrite it
204
205 return true;
206}
207
208void wxTimerHiddenWindowModule::OnExit()
209{
210 if ( ms_hwnd )
211 {
212 if ( !::DestroyWindow(ms_hwnd) )
213 {
214 wxLogLastError(wxT("DestroyWindow(wxTimerHiddenWindow)"));
215 }
216
217 ms_hwnd = NULL;
218 }
219
220 if ( ms_className )
221 {
222 if ( !::UnregisterClass(ms_className, wxGetInstance()) )
223 {
224 wxLogLastError(wxT("UnregisterClass(\"wxTimerHiddenWindow\")"));
225 }
226
227 ms_className = NULL;
228 }
229}
230
231/* static */
232HWND wxTimerHiddenWindowModule::GetHWND()
233{
234 static const wxChar *HIDDEN_WINDOW_CLASS = wxT("wxTimerHiddenWindow");
235 if ( !ms_hwnd )
236 {
237 ms_hwnd = wxCreateHiddenWindow(&ms_className, HIDDEN_WINDOW_CLASS,
238 wxTimerWndProc);
239 }
240
241 return ms_hwnd;
242}
243
244#endif // wxUSE_TIMER