1 ///////////////////////////////////////////////////////////////////////////// 
   2 // Name:        src/msw/app.cpp 
   4 // Author:      Julian Smart 
   8 // Copyright:   (c) Julian Smart 
   9 // Licence:     wxWindows licence 
  10 ///////////////////////////////////////////////////////////////////////////// 
  12 // =========================================================================== 
  14 // =========================================================================== 
  16 // --------------------------------------------------------------------------- 
  18 // --------------------------------------------------------------------------- 
  20 // For compilers that support precompilation, includes "wx.h". 
  21 #include "wx/wxprec.h" 
  23 #if defined(__BORLANDC__) 
  28     #include "wx/msw/wrapcctl.h" 
  29     #include "wx/dynarray.h" 
  33     #include "wx/gdicmn.h" 
  36     #include "wx/cursor.h" 
  38     #include "wx/palette.h" 
  40     #include "wx/dialog.h" 
  41     #include "wx/msgdlg.h" 
  45     #include "wx/module.h" 
  48 #include "wx/apptrait.h" 
  49 #include "wx/filename.h" 
  50 #include "wx/dynlib.h" 
  51 #include "wx/evtloop.h" 
  52 #include "wx/thread.h" 
  53 #include "wx/scopeguard.h" 
  54 #include "wx/vector.h" 
  56 #include "wx/msw/private.h" 
  57 #include "wx/msw/dc.h" 
  58 #include "wx/msw/ole/oleutils.h" 
  59 #include "wx/msw/private/timer.h" 
  62     #include "wx/tooltip.h" 
  63 #endif // wxUSE_TOOLTIPS 
  65 // OLE is used for drag-and-drop, clipboard, OLE Automation..., but some 
  66 // compilers don't support it (missing headers, libs, ...) 
  67 #if defined(__GNUWIN32_OLD__) || defined(__SYMANTEC__) 
  71 #endif // broken compilers 
  73 #if defined(__POCKETPC__) || defined(__SMARTPHONE__) 
  85 #include "wx/msw/missing.h" 
  87 // instead of including <shlwapi.h> which is not part of the core SDK and not 
  88 // shipped at all with other compilers, we always define the parts of it we 
  89 // need here ourselves 
  91 // NB: DLLVER_PLATFORM_WINDOWS will be defined if shlwapi.h had been somehow 
  93 #ifndef DLLVER_PLATFORM_WINDOWS 
  94     // hopefully we don't need to change packing as DWORDs should be already 
  99         DWORD dwMajorVersion
;                   // Major version 
 100         DWORD dwMinorVersion
;                   // Minor version 
 101         DWORD dwBuildNumber
;                    // Build number 
 102         DWORD dwPlatformID
;                     // DLLVER_PLATFORM_* 
 105     typedef HRESULT (CALLBACK
* DLLGETVERSIONPROC
)(DLLVERSIONINFO 
*); 
 106 #endif // defined(DLLVERSIONINFO) 
 108 #ifndef ATTACH_PARENT_PROCESS 
 109     #define ATTACH_PARENT_PROCESS ((DWORD)-1) 
 112 // --------------------------------------------------------------------------- 
 114 // --------------------------------------------------------------------------- 
 116 #if !defined(__WXMICROWIN__) && !defined(__WXWINCE__) 
 117 extern void wxSetKeyboardHook(bool doIt
); 
 120 // because of mingw32 4.3 bug this struct can't be inside the namespace below: 
 121 // see http://article.gmane.org/gmane.comp.lib.wxwidgets.devel/110282 
 124     // the base name of the class: this is used to construct the unique name in 
 125     // wxApp::GetRegisteredClassName() 
 128     // the name of the registered class with and without CS_[HV]REDRAW styles 
 136 wxVector
<ClassRegInfo
> gs_regClassesInfo
; 
 138 } // anonymous namespace 
 140 // ---------------------------------------------------------------------------- 
 142 // ---------------------------------------------------------------------------- 
 144 LRESULT WXDLLEXPORT APIENTRY 
