1 ///////////////////////////////////////////////////////////////////////////// 
   4 // Author:      Julian Smart 
   8 // Copyright:   (c) Julian Smart and Markus Holzem 
   9 // Licence:     wxWindows license 
  10 ///////////////////////////////////////////////////////////////////////////// 
  12 // =========================================================================== 
  14 // =========================================================================== 
  16 // --------------------------------------------------------------------------- 
  18 // --------------------------------------------------------------------------- 
  21     #pragma implementation "app.h" 
  24 // For compilers that support precompilation, includes "wx.h". 
  25 #include "wx/wxprec.h" 
  27 #if defined(__BORLANDC__) 
  35     #include "wx/gdicmn.h" 
  38     #include "wx/cursor.h" 
  40     #include "wx/palette.h" 
  42     #include "wx/dialog.h" 
  43     #include "wx/msgdlg.h" 
  45     #include "wx/dynarray.h" 
  46     #include "wx/wxchar.h" 
  51 #include "wx/module.h" 
  53 #include "wx/msw/private.h" 
  56     #include "wx/thread.h" 
  58     // define the array of MSG strutures 
  59     WX_DECLARE_OBJARRAY(MSG
, wxMsgArray
); 
  61     #include "wx/arrimpl.cpp" 
  63     WX_DEFINE_OBJARRAY(wxMsgArray
); 
  64 #endif // wxUSE_THREADS 
  66 #if wxUSE_WX_RESOURCES 
  67     #include "wx/resource.h" 
  70 // OLE is used for drag-and-drop, clipboard, OLE Automation... 
  71 #ifndef wxUSE_NORLANDER_HEADERS 
  72 #if defined(__GNUWIN32__) || defined(__SC__) || defined(__SALFORDC__) 
  76 #endif // broken compilers 
  86 #if (defined(__WIN95__) && !defined(__GNUWIN32__)) || defined(__TWIN32__) || defined(wxUSE_NORLANDER_HEADERS) 
  90 #include "wx/msw/msvcrt.h" 
  92 // ---------------------------------------------------------------------------- 
  93 // conditional compilation 
  94 // ---------------------------------------------------------------------------- 
  96 // The macro _WIN32_IE is defined by commctrl.h (unless it had already been 
  97 // defined before) and shows us what common control features are available 
  98 // during the compile time (it doesn't mean that they will be available during 
  99 // the run-time, use GetComCtl32Version() to test for them!). The possible 
 102 // 0x0200     for comctl32.dll 4.00 shipped with Win95/NT 4.0 
 103 // 0x0300                      4.70              IE 3.x 
 104 // 0x0400                      4.71              IE 4.0 
 105 // 0x0401                      4.72              IE 4.01 and Win98 
 106 // 0x0500                      5.00              IE 5.x and NT 5.0 (Win2000) 
 109     // minimal set of features by default 
 110     #define _WIN32_IE 0x0200 
 113 #if _WIN32_IE >= 0x0300 
 117 // --------------------------------------------------------------------------- 
 119 // --------------------------------------------------------------------------- 
 121 extern wxChar 
*wxBuffer
; 
 122 extern wxChar 
*wxOsVersion
; 
 123 extern wxList 
*wxWinHandleList
; 
 124 extern wxList WXDLLEXPORT wxPendingDelete
; 
 125 extern void wxSetKeyboardHook(bool doIt
); 
 126 extern wxCursor 
*g_globalCursor
; 
 129 wxApp 
*wxTheApp 
= NULL
; 
 131 // NB: all "NoRedraw" classes must have the same names as the "normal" classes 
 132 //     with NR suffix - wxWindow::MSWCreate() supposes this 
 133 const wxChar 
*wxFrameClassName         
= wxT("wxFrameClass"); 
 134 const wxChar 
*wxFrameClassNameNoRedraw 
= wxT("wxFrameClassNR"); 
 135 const wxChar 
*wxMDIFrameClassName      
= wxT("wxMDIFrameClass"); 
 136 const wxChar 
*wxMDIFrameClassNameNoRedraw 
= wxT("wxMDIFrameClassNR"); 
 137 const wxChar 
*wxMDIChildFrameClassName 
= wxT("wxMDIChildFrameClass"); 
 138 const wxChar 
*wxMDIChildFrameClassNameNoRedraw 
= wxT("wxMDIChildFrameClassNR"); 
 139 const wxChar 
*wxPanelClassName         
= wxT("wxPanelClass"); 
 140 const wxChar 
*wxCanvasClassName        
= wxT("wxCanvasClass"); 
 142 HICON wxSTD_FRAME_ICON 
= (HICON
) NULL
; 
 143 HICON wxSTD_MDICHILDFRAME_ICON 
= (HICON
) NULL
; 
 144 HICON wxSTD_MDIPARENTFRAME_ICON 
