// Author: Julian Smart
// Modified by: Vadim Zeitlin (use hash map instead of list, global rewrite)
// Created: 04/01/98
-// RCS-ID: $Id$
// Copyright: (c) Julian Smart
// Licence: wxWindows licence
/////////////////////////////////////////////////////////////////////////////
#endif
#include "wx/msw/private.h"
+#include "wx/msw/private/hiddenwin.h"
// ----------------------------------------------------------------------------
// private globals
// define a hash containing all the timers: it is indexed by timer id and
// contains the corresponding timer
-WX_DECLARE_HASH_MAP(unsigned long, wxMSWTimerImpl *, wxIntegerHash, wxIntegerEqual,
+WX_DECLARE_HASH_MAP(WPARAM, wxMSWTimerImpl *, wxIntegerHash, wxIntegerEqual,
wxTimerMap);
// instead of using a global here, wrap it in a static function as otherwise it
return s_timerMap;
}
+// This gets a unique, non-zero timer ID and creates an entry in the TimerMap
+UINT_PTR GetNewTimerId(wxMSWTimerImpl *t)
+{
+ static UINT_PTR lastTimerId = 0;
+
+ while (lastTimerId == 0 ||
+ TimerMap().find(lastTimerId) != TimerMap().end())
+ {
+ lastTimerId = lastTimerId + 1;
+ }
+
+ TimerMap()[lastTimerId] = t;
+
+ return lastTimerId;
+}
+
+
+
// ----------------------------------------------------------------------------
// private functions
// ----------------------------------------------------------------------------
LRESULT APIENTRY _EXPORT wxTimerWndProc(HWND hWnd, UINT message,
WPARAM wParam, LPARAM lParam);
-// implemented in utils.cpp
-extern "C" WXDLLIMPEXP_BASE HWND
-wxCreateHiddenWindow(LPCTSTR *pclassname, LPCTSTR classname, WNDPROC wndproc);
-
-
// ----------------------------------------------------------------------------
// wxTimerHiddenWindowModule: used to manage the hidden window used for
// catching timer messages (we need a module to ensure that the window is
if ( !wxTimerImpl::Start(milliseconds, oneShot) )
return false;
- m_id = ::SetTimer(
- wxTimerHiddenWindowModule::GetHWND(), // window to send the messages to
- GetId(), // timer ID
- (UINT)m_milli, // delay
- NULL // timer proc. Not used since we pass hwnd
- );
-
- if ( !m_id )
+ m_id = GetNewTimerId(this);
+ // SetTimer() normally returns just idTimer but this might change in the
+ // future so use its return value to be safe
+ UINT_PTR ret = ::SetTimer
+ (
+ wxTimerHiddenWindowModule::GetHWND(), // window for WM_TIMER
+ m_id, // timer ID to create
+ (UINT)m_milli, // delay
+ NULL // timer proc (unused)
+ );
+
+ if ( ret == 0 )
{
wxLogSysError(_("Couldn't create a timer"));
return false;
}
- // check that SetTimer() didn't reuse an existing id: according to the MSDN
- // this can happen and this would be catastrophic to us as we rely on ids
- // uniquely identifying the timers because we use them as keys in the hash
- if ( TimerMap().find(m_id) != TimerMap().end() )
- {
- wxLogError(_("Timer creation failed."));
-
- ::KillTimer(wxTimerHiddenWindowModule::GetHWND(), m_id);
- m_id = 0;
-
- return false;
- }
-
- TimerMap()[m_id] = this;
-
return true;
}
void wxMSWTimerImpl::Stop()
{
- wxASSERT_MSG( m_id, _T("should be running") );
-
::KillTimer(wxTimerHiddenWindowModule::GetHWND(), m_id);
-
TimerMap().erase(m_id);
-
m_id = 0;
}
void wxProcessTimer(wxMSWTimerImpl& timer)
{
- wxASSERT_MSG( timer.IsRunning(), _T("bogus timer id") );
+ wxASSERT_MSG( timer.IsRunning(), wxT("bogus timer id") );
if ( timer.IsOneShot() )
timer.Stop();
{
if ( message == WM_TIMER )
{
- wxTimerMap::iterator node = TimerMap().find((unsigned long)wParam);
+ wxTimerMap::iterator node = TimerMap().find(wParam);
- wxCHECK_MSG( node != TimerMap().end(), 0, wxT("bogus timer id in wxTimerProc") );
+ if ( node != TimerMap().end() )
+ {
+ wxProcessTimer(*(node->second));
- wxProcessTimer(*(node->second));
- }
- else
- {
- return ::DefWindowProc(hWnd, message, wParam, lParam);
+ return 0;
+ }
+ //else: Unknown timer, probably one of our timers that had fired just
+ // before being removed from the timers map by Stop().
}
- return 0;
+
+ return ::DefWindowProc(hWnd, message, wParam, lParam);
}
// ----------------------------------------------------------------------------
bool wxTimerHiddenWindowModule::OnInit()
{
- ms_hwnd = NULL;
- ms_className = NULL;
+ // do not initialize ms_hwnd to ms_className to NULL here: it may happen
+ // that our GetHWND() is called before the modules are initialized if a
+ // timer is created from wxApp-derived class ctor and in this case we
+ // shouldn't overwrite it
return true;
}
{
if ( !::DestroyWindow(ms_hwnd) )
{
- wxLogLastError(_T("DestroyWindow(wxTimerHiddenWindow)"));
+ wxLogLastError(wxT("DestroyWindow(wxTimerHiddenWindow)"));
}
ms_hwnd = NULL;
{
if ( !::UnregisterClass(ms_className, wxGetInstance()) )
{
- wxLogLastError(_T("UnregisterClass(\"wxTimerHiddenWindow\")"));
+ wxLogLastError(wxT("UnregisterClass(\"wxTimerHiddenWindow\")"));
}
ms_className = NULL;
/* static */
HWND wxTimerHiddenWindowModule::GetHWND()
{
- static const wxChar *HIDDEN_WINDOW_CLASS = _T("wxTimerHiddenWindow");
+ static const wxChar *HIDDEN_WINDOW_CLASS = wxT("wxTimerHiddenWindow");
if ( !ms_hwnd )
{
ms_hwnd = wxCreateHiddenWindow(&ms_className, HIDDEN_WINDOW_CLASS,