wxWndProc(HWND
, UINT
, WPARAM
, LPARAM
); 
 146 // =========================================================================== 
 147 // wxGUIAppTraits implementation 
 148 // =========================================================================== 
 150 // private class which we use to pass parameters from BeforeChildWaitLoop() to 
 151 // AfterChildWaitLoop() 
 152 struct ChildWaitLoopData
 
 154     ChildWaitLoopData(wxWindowDisabler 
*wd_
, wxWindow 
*winActive_
) 
 157         winActive 
= winActive_
; 
 160     wxWindowDisabler 
*wd
; 
 164 void *wxGUIAppTraits::BeforeChildWaitLoop() 
 167        We use a dirty hack here to disable all application windows (which we 
 168        must do because otherwise the calls to wxYield() could lead to some very 
 169        unexpected reentrancies in the users code) but to avoid losing 
 170        focus/activation entirely when the child process terminates which would 
 171        happen if we simply disabled everything using wxWindowDisabler. Indeed, 
 172        remember that Windows will never activate a disabled window and when the 
 173        last childs window is closed and Windows looks for a window to activate 
 174        all our windows are still disabled. There is no way to enable them in 
 175        time because we don't know when the childs windows are going to be 
 176        closed, so the solution we use here is to keep one special tiny frame 
 177        enabled all the time. Then when the child terminates it will get 
 178        activated and when we close it below -- after reenabling all the other 
 179        windows! -- the previously active window becomes activated again and 
 184     // first disable all existing windows 
 185     wxWindowDisabler 
*wd 
= new wxWindowDisabler
; 
 187     // then create an "invisible" frame: it has minimal size, is positioned 
 188     // (hopefully) outside the screen and doesn't appear on the taskbar 
 189     wxWindow 
*winActive 
= new wxFrame
 
 191                         wxTheApp
->GetTopWindow(), 
 194                         wxPoint(32600, 32600), 
 196                         wxDEFAULT_FRAME_STYLE 
| wxFRAME_NO_TASKBAR
 
 200     return new ChildWaitLoopData(wd
, winActive
); 
 203 void wxGUIAppTraits::AlwaysYield() 
 208 void wxGUIAppTraits::AfterChildWaitLoop(void *dataOrig
) 
 212     ChildWaitLoopData 
* const data 
= (ChildWaitLoopData 
*)dataOrig
; 
 216     // finally delete the dummy frame and, as wd has been already destroyed and 
 217     // the other windows reenabled, the activation is going to return to the 
 218     // window which had had it before 
 219     data
->winActive
->Destroy(); 
 221     // also delete the temporary data object itself 
 225 bool wxGUIAppTraits::DoMessageFromThreadWait() 
 227     // we should return false only if the app should exit, i.e. only if 
 228     // Dispatch() determines that the main event loop should terminate 
 229     wxEventLoopBase 
* const evtLoop 
= wxEventLoop::GetActive(); 
 230     if ( !evtLoop 
|| !evtLoop
->Pending() ) 
 232         // no events means no quit event 
 236     return evtLoop
->Dispatch(); 
 239 DWORD 
wxGUIAppTraits::WaitForThread(WXHANDLE hThread
) 
 241     // if we don't have a running event loop, we shouldn't wait for the 
 242     // messages as we never remove them from the message queue and so we enter 
 243     // an infinite loop as MsgWaitForMultipleObjects() keeps returning 
 245     if ( !wxEventLoop::GetActive() ) 
 246         return DoSimpleWaitForThread(hThread
); 
 248     return ::MsgWaitForMultipleObjects
 
 250                1,                   // number of objects to wait for 
 251                (HANDLE 
*)&hThread
,  // the objects 
 252                false,               // wait for any objects, not all 
 253                INFINITE
,            // no timeout 
 254                QS_ALLINPUT 
|        // return as soon as there are any events 
 259 wxPortId 
wxGUIAppTraits::GetToolkitVersion(int *majVer
, int *minVer
) const 
 264     // on Windows, the toolkit version is the same of the OS version 
 265     // as Windows integrates the OS kernel with the GUI toolkit. 
 266     info
.dwOSVersionInfoSize 
= sizeof(OSVERSIONINFO
); 
 267     if ( ::GetVersionEx(&info
) ) 
 270             *majVer 