= (HICON
) NULL
; 
 146 HICON wxDEFAULT_FRAME_ICON 
= (HICON
) NULL
; 
 147 HICON wxDEFAULT_MDICHILDFRAME_ICON 
= (HICON
) NULL
; 
 148 HICON wxDEFAULT_MDIPARENTFRAME_ICON 
= (HICON
) NULL
; 
 150 HBRUSH wxDisableButtonBrush 
= (HBRUSH
) 0; 
 152 LRESULT WXDLLEXPORT APIENTRY 
wxWndProc(HWND
, UINT
, WPARAM
, LPARAM
); 
 154 // =========================================================================== 
 156 // =========================================================================== 
 158 // --------------------------------------------------------------------------- 
 160 // --------------------------------------------------------------------------- 
 162     IMPLEMENT_DYNAMIC_CLASS(wxApp
, wxEvtHandler
) 
 164     BEGIN_EVENT_TABLE(wxApp
, wxEvtHandler
) 
 165         EVT_IDLE(wxApp::OnIdle
) 
 166         EVT_END_SESSION(wxApp::OnEndSession
) 
 167         EVT_QUERY_END_SESSION(wxApp::OnQueryEndSession
) 
 171 bool wxApp::Initialize() 
 173     // Some people may wish to use this, but 
 174     // probably it shouldn't be here by default. 
 176     //    wxRedirectIOToConsole(); 
 179     wxBuffer 
= new wxChar
[1500]; // FIXME 
 181     wxClassInfo::InitializeClasses(); 
 184     wxGetResource(wxT("wxWindows"), wxT("OsVersion"), &wxOsVersion
); 
 188     wxPendingEventsLocker 
= new wxCriticalSection
; 
 191     wxTheColourDatabase 
= new wxColourDatabase(wxKEY_STRING
); 
 192     wxTheColourDatabase
->Initialize(); 
 194     wxInitializeStockLists(); 
 195     wxInitializeStockObjects(); 
 197 #if wxUSE_WX_RESOURCES 
 198     wxInitializeResourceSystem(); 
 201     wxBitmap::InitStandardHandlers(); 
 203 #if defined(__WIN95__) 
 204     InitCommonControls(); 
 211     // for OLE, enlarge message queue to be as large as possible 
 213     while (!SetMessageQueue(iMsg
) && (iMsg 
-= 8)) 
 216     // we need to initialize OLE library 
 217     if ( FAILED(::OleInitialize(NULL
)) ) 
 218         wxLogError(_("Cannot initialize OLE")); 
 222     if (!Ctl3dRegister(wxhInstance
)) 
 223         wxLogError(wxT("Cannot register CTL3D")); 
 225     Ctl3dAutoSubclass(wxhInstance
); 
 228     g_globalCursor 
= new wxCursor
; 
 230     // VZ: these icons are not in wx.rc anyhow (but should they?)! 
 232     wxSTD_FRAME_ICON 
= LoadIcon(wxhInstance
, wxT("wxSTD_FRAME")); 
 233     wxSTD_MDIPARENTFRAME_ICON 
= LoadIcon(wxhInstance
, wxT("wxSTD_MDIPARENTFRAME")); 
 234     wxSTD_MDICHILDFRAME_ICON 
= LoadIcon(wxhInstance
, wxT("wxSTD_MDICHILDFRAME")); 
 236     wxDEFAULT_FRAME_ICON 
= LoadIcon(wxhInstance
, wxT("wxDEFAULT_FRAME")); 
 237     wxDEFAULT_MDIPARENTFRAME_ICON 
= LoadIcon(wxhInstance
, wxT("wxDEFAULT_MDIPARENTFRAME")); 
 238     wxDEFAULT_MDICHILDFRAME_ICON 
= LoadIcon(wxhInstance
, wxT("wxDEFAULT_MDICHILDFRAME")); 
 241     RegisterWindowClasses(); 
 243     // Create the brush for disabling bitmap buttons 
 246     lb
.lbStyle 
= BS_PATTERN
; 
 247     lb
.lbHatch 
= (int)LoadBitmap( wxhInstance
, wxT("wxDISABLE_BUTTON_BITMAP") ); 
 250         wxDisableButtonBrush 
= ::CreateBrushIndirect( & lb 
); 
 251         ::DeleteObject( (HGDIOBJ
)lb
.lbHatch 
); 
 253     //else: wxWindows resources are probably not linked in 
 259     wxWinHandleList 
= new wxList(wxKEY_INTEGER
); 
 261     // This is to foil optimizations in Visual C++ that throw out dummy.obj. 
 262     // PLEASE DO NOT ALTER THIS. 
 263 #if defined(__VISUALC__) && !defined(WXMAKINGDLL) 
 264     extern char wxDummyChar
; 
 265     if (wxDummyChar
) wxDummyChar
++; 
 268     wxSetKeyboardHook(TRUE
); 
 270     wxModule::RegisterModules(); 
 271     if (!wxModule::InitializeModules()) 
 276 // --------------------------------------------------------------------------- 
 277 // RegisterWindowClasses 
 278 // --------------------------------------------------------------------------- 
 280 // TODO we should only register classes really used by the app. For this it 
 281 //      would be enough to just delay the class registration until an attempt 
 282 //      to create a window of this class is made. 
 283 bool wxApp::RegisterWindowClasses() 
 287     // for each class we register one with CS_(V|H)REDRAW style and one 
 288     // without for windows created with wxNO_FULL_REDRAW_ON_REPAINT flag 
 289     static const long styleNormal 
= CS_HREDRAW 
| CS_VREDRAW 
| CS_DBLCLKS
; 
 290     static const long styleNoRedraw 
= CS_DBLCLKS
; 
 292     // the fields which are common to all classes 
 293     wndclass
.lpfnWndProc   
= (WNDPROC
)wxWndProc
; 
 294     wndclass
.cbClsExtra    
= 0; 
 295     wndclass
.cbWndExtra    
= sizeof( DWORD 
); // VZ: what is this DWORD used for? 
 296     wndclass
.hInstance     
= wxhInstance
; 
 297     wndclass
.hIcon         
= (HICON
) NULL
; 
 298     wndclass
.hCursor       
= ::LoadCursor((HINSTANCE
)NULL
, IDC_ARROW
); 
 299     wndclass
.lpszMenuName  
= NULL
; 
 301     // Register the frame window class. 
 302     wndclass
.hbrBackground 
= (HBRUSH
)(COLOR_APPWORKSPACE 
+ 1); 
 303     wndclass
.lpszClassName 
= wxFrameClassName
; 
 304     wndclass
.style         
= styleNormal
; 
 306     if ( !RegisterClass(&wndclass
) ) 
 308         wxLogLastError("RegisterClass(frame)"); 
 314     wndclass
.lpszClassName 
= wxFrameClassNameNoRedraw
; 
 315     wndclass
.style         
= styleNoRedraw
; 
 317     if ( !RegisterClass(&wndclass
) ) 
 319         wxLogLastError("RegisterClass(no redraw frame)"); 
 324     // Register the MDI frame window class. 
 325     wndclass
.hbrBackground 
= (HBRUSH
)NULL
; // paint MDI frame ourselves 
 326     wndclass
.lpszClassName 
= wxMDIFrameClassName
; 
 327     wndclass
.style         
= styleNormal
; 
 329     if ( !RegisterClass(&wndclass
) ) 
 331         wxLogLastError("RegisterClass(MDI parent)"); 
 336     // "no redraw" MDI frame 
 337     wndclass
.lpszClassName 
= wxMDIFrameClassNameNoRedraw
; 
 338     wndclass
.style         
= styleNoRedraw
; 
 340     if ( !RegisterClass(&wndclass
) ) 
 342         wxLogLastError("RegisterClass(no redraw MDI parent frame)"); 
 347     // Register the MDI child frame window class. 
 348     wndclass
.hbrBackground 
= (HBRUSH
)(COLOR_WINDOW 
+ 1); 
 349     wndclass
.lpszClassName 
= wxMDIChildFrameClassName
; 
 350     wndclass
.style         
= styleNormal
; 
 352     if ( !RegisterClass(&wndclass
) ) 
 354         wxLogLastError("RegisterClass(MDI child)"); 
 359     // "no redraw" MDI child frame 
 360     wndclass
.lpszClassName 
= wxMDIChildFrameClassNameNoRedraw
; 
 361     wndclass
.style         
= styleNoRedraw
; 
 363     if ( !RegisterClass(&wndclass
) ) 
 365         wxLogLastError("RegisterClass(no redraw MDI child)"); 
 370     // Register the panel window class. 
 371     wndclass
.hbrBackground 
= (HBRUSH
) GetStockObject( LTGRAY_BRUSH 
); 
 372     wndclass
.lpszClassName 
= wxPanelClassName
; 
 373     wndclass
.style         
= styleNormal
; 
 375     if ( !RegisterClass(&wndclass
) ) 
 377         wxLogLastError("RegisterClass(panel)"); 
 382     // Register the canvas and textsubwindow class name 
 383     wndclass
.hbrBackground 
= (HBRUSH
)NULL
; 
 384     wndclass
.lpszClassName 
= wxCanvasClassName
; 
 386     if ( !RegisterClass(&wndclass
) ) 
 388         wxLogLastError("RegisterClass(canvas)"); 
 396 // --------------------------------------------------------------------------- 
 397 // Convert Windows to argc, argv style 
 398 // --------------------------------------------------------------------------- 
 400 void wxApp::ConvertToStandardCommandArgs(char* lpCmdLine
) 
 404     wxString 