= info
.dwMajorVersion
; 
 272             *minVer 
= info
.dwMinorVersion
; 
 275 #if defined(__WXHANDHELD__) || defined(__WXWINCE__) 
 284 wxTimerImpl 
*wxGUIAppTraits::CreateTimerImpl(wxTimer 
*timer
) 
 286     return new wxMSWTimerImpl(timer
); 
 289 #endif // wxUSE_TIMER 
 291 wxEventLoopBase
* wxGUIAppTraits::CreateEventLoop() 
 293     return new wxEventLoop
; 
 296 // --------------------------------------------------------------------------- 
 297 // Stuff for using console from the GUI applications 
 298 // --------------------------------------------------------------------------- 
 302 #include <wx/dynlib.h> 
 308     Helper class to manipulate console from a GUI app. 
 310     Notice that console output is available in the GUI app only if: 
 311     - AttachConsole() returns TRUE (which means it never works under pre-XP) 
 312     - we have a valid STD_ERROR_HANDLE 
 313     - command history hasn't been changed since our startup 
 315     To check if all these conditions are verified, you need to simple call 
 316     IsOkToUse(). It will check the first two conditions above the first time it 
 317     is called (and if this fails, the subsequent calls will return immediately) 
 318     and also recheck the last one every time it is called. 
 320 class wxConsoleStderr
 
 323     // default ctor does nothing, call Init() before using this class 
 326         m_hStderr 
= INVALID_HANDLE_VALUE
; 
 336         if ( m_hStderr 
!= INVALID_HANDLE_VALUE 
) 
 338             if ( !::FreeConsole() ) 
 340                 wxLogLastError(_T("FreeConsole")); 
 345     // return true if we were successfully initialized and there had been no 
 346     // console activity which would interfere with our output since then 
 347     bool IsOkToUse() const 
 351             wxConsoleStderr 
* const self 
= const_cast<wxConsoleStderr 
*>(this); 
 352             self
->m_ok 
= self
->DoInit(); 
 354             // no need to call IsHistoryUnchanged() as we just initialized 
 359         return m_ok 
&& IsHistoryUnchanged(); 
 363     // output the provided text on the console, return true if ok 
 364     bool Write(const wxString
& text
); 
 367     // called by Init() once only to do the real initialization 
 370     // retrieve the command line history into the provided buffer and return 
 372     int GetCommandHistory(wxWxCharBuffer
& buf
) const; 
 374     // check if the console history has changed 
 375     bool IsHistoryUnchanged() const; 
 377     int m_ok
;                   // initially -1, set to true or false by Init() 
 379     wxDynamicLibrary m_dllKernel32
; 
 381     HANDLE m_hStderr
;           // console handle, if it's valid we must call 
 382                                 // FreeConsole() (even if m_ok != 1) 
 384     wxWxCharBuffer m_history
;   // command history on startup 
 385     int m_historyLen
;           // length command history buffer 
 387     wxCharBuffer m_data
;        // data between empty line and cursor position 
 388     int m_dataLen
;              // length data buffer 
 389     int m_dataLine
;             // line offset 
 391     typedef DWORD (WINAPI 
*GetConsoleCommandHistory_t
)(LPTSTR sCommands
, 
 394     typedef DWORD (WINAPI 
*GetConsoleCommandHistoryLength_t
)(LPCTSTR sExeName
); 
 396     GetConsoleCommandHistory_t m_pfnGetConsoleCommandHistory
; 
 397     GetConsoleCommandHistoryLength_t m_pfnGetConsoleCommandHistoryLength
; 
 399     wxDECLARE_NO_COPY_CLASS(wxConsoleStderr
); 
 402 bool wxConsoleStderr::DoInit() 
 404     HANDLE hStderr 
= ::GetStdHandle(STD_ERROR_HANDLE
); 
 406     if ( hStderr 
== INVALID_HANDLE_VALUE 
|| !hStderr 
) 
 409     if ( !m_dllKernel32
.Load(_T("kernel32.dll")) ) 
 412     typedef BOOL (WINAPI 
*AttachConsole_t
)(DWORD dwProcessId
); 
 413     AttachConsole_t 
wxDL_INIT_FUNC(pfn
, AttachConsole
, m_dllKernel32
); 
 415     if ( !pfnAttachConsole 
|| !pfnAttachConsole(ATTACH_PARENT_PROCESS
) ) 
 418     // console attached, set m_hStderr now to ensure that we free it in the 
 422     wxDL_INIT_FUNC_AW(m_pfn
, GetConsoleCommandHistory
, m_dllKernel32
); 
 423     if ( !m_pfnGetConsoleCommandHistory 
) 
 426     wxDL_INIT_FUNC_AW(m_pfn
, GetConsoleCommandHistoryLength
, m_dllKernel32
); 
 427     if ( !m_pfnGetConsoleCommandHistoryLength 
) 
 430     // remember the current command history to be able to compare with it later 
 431     // in IsHistoryUnchanged() 
 432     m_historyLen 
= GetCommandHistory(m_history
); 
 437     // now find the first blank line above the current position 
 438     CONSOLE_SCREEN_BUFFER_INFO csbi
; 
 440     if ( !::GetConsoleScreenBufferInfo(m_hStderr
, &csbi
) ) 
 442         wxLogLastError(_T("GetConsoleScreenBufferInfo")); 
 448     pos
.Y 
= csbi
.dwCursorPosition
.Y 
+ 1; 
 450     // we decide that a line is empty if first 4 characters are spaces 
 456         if ( !::ReadConsoleOutputCharacterA(m_hStderr
, buf
, WXSIZEOF(buf
), 
 459             wxLogLastError(_T("ReadConsoleOutputCharacterA")); 
 462     } while ( wxStrncmp("    ", buf
, WXSIZEOF(buf
)) != 0 ); 
 464     // calculate line offset and length of data 
 465     m_dataLine 
= csbi
.dwCursorPosition
.Y 
- pos
.Y
; 
 466     m_dataLen 
= m_dataLine
*csbi
.dwMaximumWindowSize
.X 
+ csbi
.dwCursorPosition
.X
; 
 470         m_data
.extend(m_dataLen
); 
 471         if ( !::ReadConsoleOutputCharacterA(m_hStderr
, m_data
.data(), m_dataLen
, 
 474             wxLogLastError(_T("ReadConsoleOutputCharacterA")); 
 482 int wxConsoleStderr::GetCommandHistory(wxWxCharBuffer
& buf
) const 
 484     // these functions are internal and may only be called by cmd.exe 
 485     static const wxChar 
*CMD_EXE 
= _T("cmd.exe"); 
 487     const int len 
= m_pfnGetConsoleCommandHistoryLength(CMD_EXE
); 
 492         int len2 
= m_pfnGetConsoleCommandHistory(buf
.data(), len
, CMD_EXE
); 
 495         // there seems to be a bug in the GetConsoleCommandHistoryA(), it 
 496         // returns the length of Unicode string and not ANSI one 
 498 #endif // !wxUSE_UNICODE 
 502             wxFAIL_MSG( _T("failed getting history?") ); 
 509 bool wxConsoleStderr::IsHistoryUnchanged() const 
 511     wxASSERT_MSG( m_ok 
== 1, _T("shouldn't be called if not initialized") ); 
 513     // get (possibly changed) command history 
 514     wxWxCharBuffer history
; 
 515     const int historyLen 
= GetCommandHistory(history
); 
 517     // and compare it with the original one 
 518     return historyLen 
== m_historyLen 
&& history 
&& 
 519                 memcmp(m_history
, history
, historyLen
) == 0; 
 522 bool wxConsoleStderr::Write(const wxString
& text
) 
 524     wxASSERT_MSG( m_hStderr 
!= INVALID_HANDLE_VALUE
, 
 525                     _T("should only be called if Init() returned true") ); 
 527     // get current position 
 528     CONSOLE_SCREEN_BUFFER_INFO csbi
; 
 529     if ( !::GetConsoleScreenBufferInfo(m_hStderr
, &csbi
) ) 
 531         wxLogLastError(_T("GetConsoleScreenBufferInfo")); 
 535     // and calculate new position (where is empty line) 
 536     csbi
.dwCursorPosition
.X 
= 0; 
 537     csbi
.dwCursorPosition
.Y 
-= m_dataLine
; 
 539     if ( !::SetConsoleCursorPosition(m_hStderr
, csbi
.dwCursorPosition
) ) 
 541         wxLogLastError(_T("SetConsoleCursorPosition")); 
 546     if ( !::FillConsoleOutputCharacter(m_hStderr
, _T(' '), m_dataLen
, 
 547                                        csbi
.dwCursorPosition
, &ret
) ) 
 549         wxLogLastError(_T("FillConsoleOutputCharacter")); 
 553     if ( !::WriteConsole(m_hStderr
, text
.wx_str(), text
.length(), &ret
, NULL
) ) 
 555         wxLogLastError(_T("WriteConsole")); 
 559     WriteConsoleA(m_hStderr
, m_data
, m_dataLen
, &ret
, 0); 
 564 wxConsoleStderr s_consoleStderr
; 
 566 } // anonymous namespace 
 568 bool wxGUIAppTraits::CanUseStderr() 
 570     return s_consoleStderr
.IsOkToUse(); 
 573 bool wxGUIAppTraits::WriteToStderr(const wxString
& text
) 
 575     return s_consoleStderr
.IsOkToUse() && s_consoleStderr
.Write(text
); 
 578 #endif // !__WXWINCE__ 
 580 // =========================================================================== 
 581 // wxApp implementation 
 582 // =========================================================================== 
 584 int wxApp::m_nCmdShow 
= SW_SHOWNORMAL
; 
 586 // --------------------------------------------------------------------------- 
 588 // --------------------------------------------------------------------------- 
 590 IMPLEMENT_DYNAMIC_CLASS(wxApp
, wxEvtHandler
) 
 592 BEGIN_EVENT_TABLE(wxApp
, wxEvtHandler
) 
 593     EVT_IDLE(wxApp::OnIdle
) 
 594     EVT_END_SESSION(wxApp::OnEndSession
) 
 595     EVT_QUERY_END_SESSION(wxApp::OnQueryEndSession
) 
 598 // class to ensure that wxAppBase::CleanUp() is called if our Initialize() 
 600 class wxCallBaseCleanup
 
 603     wxCallBaseCleanup(wxApp 
*app
) : m_app(app
) { } 
 604     ~wxCallBaseCleanup() { if ( m_app 
) m_app
->wxAppBase::CleanUp(); } 
 606     void Dismiss() { m_app 
= NULL
; } 
 613 bool wxApp::Initialize(int& argc
, wxChar 
**argv
) 
 615     if ( !wxAppBase::Initialize(argc
, argv
) ) 
 618     // ensure that base cleanup is done if we return too early 
 619     wxCallBaseCleanup 
callBaseCleanup(this); 
 621 #if !defined(__WXMICROWIN__) 
 622     InitCommonControls(); 
 623 #endif // !defined(__WXMICROWIN__) 
 625 #if defined(__SMARTPHONE__) || defined(__POCKETPC__) 
 626     SHInitExtraControls(); 
 630     // Don't show a message box if a function such as SHGetFileInfo 
 631     // fails to find a device. 
 632     SetErrorMode(SEM_FAILCRITICALERRORS
|SEM_NOOPENFILEERRORBOX
); 
 637 #if !defined(__WXMICROWIN__) && !defined(__WXWINCE__) 
 638     wxSetKeyboardHook(true); 
 641     callBaseCleanup
.Dismiss(); 
 646 // --------------------------------------------------------------------------- 
 647 // Win32 window class registration 
 648 // --------------------------------------------------------------------------- 
 651 const wxChar 
*wxApp::GetRegisteredClassName(const wxChar 
*name
, 
 655     const size_t count 
= gs_regClassesInfo
.size(); 
 656     for ( size_t n 
= 0; n 
< count
; n
++ ) 
 658         if ( gs_regClassesInfo
[n
].basename 
== name 
) 
 659             return gs_regClassesInfo
[n
].regname
.c_str(); 
 662     // we need to register this class 
 664     wxZeroMemory(wndclass
); 
 666     wndclass
.lpfnWndProc   
= (WNDPROC
)wxWndProc
; 
 667     wndclass
.hInstance     
= wxhInstance
; 
 668     wndclass
.hCursor       
= ::LoadCursor(NULL
, IDC_ARROW
); 
 669     wndclass
.hbrBackground 
= (HBRUSH
)wxUIntToPtr(bgBrushCol 
+ 1); 
 670     wndclass
.style         
= CS_HREDRAW 
| CS_VREDRAW 
| CS_DBLCLKS 
| extraStyles
; 
 673     ClassRegInfo regClass
; 
 674     regClass
.basename 
= name
; 
 676     // constuct a unique suffix to allow registering the class with the same 
 677     // base name in a main application using wxWidgets and a DLL using 
 678     // wxWidgets loaded into its address space: as gs_regClassesInfo variable 
 679     // is different in them, we're going to obtain a unique prefix by using its 
 681     regClass
.regname 
= regClass
.basename 
+ 
 682                             wxString::Format(wxT("@%p"), &gs_regClassesInfo
); 
 683     wndclass
.lpszClassName 
= regClass
.regname
.wx_str(); 
 684     if ( !::RegisterClass(&wndclass
) ) 
 686         wxLogLastError(wxString::Format(wxT("RegisterClass(%s)"), 
 691     // NB: remember that code elsewhere supposes that no redraw class names 
 692     //     use the same names as normal classes with "NR" suffix so we must put 
 693     //     "NR" at the end instead of using more natural basename+"NR"+suffix 
 694     regClass
.regnameNR 
= regClass
.regname 
+ GetNoRedrawClassSuffix(); 
 695     wndclass
.style 
&= ~(CS_HREDRAW 
| CS_VREDRAW
); 
 696     wndclass
.lpszClassName 
= regClass
.regnameNR
.wx_str(); 
 697     if ( !::RegisterClass(&wndclass
) ) 
 699         wxLogLastError(wxString::Format(wxT("RegisterClass(%s)"), 
 701         ::UnregisterClass(regClass
.regname
.c_str(), wxhInstance
); 
 705     gs_regClassesInfo
.push_back(regClass
); 
 707     // take care to return the pointer which will remain valid after the 
 708     // function returns (it could be invalidated later if new elements are 
 709     // added to the vector and it's reallocated but this shouldn't matter as 
 710     // this pointer should be used right now, not stored) 
 711     return gs_regClassesInfo
.back().regname
.wx_str(); 
 714 bool wxApp::IsRegisteredClassName(const wxString
& name
) 
 716     const size_t count 
= gs_regClassesInfo
.size(); 
 717     for ( size_t n 
= 0; n 
< count
; n
++ ) 
 719         if ( gs_regClassesInfo
[n
].regname 
== name 
|| 
 720                 gs_regClassesInfo
[n
].regnameNR 
== name 
) 
 727 void wxApp::UnregisterWindowClasses() 
 729     const size_t count 
= gs_regClassesInfo
.size(); 
 730     for ( size_t n 
= 0; n 
< count
; n
++ ) 
 732         const ClassRegInfo
& regClass 
= gs_regClassesInfo
[n
]; 
 733         if ( !::UnregisterClass(regClass
.regname
.c_str(), wxhInstance
) ) 
 735             wxLogLastError(wxString::Format(wxT("UnregisterClass(%s)"), 
 739         if ( !::UnregisterClass(regClass
.regnameNR
.c_str(), wxhInstance
) ) 
 741             wxLogLastError(wxString::Format(wxT("UnregisterClass(%s)"), 
 742                            regClass
.regnameNR
)); 
 746     gs_regClassesInfo
.clear(); 
 749 void wxApp::CleanUp() 
 751     // all objects pending for deletion must be deleted first, otherwise 
 752     // UnregisterWindowClasses() call wouldn't succeed (because windows 
 753     // using the classes being unregistered still exist), so call the base 
 754     // class method first and only then do our clean up 
 755     wxAppBase::CleanUp(); 
 757 #if !defined(__WXMICROWIN__) && !defined(__WXWINCE__) 
 758     wxSetKeyboardHook(false); 
 763     // for an EXE the classes are unregistered when it terminates but DLL may 
 764     // be loaded several times (load/unload/load) into the same process in 
 765     // which case the registration will fail after the first time if we don't 
 766     // unregister the classes now 
 767     UnregisterWindowClasses(); 
 770 // ---------------------------------------------------------------------------- 
 772 // ---------------------------------------------------------------------------- 
 776     m_printMode 
= wxPRINT_WINDOWS
; 
 783 // ---------------------------------------------------------------------------- 
 784 // wxApp idle handling 
 785 // ---------------------------------------------------------------------------- 
 787 void wxApp::OnIdle(wxIdleEvent
& WXUNUSED(event
)) 
 789 #if wxUSE_DC_CACHEING 
 790     // automated DC cache management: clear the cached DCs and bitmap 
 791     // if it's likely that the app has finished with them, that is, we 
 792     // get an idle event and we're not dragging anything. 
 793     if (!::GetKeyState(MK_LBUTTON
) && !::GetKeyState(MK_MBUTTON
) && !::GetKeyState(MK_RBUTTON
)) 
 794         wxMSWDCImpl::ClearCache(); 
 795 #endif // wxUSE_DC_CACHEING 
 798 void wxApp::WakeUpIdle() 
 800     // Send the top window a dummy message so idle handler processing will 
 801     // start up again.  Doing it this way ensures that the idle handler 
 802     // wakes up in the right thread (see also wxWakeUpMainThread() which does 
 803     // the same for the main app thread only) 
 804     wxWindow 
* const topWindow 
= wxTheApp
->GetTopWindow(); 
 807         HWND hwndTop 
= GetHwndOf(topWindow
); 
 809         // Do not post WM_NULL if there's already a pending WM_NULL to avoid 
 810         // overflowing the message queue. 
 812         // Notice that due to a limitation of PeekMessage() API (which handles 
 813         // 0,0 range specially), we have to check the range from 0-1 instead. 
 814         // This still makes it possible to overflow the queue with WM_NULLs by 
 815         // interspersing the calles to WakeUpIdle() with windows creation but 
 816         // it should be rather hard to do it accidentally. 
 818         if ( !::PeekMessage(&msg
, hwndTop
, 0, 1, PM_NOREMOVE
) || 
 819               ::PeekMessage(&msg
, hwndTop
, 1, 1, PM_NOREMOVE
) ) 
 821             if ( !::PostMessage(hwndTop
, WM_NULL
, 0, 0) ) 
 823                 // should never happen 
 824                 wxLogLastError(wxT("PostMessage(WM_NULL)")); 
 830 // ---------------------------------------------------------------------------- 
 831 // other wxApp event hanlders 
 832 // ---------------------------------------------------------------------------- 
 834 void wxApp::OnEndSession(wxCloseEvent
& WXUNUSED(event
)) 
 836     // Windows will terminate the process soon after we return from 
 837     // WM_ENDSESSION handler or when we delete our last window, so make sure we 
 838     // at least execute our cleanup code before 
 840     // prevent the window from being destroyed when the corresponding wxTLW is 
 841     // destroyed: this will result in a leak of a HWND, of course, but who 
 842     // cares when the process is being killed anyhow 
 843     if ( !wxTopLevelWindows
.empty() ) 
 844         wxTopLevelWindows
[0]->SetHWND(0); 
 846     const int rc 
= OnExit(); 
 850     // calling exit() instead of ExitProcess() or not doing anything at all and 
 851     // being killed by Windows has the advantage of executing the dtors of 
 856 // Default behaviour: close the application with prompts. The 
 857 // user can veto the close, and therefore the end session. 
 858 void wxApp::OnQueryEndSession(wxCloseEvent
& event
) 
 862         if (!GetTopWindow()->Close(!event
.CanVeto())) 
 867 // ---------------------------------------------------------------------------- 
 868 // system DLL versions 
 869 // ---------------------------------------------------------------------------- 
 871 // these functions have trivial inline implementations for CE 
 874 #if wxUSE_DYNLIB_CLASS 
 879 // helper function: retrieve the DLL version by using DllGetVersion(), returns 
 880 // 0 if the DLL doesn't export such function 
 881 int CallDllGetVersion(wxDynamicLibrary
& dll
) 
 883     // now check if the function is available during run-time 
 884     wxDYNLIB_FUNCTION( DLLGETVERSIONPROC
, DllGetVersion
, dll 
); 
 885     if ( !pfnDllGetVersion 
) 
 889     dvi
.cbSize 
= sizeof(dvi
); 
 891     HRESULT hr 
= (*pfnDllGetVersion
)(&dvi
); 
 894         wxLogApiError(_T("DllGetVersion"), hr
); 
 899     return 100*dvi
.dwMajorVersion 
+ dvi
.dwMinorVersion
; 
 902 } // anonymous namespace 
 905 int wxApp::GetComCtl32Version() 
 909     // NB: this is MT-ok as in the worst case we'd compute s_verComCtl32 twice, 
 910     //     but as its value should be the same both times it doesn't matter 
 911     static int s_verComCtl32 
= -1; 
 913     if ( s_verComCtl32 
== -1 ) 
 915         // we're prepared to handle the errors 
 918         // we don't want to load comctl32.dll, it should be already loaded but, 
 919         // depending on the OS version and the presence of the manifest, it can 
 920         // be either v5 or v6 and instead of trying to guess it just get the 
 921         // handle of the already loaded version 
 922         wxLoadedDLL 
dllComCtl32(_T("comctl32.dll")); 
 923         if ( !dllComCtl32
.IsLoaded() ) 
 929         // try DllGetVersion() for recent DLLs 
 930         s_verComCtl32 
= CallDllGetVersion(dllComCtl32
); 
 932         // if DllGetVersion() is unavailable either during compile or 
 933         // run-time, try to guess the version otherwise 
 934         if ( !s_verComCtl32 
) 
 936             // InitCommonControlsEx is unique to 4.70 and later 
 937             void *pfn 
= dllComCtl32
.GetSymbol(_T("InitCommonControlsEx")); 
 940                 // not found, must be 4.00 
 945                 // many symbols appeared in comctl32 4.71, could use any of 
 946                 // them except may be DllInstall() 
 947                 pfn 
= dllComCtl32
.GetSymbol(_T("InitializeFlatSB")); 
 950                     // not found, must be 4.70 
 955                     // found, must be 4.71 or later 
 962     return s_verComCtl32
; 
 966 int wxApp::GetShell32Version() 
 968     static int s_verShell32 
= -1; 
 969     if ( s_verShell32 
== -1 ) 
 971         // we're prepared to handle the errors 
 974         wxDynamicLibrary 
dllShell32(_T("shell32.dll"), wxDL_VERBATIM
); 
 975         if ( dllShell32
.IsLoaded() ) 
 977             s_verShell32 
= CallDllGetVersion(dllShell32
); 
 981                 // there doesn't seem to be any way to distinguish between 4.00 
 982                 // and 4.70 (starting from 4.71 we have DllGetVersion()) so 
 983                 // just assume it is 4.0 
 987         else // failed load the DLL? 
 996 #else // !wxUSE_DYNLIB_CLASS 
 999 int wxApp::GetComCtl32Version() 
1005 int wxApp::GetShell32Version() 
1010 #endif // wxUSE_DYNLIB_CLASS/!wxUSE_DYNLIB_CLASS 
1012 #endif // !__WXWINCE__ 
1014 #if wxUSE_EXCEPTIONS 
1016 // ---------------------------------------------------------------------------- 
1017 // exception handling 
1018 // ---------------------------------------------------------------------------- 
1020 bool wxApp::OnExceptionInMainLoop() 
1022     // ask the user about what to do: use the Win32 API function here as it 
1023     // could be dangerous to use any wxWidgets code in this state 
1028                 _T("An unhandled exception occurred. Press \"Abort\" to \ 
1029 terminate the program,\r\n\ 
1030 \"Retry\" to exit the program normally and \"Ignore\" to try to continue."), 
1031                 _T("Unhandled exception"), 
1032                 MB_ABORTRETRYIGNORE 
| 
1042             wxFAIL_MSG( _T("unexpected MessageBox() return code") ); 
1053 #endif // wxUSE_EXCEPTIONS