cmdLine(lpCmdLine
); 
 407     // Get application name 
 408     wxChar name
[260]; // 260 is MAX_PATH value from windef.h 
 409     ::GetModuleFileName(wxhInstance
, name
, WXSIZEOF(name
)); 
 414     wxStrcpy(name
, wxFileNameFromPath(name
)); 
 415     wxStripExtension(name
); 
 416     wxTheApp
->SetAppName(name
); 
 419     // Treat strings enclosed in double-quotes as single arguments 
 421     int len 
= cmdLine
.Length(); 
 425         while ((i 
< len
) && wxIsspace(cmdLine
.GetChar(i
))) 
 430             if (cmdLine
.GetChar(i
) == wxT('"')) // We found the start of a string 
 434                 while ((i 
< len
) && (cmdLine
.GetChar(i
) != wxT('"'))) 
 437                 wxString 
arg(cmdLine
.Mid(first
, (i 
- first
))); 
 443                     i 
++; // Skip past 2nd quote 
 445             else // Unquoted argument 
 448                 while ((i 
< len
) && !wxIsspace(cmdLine
.GetChar(i
))) 
 451                 wxString 
arg(cmdLine
.Mid(first
, (i 
- first
))); 
 459     wxTheApp
->argv 
= new wxChar
*[count 
+ 1]; 
 460     for (i 
= 0; i 
< count
; i
++) 
 462         wxString 
arg(args
[i
]); 
 463         wxTheApp
->argv
[i
] = copystring((const wxChar
*)arg
); 
 465     wxTheApp
->argv
[count
] = NULL
; // argv[] is a NULL-terminated list 
 466     wxTheApp
->argc 
= count
; 
 469 //// Cleans up any wxWindows internal structures left lying around 
 471 void wxApp::CleanUp() 
 476     // flush the logged messages if any and install a 'safer' log target: the 
 477     // default one (wxLogGui) can't be used after the resources are freed just 
 478     // below and the user suppliedo ne might be even more unsafe (using any 
 479     // wxWindows GUI function is unsafe starting from now) 
 480     wxLog::DontCreateOnDemand(); 
 482     // this will flush the old messages if any 
 483     delete wxLog::SetActiveTarget(new wxLogStderr
); 
 486     // One last chance for pending objects to be cleaned up 
 487     wxTheApp
->DeletePendingObjects(); 
 489     wxModule::CleanUpModules(); 
 491 #if wxUSE_WX_RESOURCES 
 492     wxCleanUpResourceSystem(); 
 494     //  wxDefaultResourceTable->ClearTable(); 
 497     // Indicate that the cursor can be freed, so that cursor won't be deleted 
 498     // by deleting the bitmap list before g_globalCursor goes out of scope 
 499     // (double deletion of the cursor). 
 500     wxSetCursor(wxNullCursor
); 
 501     delete g_globalCursor
; 
 502     g_globalCursor 
= NULL
; 
 504     wxDeleteStockObjects(); 
 506     // Destroy all GDI lists, etc. 
 507     wxDeleteStockLists(); 
 509     delete wxTheColourDatabase
; 
 510     wxTheColourDatabase 
= NULL
; 
 512     wxBitmap::CleanUpHandlers(); 
 517     //// WINDOWS-SPECIFIC CLEANUP 
 519     wxSetKeyboardHook(FALSE
); 
 525     if (wxSTD_FRAME_ICON
) 
 526         DestroyIcon(wxSTD_FRAME_ICON
); 
 527     if (wxSTD_MDICHILDFRAME_ICON
) 
 528         DestroyIcon(wxSTD_MDICHILDFRAME_ICON
); 
 529     if (wxSTD_MDIPARENTFRAME_ICON
) 
 530         DestroyIcon(wxSTD_MDIPARENTFRAME_ICON
); 
 532     if (wxDEFAULT_FRAME_ICON
) 
 533         DestroyIcon(wxDEFAULT_FRAME_ICON
); 
 534     if (wxDEFAULT_MDICHILDFRAME_ICON
) 
 535         DestroyIcon(wxDEFAULT_MDICHILDFRAME_ICON
); 
 536     if (wxDEFAULT_MDIPARENTFRAME_ICON
) 
 537         DestroyIcon(wxDEFAULT_MDIPARENTFRAME_ICON
); 
 539     if ( wxDisableButtonBrush 
) 
 540         ::DeleteObject( wxDisableButtonBrush 
); 
 547     Ctl3dUnregister(wxhInstance
); 
 551         delete wxWinHandleList
; 
 553     // GL: I'm annoyed ... I don't know where to put this and I don't want to 
 554     // create a module for that as it's part of the core. 
 555     delete wxPendingEvents
; 
 557     delete wxPendingEventsLocker
; 
 558     // If we don't do the following, we get an apparent memory leak. 
 559     ((wxEvtHandler
&) wxDefaultValidator
).ClearEventLocker(); 
 562     wxClassInfo::CleanUpClasses(); 
 567 #if (defined(__WXDEBUG__) && wxUSE_MEMORY_TRACING) || wxUSE_DEBUG_CONTEXT 
 568     // At this point we want to check if there are any memory 
 569     // blocks that aren't part of the wxDebugContext itself, 
 570     // as a special case. Then when dumping we need to ignore 
 571     // wxDebugContext, too. 
 572     if (wxDebugContext::CountObjectsLeft(TRUE
) > 0) 
 574         wxLogDebug(wxT("There were memory leaks.")); 
 575         wxDebugContext::Dump(); 
 576         wxDebugContext::PrintStatistics(); 
 578     //  wxDebugContext::SetStream(NULL, NULL); 
 582     // do it as the very last thing because everything else can log messages 
 583     delete wxLog::SetActiveTarget(NULL
); 
 587 #if !defined(_WINDLL) || (defined(_WINDLL) && defined(WXMAKINGDLL)) 
 589 // temporarily disable this warning which would be generated in release builds 
 592     #pragma warning(disable: 4715) // not all control paths return a value 
 595 //// Main wxWindows entry point 
 596 int wxEntry(WXHINSTANCE hInstance
, 
 597             WXHINSTANCE 
WXUNUSED(hPrevInstance
), 
 602     // do check for memory leaks on program exit 
 603     // (another useful flag is _CRTDBG_DELAY_FREE_MEM_DF which doesn't free 
 604     //  deallocated memory which may be used to simulate low-memory condition) 
 605     wxCrtSetDbgFlag(_CRTDBG_LEAK_CHECK_DF
); 
 607 #if (defined(__WXDEBUG__) && wxUSE_MEMORY_TRACING) || wxUSE_DEBUG_CONTEXT 
 608     // This seems to be necessary since there are 'rogue' 
 609     // objects present at this point (perhaps global objects?) 
 610     // Setting a checkpoint will ignore them as far as the 
 611     // memory checking facility is concerned. 
 612     // Of course you may argue that memory allocated in globals should be 
 613     // checked, but this is a reasonable compromise. 
 614     wxDebugContext::SetCheckpoint(); 
 617     // take everything into a try-except block in release build 
 618     // FIXME other compilers must support Win32 SEH (structured exception 
 619     //       handling) too, just find the appropriate keyword in their docs! 
 620     //       Please note that it's _not_ the same as C++ exceptions! 
 621 #if !defined(__WXDEBUG__) && defined(__VISUALC__) 
 622     #define CATCH_PROGRAM_EXCEPTIONS 
 626     #undef  CATCH_PROGRAM_EXCEPTIONS 
 628         wxhInstance 
= (HINSTANCE
) hInstance
; 
 630         if (!wxApp::Initialize()) 
 633         // create the application object or ensure that one already exists 
 636             // The app may have declared a global application object, but we recommend 
 637             // the IMPLEMENT_APP macro is used instead, which sets an initializer 
 638             // function for delayed, dynamic app object construction. 
 639             wxCHECK_MSG( wxApp::GetInitializerFunction(), 0, 
 640                          wxT("No initializer - use IMPLEMENT_APP macro.") ); 
 642             wxTheApp 
= (*wxApp::GetInitializerFunction()) (); 
 645         wxCHECK_MSG( wxTheApp
, 0, wxT("You have to define an instance of wxApp!") ); 
 647         // save the WinMain() parameters 
 648         wxTheApp
->ConvertToStandardCommandArgs(lpCmdLine
); 
 649         wxTheApp
->m_nCmdShow 
= nCmdShow
; 
 651         // GUI-specific initialisation. In fact on Windows we don't have any, 
 652         // but this call is provided for compatibility across platforms. 
 653         wxTheApp
->OnInitGui(); 
 655         // We really don't want timestamps by default, because it means 
 656         // we can't simply double-click on the error message and get to that 
 657         // line in the source. So VC++ at least, let's have a sensible default. 
 659         wxLog::SetTimestamp(NULL
); 
 664         if ( wxTheApp
->OnInit() ) 
 668                 retValue 
= wxTheApp
->OnRun(); 
 671                 // We want to initialize, but not run or exit immediately. 
 674         //else: app initialization failed, so we skipped OnRun() 
 676         wxWindow 
*topWindow 
= wxTheApp
->GetTopWindow(); 
 679             // Forcibly delete the window. 
 680             if ( topWindow
->IsKindOf(CLASSINFO(wxFrame
)) || 
 681                     topWindow
->IsKindOf(CLASSINFO(wxDialog
)) ) 
 683                 topWindow
->Close(TRUE
); 
 684                 wxTheApp
->DeletePendingObjects(); 
 689                 wxTheApp
->SetTopWindow(NULL
); 
 699 #ifdef CATCH_PROGRAM_EXCEPTIONS 
 701     __except ( EXCEPTION_EXECUTE_HANDLER 
) { 
 704            wxTheApp->OnFatalException(); 
 707         // using wxLog would be unsafe here 
 709                      _("Unrecoverable program error detected: " 
 710                      " the application will terminate."), 
 712                      MB_APPLMODAL 
| MB_ICONSTOP 
| MB_OK
); 
 714         ::ExitProcess(3); // the same exit code as abort() 
 718 #endif // CATCH_PROGRAM_EXCEPTIONS 
 721 // restore warning state 
 723     #pragma warning(default: 4715) // not all control paths return a value 
 728 //// Entry point for DLLs 
 730 int wxEntry(WXHINSTANCE hInstance
) 
 732     wxhInstance 
= (HINSTANCE
) hInstance
; 
 735     // The app may have declared a global application object, but we recommend 
 736     // the IMPLEMENT_APP macro is used instead, which sets an initializer function 
 737     // for delayed, dynamic app object construction. 
 740         wxCHECK_MSG( wxApp::GetInitializerFunction(), 0, 
 741                      "No initializer - use IMPLEMENT_APP macro." ); 
 743         wxTheApp 
= (* wxApp::GetInitializerFunction()) (); 
 746     wxCHECK_MSG( wxTheApp
, 0, "You have to define an instance of wxApp!" ); 
 749     wxTheApp
->argv 
= NULL
; 
 751     wxTheApp
->OnInitGui(); 
 755     wxWindow 
*topWindow 
= wxTheApp
->GetTopWindow(); 
 756     if ( topWindow 
&& topWindow
->GetHWND()) 
 758         topWindow
->Show(TRUE
); 
 765 //// Static member initialization 
 767 wxAppInitializerFunction 
wxAppBase::m_appInitFn 
= (wxAppInitializerFunction
) NULL
; 
 773     m_wantDebugOutput 
= TRUE
; 
 777     m_printMode 
= wxPRINT_WINDOWS
; 
 778     m_exitOnFrameDelete 
= TRUE
; 
 784     // Delete command-line args 
 786     for (i 
= 0; i 
< argc
; i
++) 
 793 bool wxApp::Initialized() 
 801 #ifdef _WINDLL // Assume initialized if DLL (no way of telling) 
 807  * Get and process a message, returning FALSE if WM_QUIT 
 808  * received (and also set the flag telling the app to exit the main loop) 
 811 bool wxApp::DoMessage() 
 813     BOOL rc 
= ::GetMessage(&s_currentMsg
, (HWND
) NULL
, 0, 0); 
 823         // should never happen, but let's test for it nevertheless 
 824         wxLogLastError("GetMessage"); 
 829         wxASSERT_MSG( wxThread::IsMain(), 
 830                       wxT("only the main thread can process Windows messages") ); 
 832         static bool s_hadGuiLock 
= TRUE
; 
 833         static wxMsgArray s_aSavedMessages
; 
 835         // if a secondary thread owns is doing GUI calls, save all messages for 
 836         // later processing - we can't process them right now because it will 
 837         // lead to recursive library calls (and we're not reentrant) 
 838         if ( !wxGuiOwnedByMainThread() ) 
 840             s_hadGuiLock 
= FALSE
; 
 842             // leave out WM_COMMAND messages: too dangerous, sometimes 
 843             // the message will be processed twice 
 844             if ( !wxIsWaitingForThread() || 
 845                     s_currentMsg
.message 
!= WM_COMMAND 
) 
 847                 s_aSavedMessages
.Add(s_currentMsg
); 
 854             // have we just regained the GUI lock? if so, post all of the saved 
 857             // FIXME of course, it's not _exactly_ the same as processing the 
 858             //       messages normally - expect some things to break... 
 863                 size_t count 
= s_aSavedMessages
.Count(); 
 864                 for ( size_t n 
= 0; n 
< count
; n
++ ) 
 866                     MSG
& msg 
= s_aSavedMessages
[n
]; 
 868                     if ( !ProcessMessage((WXMSG 
*)&msg
) ) 
 870                         ::TranslateMessage(&msg
); 
 871                         ::DispatchMessage(&msg
); 
 875                 s_aSavedMessages
.Empty(); 
 878 #endif // wxUSE_THREADS 
 880         // Process the message 
 881         if ( !ProcessMessage((WXMSG 
*)&s_currentMsg
) ) 
 883             ::TranslateMessage(&s_currentMsg
); 
 884             ::DispatchMessage(&s_currentMsg
); 
 892  * Keep trying to process messages until WM_QUIT 
 895  * If there are messages to be processed, they will all be 
 896  * processed and OnIdle will not be called. 
 897  * When there are no more messages, OnIdle is called. 
 898  * If OnIdle requests more time, 
 899  * it will be repeatedly called so long as there are no pending messages. 
 900  * A 'feature' of this is that once OnIdle has decided that no more processing 
 901  * is required, then it won't get processing time until further messages 
 902  * are processed (it'll sit in DoMessage). 
 905 int wxApp::MainLoop() 
 909     while ( m_keepGoing 
) 
 912         wxMutexGuiLeaveOrEnter(); 
 913 #endif // wxUSE_THREADS 
 915         while ( !::PeekMessage(&s_currentMsg
, 0, 0, 0, PM_NOREMOVE
) && 
 924     return s_currentMsg
.wParam
; 
 927 // Returns TRUE if more time is needed. 
 928 bool wxApp::ProcessIdle() 
 931     event
.SetEventObject(this); 
 934     return event
.MoreRequested(); 
 937 void wxApp::ExitMainLoop() 
 942 bool wxApp::Pending() 
 944     return (::PeekMessage(&s_currentMsg
, 0, 0, 0, PM_NOREMOVE
) != 0); 
 947 void wxApp::Dispatch() 
 953  * Give all windows a chance to preprocess 
 954  * the message. Some may have accelerator tables, or have 
 958 bool wxApp::ProcessMessage(WXMSG 
*wxmsg
) 
 960     MSG 
*msg 
= (MSG 
*)wxmsg
; 
 961     HWND hWnd 
= msg
->hwnd
; 
 962     wxWindow 
*wndThis 
= wxFindWinFromHandle((WXHWND
)hWnd
), *wnd
; 
 964     // for some composite controls (like a combobox), wndThis might be NULL 
 965     // because the subcontrol is not a wxWindow, but only the control itself 
 966     // is - try to catch this case 
 967     while ( hWnd 
&& !wndThis 
) 
 969         hWnd 
= ::GetParent(hWnd
); 
 970         wndThis 
= wxFindWinFromHandle((WXHWND
)hWnd
); 
 973     // Try translations first; find the youngest window with 
 974     // a translation table. 
 975     for ( wnd 
= wndThis
; wnd
; wnd 
= wnd
->GetParent() ) 
 977         if ( wnd
->MSWTranslateMessage(wxmsg
) ) 
 981     // Anyone for a non-translation message? Try youngest descendants first. 
 982     for ( wnd 
= wndThis
; wnd
; wnd 
= wnd
->GetParent() ) 
 984         if ( wnd
->MSWProcessMessage(wxmsg
) ) 
 991 void wxApp::OnIdle(wxIdleEvent
& event
) 
 993     static bool s_inOnIdle 
= FALSE
; 
 995     // Avoid recursion (via ProcessEvent default case) 
1001     // 'Garbage' collection of windows deleted with Close(). 
1002     DeletePendingObjects(); 
1005     // flush the logged messages if any 
1006     wxLog 
*pLog 
= wxLog::GetActiveTarget(); 
1007     if ( pLog 
!= NULL 
&& pLog
->HasPendingMessages() ) 
1011     // Send OnIdle events to all windows 
1012     if ( SendIdleEvents() ) 
1014         // SendIdleEvents() returns TRUE if at least one window requested more 
1016         event
.RequestMore(TRUE
); 
1019     // If they are pending events, we must process them: pending events are 
1020     // either events to the threads other than main or events posted with 
1021     // wxPostEvent() functions 
1022     ProcessPendingEvents(); 
1027 // Send idle event to all top-level windows 
1028 bool wxApp::SendIdleEvents() 
1030     bool needMore 
= FALSE
; 
1032     wxWindowList::Node
* node 
= wxTopLevelWindows
.GetFirst(); 
1035         wxWindow
* win 
= node
->GetData(); 
1036         if (SendIdleEvents(win
)) 
1038         node 
= node
->GetNext(); 
1044 // Send idle event to window and all subwindows 
1045 bool wxApp::SendIdleEvents(wxWindow
* win
) 
1047     bool needMore 
= FALSE
; 
1050     event
.SetEventObject(win
); 
1051     win
->GetEventHandler()->ProcessEvent(event
); 
1053     if (event
.MoreRequested()) 
1056     wxNode
* node 
= win
->GetChildren().First(); 
1059         wxWindow
* win 
= (wxWindow
*) node
->Data(); 
1060         if (SendIdleEvents(win
)) 
1063         node 
= node
->Next(); 
1068 void wxApp::DeletePendingObjects() 
1070     wxNode 
*node 
= wxPendingDelete
.First(); 
1073         wxObject 
*obj 
= (wxObject 
*)node
->Data(); 
1077         if (wxPendingDelete
.Member(obj
)) 
1080         // Deleting one object may have deleted other pending 
1081         // objects, so start from beginning of list again. 
1082         node 
= wxPendingDelete
.First(); 
1086 void wxApp::OnEndSession(wxCloseEvent
& WXUNUSED(event
)) 
1089         GetTopWindow()->Close(TRUE
); 
1092 // Default behaviour: close the application with prompts. The 
1093 // user can veto the close, and therefore the end session. 
1094 void wxApp::OnQueryEndSession(wxCloseEvent
& event
) 
1098         if (!GetTopWindow()->Close(!event
.CanVeto())) 
1104 int wxApp::GetComCtl32Version() 
1107     static int s_verComCtl32 
= -1; 
1109     wxCRIT_SECT_DECLARE(csComCtl32
); 
1110     wxCRIT_SECT_LOCKER(lock
, csComCtl32
); 
1112     if ( s_verComCtl32 
== -1 ) 
1114         // initally assume no comctl32.dll at all 
1118         HMODULE hModuleComCtl32 
= ::GetModuleHandle(wxT("COMCTL32")); 
1120         // if so, then we can check for the version 
1121         if ( hModuleComCtl32 
) 
1123             // try to use DllGetVersion() if available in _headers_ 
1124             #ifdef DLLVER_PLATFORM_WINDOWS // defined in shlwapi.h 
1125                 DLLGETVERSIONPROC pfnDllGetVersion 
= (DLLGETVERSIONPROC
) 
1126                     ::GetProcAddress(hModuleComCtl32
, _T("DllGetVersion")); 
1127                 if ( pfnDllGetVersion 
) 
1130                     dvi
.cbSize 
= sizeof(dvi
); 
1132                     HRESULT hr 
= (*pfnDllGetVersion
)(&dvi
); 
1135                         wxLogApiError(_T("DllGetVersion"), hr
); 
1139                         // this is incompatible with _WIN32_IE values, but 
1140                         // compatible with the other values returned by 
1141                         // GetComCtl32Version() 
1142                         s_verComCtl32 
= 100*dvi
.dwMajorVersion 
+ 
1147                 // DllGetVersion() unavailable either during compile or 
1148                 // run-time, try to guess the version otherwise 
1149                 if ( !s_verComCtl32 
) 
1151                     // InitCommonControlsEx is unique to 4.70 and later 
1152                     FARPROC theProc 
= ::GetProcAddress
 
1155                                          _T("InitCommonControlsEx") 
1160                         // not found, must be 4.00 
1161                         s_verComCtl32 
= 400; 
1165                         // many symbols appeared in comctl32 4.71, could use 
1166                         // any of them except may be DllInstall 
1167                         theProc 
= ::GetProcAddress
 
1170                                      _T("InitializeFlatSB") 
1174                             // not found, must be 4.70 
1175                             s_verComCtl32 
= 470; 
1179                             // found, must be 4.71 
1180                             s_verComCtl32 
= 471; 
1187     return s_verComCtl32
; 
1192     wxLogError(_("Fatal error: exiting")); 
1197 // Yield to incoming messages 
1200     // we don't want to process WM_QUIT from here - it should be processed in 
1201     // the main event loop in order to stop it 
1204     while ( PeekMessage(&msg
, (HWND
)0, 0, 0, PM_NOREMOVE
) && 
1205             msg
.message 
!= WM_QUIT 
) 
1207         if ( !wxTheApp
->DoMessage() ) 
1211     // If they are pending events, we must process them. 
1212     wxTheApp
->ProcessPendingEvents(); 
1217 //----------------------------------------------------------------------------- 
1219 //----------------------------------------------------------------------------- 
1223     // Send the top window a dummy message so idle handler processing will 
1224     // start up again.  Doing it this way ensures that the idle handler 
1225     // wakes up in the right thread (see also wxWakeUpMainThread() which does 
1226     // the same for the main app thread only) 
1227     wxWindow 
*topWindow 
= wxTheApp
->GetTopWindow(); 
1230         if ( !::PostMessage(GetHwndOf(topWindow
), WM_NULL
, 0, 0) ) 
1232             // should never happen 
1233             wxLogLastError("PostMessage(WM_NULL)"); 
1238 //----------------------------------------------------------------------------- 
1241 wxApp::GetStdIcon(int which
) const 
1245         case wxICON_INFORMATION
: 
1246             return wxIcon("wxICON_INFO"); 
1248         case wxICON_QUESTION
: 
1249             return wxIcon("wxICON_QUESTION"); 
1251         case wxICON_EXCLAMATION
: 
1252             return wxIcon("wxICON_WARNING"); 
1255             wxFAIL_MSG(wxT("requested non existent standard icon")); 
1256             // still fall through 
1259             return wxIcon("wxICON_ERROR"); 
1263 // For some reason, with MSVC++ 1.5, WinMain isn't linked in properly 
1264 // if in a separate file. So include it here to ensure it's linked. 
1265 #if (defined(__VISUALC__) && !defined(__WIN32__)) || (defined(__GNUWIN32__) && !defined(__TWIN32__))