1 /////////////////////////////////////////////////////////////////////////////// 
   2 // Name:        msw/toplevel.cpp 
   3 // Purpose:     implements wxTopLevelWindow for MSW 
   4 // Author:      Vadim Zeitlin 
   8 // Copyright:   (c) 2001 SciTech Software, Inc. (www.scitechsoft.com) 
   9 // License:     wxWindows licence 
  10 /////////////////////////////////////////////////////////////////////////////// 
  12 // ============================================================================ 
  14 // ============================================================================ 
  16 // ---------------------------------------------------------------------------- 
  18 // ---------------------------------------------------------------------------- 
  20 #if defined(__GNUG__) && !defined(NO_GCC_PRAGMA) 
  21     #pragma implementation "toplevel.h" 
  24 // For compilers that support precompilation, includes "wx.h". 
  25 #include "wx/wxprec.h" 
  33     #include "wx/toplevel.h" 
  34     #include "wx/string.h" 
  38     #include "wx/containr.h"        // wxSetFocusToChild() 
  41 #include "wx/module.h" 
  43 #include "wx/msw/private.h" 
  44 #if defined(__WXWINCE__) 
  50 #include "wx/msw/wince/missing.h" 
  53 #include "wx/msw/missing.h" 
  54 #include "wx/msw/winundef.h" 
  56 // This can't be undefed in winundef.h or 
  57 // there are further errors 
  58 #if defined(__WXWINCE__) && defined(CreateDialog) 
  62 #include "wx/display.h" 
  72 // ---------------------------------------------------------------------------- 
  73 // stubs for missing functions under MicroWindows 
  74 // ---------------------------------------------------------------------------- 
  78 // static inline bool IsIconic(HWND WXUNUSED(hwnd)) { return FALSE; } 
  79 static inline bool IsZoomed(HWND 
WXUNUSED(hwnd
)) { return FALSE
; } 
  81 #endif // __WXMICROWIN__ 
  83 // NB: wxDlgProc must be defined here and not in dialog.cpp because the latter 
  84 //     is not included by wxUniv build which does need wxDlgProc 
  86 wxDlgProc(HWND hDlg
, UINT message
, WPARAM wParam
, LPARAM lParam
); 
  88 // ---------------------------------------------------------------------------- 
  90 // ---------------------------------------------------------------------------- 
  92 // list of all frames and modeless dialogs 
  93 wxWindowList wxModelessWindows
; 
  95 // the name of the default wxWindows class 
  96 extern const wxChar 
*wxCanvasClassName
; 
  98 // ---------------------------------------------------------------------------- 
  99 // wxTLWHiddenParentModule: used to manage the hidden parent window (we need a 
 100 // module to ensure that the window is always deleted) 
 101 // ---------------------------------------------------------------------------- 
 103 class wxTLWHiddenParentModule 
: public wxModule
 
 106     // module init/finalize 
 107     virtual bool OnInit(); 
 108     virtual void OnExit(); 
 110     // get the hidden window (creates on demand) 
 111     static HWND 
GetHWND(); 
 114     // the HWND of the hidden parent 
 117     // the class used to create it 
 118     static const wxChar 
*ms_className
; 
 120     DECLARE_DYNAMIC_CLASS(wxTLWHiddenParentModule
) 
 123 IMPLEMENT_DYNAMIC_CLASS(wxTLWHiddenParentModule
, wxModule
) 
 125 // ============================================================================ 
 126 // wxTopLevelWindowMSW implementation 
 127 // ============================================================================ 
 129 BEGIN_EVENT_TABLE(wxTopLevelWindowMSW
, wxTopLevelWindowBase
) 
 130     EVT_ACTIVATE(wxTopLevelWindowMSW::OnActivate
) 
 133 // ---------------------------------------------------------------------------- 
 134 // wxTopLevelWindowMSW creation 
 135 // ---------------------------------------------------------------------------- 
 137 void wxTopLevelWindowMSW::Init() 
 140     m_maximizeOnShow 
= FALSE
; 
 142     // unlike (almost?) all other windows, frames are created hidden 
 145     // Data to save/restore when calling ShowFullScreen 
 147     m_fsOldWindowStyle 
= 0; 
 148     m_fsIsMaximized 
= FALSE
; 
 149     m_fsIsShowing 
= FALSE
; 
 151     m_winLastFocused 
= (wxWindow 
*)NULL
; 
 154 WXDWORD 
wxTopLevelWindowMSW::MSWGetStyle(long style
, WXDWORD 
*exflags
) const 
 156     // let the base class deal with the common styles but fix the ones which 
 157     // don't make sense for us (we also deal with the borders ourselves) 
 158     WXDWORD msflags 
= wxWindow::MSWGetStyle
 
 160                         (style 
& ~wxBORDER_MASK
) | wxBORDER_NONE
, exflags
 
 161                       ) & ~WS_CHILD 
& ~WS_VISIBLE
; 
 163     // first select the kind of window being created 
 165     // note that if we don't set WS_POPUP, Windows assumes WS_OVERLAPPED and 
 166     // creates a window with both caption and border, hence we also test it 
 167     // below in some other cases 
 168     if ( style 
& wxFRAME_TOOL_WINDOW 
) 
 173         if (msflags 
& WS_BORDER
) 
 175             msflags 
|= WS_OVERLAPPED
; 
 178     // border and caption styles 
 179     if ( style 
& wxRESIZE_BORDER 
) 
 182         msflags 
|= WS_THICKFRAME
; 
 185     else if ( exflags 
&& ((style 
& wxBORDER_DOUBLE
) || (style 
& wxBORDER_RAISED
)) ) 
 186         *exflags 
|= WS_EX_DLGMODALFRAME
; 
 187     else if ( !(style 
& wxBORDER_NONE
) ) 
 188         msflags 
|= WS_BORDER
; 
 194     if ( style 
& wxCAPTION 
) 
 195         msflags 
|= WS_CAPTION
; 
 201     // next translate the individual flags 
 202     if ( style 
& wxMINIMIZE_BOX 
) 
 203         msflags 
|= WS_MINIMIZEBOX
; 
 204     if ( style 
& wxMAXIMIZE_BOX 
) 
 205         msflags 
|= WS_MAXIMIZEBOX
; 
 206     if ( style 
& wxSYSTEM_MENU 
) 
 207         msflags 
|= WS_SYSMENU
; 
 209     if ( style 
& wxMINIMIZE 
) 
 210         msflags 
|= WS_MINIMIZE
; 
 211     if ( style 
& wxMAXIMIZE 
) 
 212         msflags 
|= WS_MAXIMIZE
; 
 215     // Keep this here because it saves recoding this function in wxTinyFrame 
 216 #if wxUSE_ITSY_BITSY && !defined(__WIN32__) 
 217     if ( style 
& wxTINY_CAPTION_VERT 
) 
 218         msflags 
|= IBS_VERTCAPTION
; 
 219     if ( style 
& wxTINY_CAPTION_HORIZ 
) 
 220         msflags 
|= IBS_HORZCAPTION
; 
 222     if ( style 
& (wxTINY_CAPTION_VERT 
| wxTINY_CAPTION_HORIZ
) ) 
 223         msflags 
|= WS_CAPTION
; 
 228 #if !defined(__WIN16__) 
 229         if ( !(GetExtraStyle() & wxTOPLEVEL_EX_DIALOG
) ) 
 231             if ( style 
& wxFRAME_TOOL_WINDOW 
) 
 233                 // create the palette-like window 
 234                 *exflags 
|= WS_EX_TOOLWINDOW
; 
 236                 // tool windows shouldn't appear on the taskbar (as documented) 
 237                 style 
|= wxFRAME_NO_TASKBAR
; 
 240             // We have to solve 2 different problems here: 
 242             // 1. frames with wxFRAME_NO_TASKBAR flag shouldn't appear in the 
 243             //    taskbar even if they don't have a parent 
 245             // 2. frames without this style should appear in the taskbar even 
 246             //    if they're owned (Windows only puts non owned windows into 
 247             //    the taskbar normally) 
 249             // The second one is solved here by using WS_EX_APPWINDOW flag, the 
 250             // first one is dealt with in our MSWGetParent() method 
 253             if ( !(style 
& wxFRAME_NO_TASKBAR
) && GetParent() ) 
 255                 // need to force the frame to appear in the taskbar 
 256                 *exflags 
|= WS_EX_APPWINDOW
; 
 259             //else: nothing to do [here] 
 263         if ( style 
& wxSTAY_ON_TOP 
) 
 264             *exflags 
|= WS_EX_TOPMOST
; 
 267         if ( GetExtraStyle() & wxFRAME_EX_CONTEXTHELP 
) 
 268             *exflags 
|= WS_EX_CONTEXTHELP
; 
 275 WXHWND 
wxTopLevelWindowMSW::MSWGetParent() const 
 277     // for the frames without wxFRAME_FLOAT_ON_PARENT style we should use NULL 
 278     // parent HWND or it would be always on top of its parent which is not what 
 279     // we usually want (in fact, we only want it for frames with the 
 280     // wxFRAME_FLOAT_ON_PARENT flag) 
 281     HWND hwndParent 
= NULL
; 
 282     if ( HasFlag(wxFRAME_FLOAT_ON_PARENT
) ) 
 284         const wxWindow 
*parent 
= GetParent(); 
 288             // this flag doesn't make sense then and will be ignored 
 289             wxFAIL_MSG( _T("wxFRAME_FLOAT_ON_PARENT but no parent?") ); 
 293             hwndParent 
= GetHwndOf(parent
); 
 296     //else: don't float on parent, must not be owned 
 298     // now deal with the 2nd taskbar-related problem (see comments above in 
 300     if ( HasFlag(wxFRAME_NO_TASKBAR
) && !hwndParent 
) 
 303         hwndParent 
= wxTLWHiddenParentModule::GetHWND(); 
 306     return (WXHWND
)hwndParent
; 
 309 bool wxTopLevelWindowMSW::CreateDialog(const void *dlgTemplate
, 
 310                                        const wxString
& title
, 
 314 #ifdef __WXMICROWIN__ 
 315     // no dialogs support under MicroWin yet 
 316     return CreateFrame(title
, pos
, size
); 
 317 #else // !__WXMICROWIN__ 
 318     wxWindow 
*parent 
= GetParent(); 
 320     // for the dialogs without wxDIALOG_NO_PARENT style, use the top level 
 321     // app window as parent - this avoids creating modal dialogs without 
 323     if ( !parent 
&& !(GetWindowStyleFlag() & wxDIALOG_NO_PARENT
) ) 
 325         parent 
= wxTheApp
->GetTopWindow(); 
 329             // don't use transient windows as parents, this is dangerous as it 
 330             // can lead to a crash if the parent is destroyed before the child 
 332             // also don't use the window which is currently hidden as then the 
 333             // dialog would be hidden as well 
 334             if ( (parent
->GetExtraStyle() & wxWS_EX_TRANSIENT
) || 
 342     m_hWnd 
= (WXHWND
)::CreateDialogIndirect
 
 345                         (DLGTEMPLATE
*)dlgTemplate
, 
 346                         parent 
? GetHwndOf(parent
) : NULL
, 
 352         wxFAIL_MSG(wxT("Failed to create dialog. Incorrect DLGTEMPLATE?")); 
 354         wxLogSysError(wxT("Can't create dialog using memory template")); 
 360     (void)MSWGetCreateWindowFlags(&exflags
); 
 364         ::SetWindowLong(GetHwnd(), GWL_EXSTYLE
, exflags
); 
 365         ::SetWindowPos(GetHwnd(), 
 366                        exflags 
& WS_EX_TOPMOST 
? HWND_TOPMOST 
: 0, 
 370                        (exflags 
& WS_EX_TOPMOST 
? 0 : SWP_NOZORDER
) | 
 374 #if defined(__WIN95__) 
 375     // For some reason, the system menu is activated when we use the 
 376     // WS_EX_CONTEXTHELP style, so let's set a reasonable icon 
 377     if ( exflags 
& WS_EX_CONTEXTHELP 
) 
 379         wxFrame 
*winTop 
= wxDynamicCast(wxTheApp
->GetTopWindow(), wxFrame
); 
 382             wxIcon icon 
= winTop
->GetIcon(); 
 385                 ::SendMessage(GetHwnd(), WM_SETICON
, 
 387                               (LPARAM
)GetHiconOf(icon
)); 
 393     // move the dialog to its initial position without forcing repainting 
 395     (void)MSWGetCreateWindowCoords(pos
, size
, x
, y
, w
, h
); 
 397     if ( x 
== (int)CW_USEDEFAULT 
) 
 399         // centre it on the screen - what else can we do? 
 400         wxSize sizeDpy 
= wxGetDisplaySize(); 
 402         x 
= (sizeDpy
.x 
- w
) / 2; 
 403         y 
= (sizeDpy
.y 
- h
) / 2; 
 406     if ( !::MoveWindow(GetHwnd(), x
, y
, w
, h
, FALSE
) ) 
 408         wxLogLastError(wxT("MoveWindow")); 
 411     if ( !title
.empty() ) 
 413         ::SetWindowText(GetHwnd(), title
); 
 419 #endif // __WXMICROWIN__/!__WXMICROWIN__ 
 422 bool wxTopLevelWindowMSW::CreateFrame(const wxString
& title
, 
 427     WXDWORD flags 
= MSWGetCreateWindowFlags(&exflags
); 
 429     return MSWCreate(wxCanvasClassName
, title
, pos
, size
, flags
, exflags
); 
 432 bool wxTopLevelWindowMSW::Create(wxWindow 
*parent
, 
 434                                  const wxString
& title
, 
 438                                  const wxString
& name
) 
 440     bool ret 
wxDUMMY_INITIALIZE(false); 
 445     w 
= WidthDefault(size
.x
); 
 446     h 
= HeightDefault(size
.y
); 
 448     m_windowStyle 
= style
; 
 452     m_windowId 
= id 
== -1 ? NewControlId() : id
; 
 454     wxTopLevelWindows
.Append(this); 
 457         parent
->AddChild(this); 
 459     if ( GetExtraStyle() & wxTOPLEVEL_EX_DIALOG 
) 
 461         // we have different dialog templates to allows creation of dialogs 
 462         // with & without captions under MSWindows, resizeable or not (but a 
 463         // resizeable dialog always has caption - otherwise it would look too 
 466         // we need 3 additional WORDs for dialog menu, class and title (as we 
 467         // don't use DS_SETFONT we don't need the fourth WORD for the font) 
 468         static const int dlgsize 
= sizeof(DLGTEMPLATE
) + (sizeof(WORD
) * 3); 
 469         DLGTEMPLATE 
*dlgTemplate 
= (DLGTEMPLATE 
*)malloc(dlgsize
); 
 470         memset(dlgTemplate
, 0, dlgsize
); 
 472         // these values are arbitrary, they won't be used normally anyhow 
 475         dlgTemplate
->cx 
= 144; 
 476         dlgTemplate
->cy 
= 75; 
 478         // reuse the code in MSWGetStyle() but correct the results slightly for 
 480         dlgTemplate
->style 
= MSWGetStyle(style
, NULL
); 
 482         // all dialogs are popups 
 483         dlgTemplate
->style 
|= WS_POPUP
; 
 485         // force 3D-look if necessary, it looks impossibly ugly otherwise 
 486         if ( style 
& (wxRESIZE_BORDER 
| wxCAPTION
) ) 
 487             dlgTemplate
->style 
|= DS_MODALFRAME
; 
 489         ret 
= CreateDialog(dlgTemplate
, title
, pos
, wxSize(w
,h
)); 
 494         ret 
= CreateFrame(title
, pos
, wxSize(w
,h
)); 
 497     if ( ret 
&& !(GetWindowStyleFlag() & wxCLOSE_BOX
) ) 
 499         EnableCloseButton(false); 
 502     // for some reason we need to manually send ourselves this message as 
 503     // otherwise the mnemonics are always shown -- even if they're configured 
 504     // to be hidden until "Alt" is pressed in the control panel 
 506     // this could indicate a bug somewhere else but for now this is the only 
 514             MAKEWPARAM(UIS_INITIALIZE
, UISF_HIDEFOCUS 
| UISF_HIDEACCEL
), 
 522 wxTopLevelWindowMSW::~wxTopLevelWindowMSW() 
 524     if ( wxModelessWindows
.Find(this) ) 
 525         wxModelessWindows
.DeleteObject(this); 
 527     // after destroying an owned window, Windows activates the next top level 
 528     // window in Z order but it may be different from our owner (to reproduce 
 529     // this simply Alt-TAB to another application and back before closing the 
 530     // owned frame) whereas we always want to yield activation to our parent 
 531     if ( HasFlag(wxFRAME_FLOAT_ON_PARENT
) ) 
 533         wxWindow 
*parent 
= GetParent(); 
 536             ::BringWindowToTop(GetHwndOf(parent
)); 
 541 // ---------------------------------------------------------------------------- 
 542 // wxTopLevelWindowMSW showing 
 543 // ---------------------------------------------------------------------------- 
 545 void wxTopLevelWindowMSW::DoShowWindow(int nShowCmd
) 
 547     ::ShowWindow(GetHwnd(), nShowCmd
); 
 549     m_iconized 
= nShowCmd 
== SW_MINIMIZE
; 
 552 bool wxTopLevelWindowMSW::Show(bool show
) 
 554     // don't use wxWindow version as we want to call DoShowWindow() ourselves 
 555     if ( !wxWindowBase::Show(show
) ) 
 561         if ( m_maximizeOnShow 
) 
 564             nShowCmd 
= SW_MAXIMIZE
; 
 566             m_maximizeOnShow 
= FALSE
; 
 570            if ( GetWindowStyle() & wxFRAME_TOOL_WINDOW 
) 
 571                nShowCmd 
= SW_SHOWNA
; 
 581     DoShowWindow(nShowCmd
); 
 585         ::BringWindowToTop(GetHwnd()); 
 587         wxActivateEvent 
event(wxEVT_ACTIVATE
, TRUE
, m_windowId
); 
 588         event
.SetEventObject( this ); 
 589         GetEventHandler()->ProcessEvent(event
); 
 593         // Try to highlight the correct window (the parent) 
 596             HWND hWndParent 
= GetHwndOf(GetParent()); 
 598                 ::BringWindowToTop(hWndParent
); 
 605 // ---------------------------------------------------------------------------- 
 606 // wxTopLevelWindowMSW maximize/minimize 
 607 // ---------------------------------------------------------------------------- 
 609 void wxTopLevelWindowMSW::Maximize(bool maximize
) 
 613         // just maximize it directly 
 614         DoShowWindow(maximize 
? SW_MAXIMIZE 
: SW_RESTORE
); 
 618         // we can't maximize the hidden frame because it shows it as well, so 
 619         // just remember that we should do it later in this case 
 620         m_maximizeOnShow 
= maximize
; 
 624 bool wxTopLevelWindowMSW::IsMaximized() const 
 629     return ::IsZoomed(GetHwnd()) != 0; 
 633 void wxTopLevelWindowMSW::Iconize(bool iconize
) 
 635     DoShowWindow(iconize 
? SW_MINIMIZE 
: SW_RESTORE
); 
 638 bool wxTopLevelWindowMSW::IsIconized() const 
 643     // also update the current state 
 644     ((wxTopLevelWindowMSW 
*)this)->m_iconized 
= ::IsIconic(GetHwnd()) != 0; 
 650 void wxTopLevelWindowMSW::Restore() 
 652     DoShowWindow(SW_RESTORE
); 
 655 // ---------------------------------------------------------------------------- 
 656 // wxTopLevelWindowMSW fullscreen 
 657 // ---------------------------------------------------------------------------- 
 659 bool wxTopLevelWindowMSW::ShowFullScreen(bool show
, long style
) 
 661     if ( show 
== IsFullScreen() ) 
 667     m_fsIsShowing 
= show
; 
 673         // zap the frame borders 
 675         // save the 'normal' window style 
 676         m_fsOldWindowStyle 
= GetWindowLong(GetHwnd(), GWL_STYLE
); 
 678         // save the old position, width & height, maximize state 
 679         m_fsOldSize 
= GetRect(); 
 680         m_fsIsMaximized 
= IsMaximized(); 
 682         // decide which window style flags to turn off 
 683         LONG newStyle 
= m_fsOldWindowStyle
; 
 686         if (style 
& wxFULLSCREEN_NOBORDER
) 
 688             offFlags 
|= WS_BORDER
; 
 690             offFlags 
|= WS_THICKFRAME
; 
 693         if (style 
& wxFULLSCREEN_NOCAPTION
) 
 694             offFlags 
|= WS_CAPTION 
| WS_SYSMENU
; 
 696         newStyle 
&= ~offFlags
; 
 698         // change our window style to be compatible with full-screen mode 
 699         ::SetWindowLong(GetHwnd(), GWL_STYLE
, newStyle
); 
 703         // resize to the size of the display containing us 
 704         int dpy 
= wxDisplay::GetFromWindow(this); 
 705         if ( dpy 
!= wxNOT_FOUND 
) 
 707             rect 
= wxDisplay(dpy
).GetGeometry(); 
 709         else // fall back to the main desktop 
 710 #else // wxUSE_DISPLAY 
 712             // resize to the size of the desktop 
 713             wxCopyRECTToRect(wxGetWindowRect(::GetDesktopWindow()), rect
); 
 715             // FIXME: size of the bottom menu (toolbar) 
 716             // should be taken in account 
 717             rect
.height 
+= rect
.y
; 
 721 #endif // wxUSE_DISPLAY 
 725         // now flush the window style cache and actually go full-screen 
 726         long flags 
= SWP_FRAMECHANGED
; 
 728         // showing the frame full screen should also show it if it's still 
 732             // don't call wxWindow version to avoid flicker from calling 
 733             // ::ShowWindow() -- we're going to show the window at the correct 
 734             // location directly below -- but do call the wxWindowBase version 
 735             // to sync the internal m_isShown flag 
 736             wxWindowBase::Show(); 
 738             flags 
|= SWP_SHOWWINDOW
; 
 741         SetWindowPos(GetHwnd(), HWND_TOP
, 
 742                      rect
.x
, rect
.y
, rect
.width
, rect
.height
, 
 745 #if defined(__WXWINCE__) && _WIN32_WCE < 400 
 746         ::SHFullScreen(GetHwnd(), SHFS_HIDETASKBAR 
| SHFS_HIDESIPBUTTON
); 
 749         // finally send an event allowing the window to relayout itself &c 
 750         wxSizeEvent 
event(rect
.GetSize(), GetId()); 
 751         GetEventHandler()->ProcessEvent(event
); 
 753     else // stop showing full screen 
 755 #if defined(__WXWINCE__) && _WIN32_WCE < 400 
 756         ::SHFullScreen(GetHwnd(), SHFS_SHOWTASKBAR 
| SHFS_SHOWSIPBUTTON
); 
 758         Maximize(m_fsIsMaximized
); 
 759         SetWindowLong(GetHwnd(),GWL_STYLE
, m_fsOldWindowStyle
); 
 760         SetWindowPos(GetHwnd(),HWND_TOP
,m_fsOldSize
.x
, m_fsOldSize
.y
, 
 761             m_fsOldSize
.width
, m_fsOldSize
.height
, SWP_FRAMECHANGED
); 
 767 // ---------------------------------------------------------------------------- 
 768 // wxTopLevelWindowMSW misc 
 769 // ---------------------------------------------------------------------------- 
 771 void wxTopLevelWindowMSW::SetIcon(const wxIcon
& icon
) 
 773     SetIcons( wxIconBundle( icon 
) ); 
 776 void wxTopLevelWindowMSW::SetIcons(const wxIconBundle
& icons
) 
 778     wxTopLevelWindowBase::SetIcons(icons
); 
 780 #if defined(__WIN95__) && !defined(__WXMICROWIN__) 
 781     const wxIcon
& sml 
= icons
.GetIcon( wxSize( 16, 16 ) ); 
 782     if( sml
.Ok() && sml
.GetWidth() == 16 && sml
.GetHeight() == 16 ) 
 784         ::SendMessage( GetHwndOf( this ), WM_SETICON
, ICON_SMALL
, 
 785                        (LPARAM
)GetHiconOf(sml
) ); 
 788     const wxIcon
& big 
= icons
.GetIcon( wxSize( 32, 32 ) ); 
 789     if( big
.Ok() && big
.GetWidth() == 32 && big
.GetHeight() == 32 ) 
 791         ::SendMessage( GetHwndOf( this ), WM_SETICON
, ICON_BIG
, 
 792                        (LPARAM
)GetHiconOf(big
) ); 
 797 bool wxTopLevelWindowMSW::EnableCloseButton(bool enable
) 
 799 #if !defined(__WXMICROWIN__) 
 800     // get system (a.k.a. window) menu 
 801     HMENU hmenu 
= GetSystemMenu(GetHwnd(), FALSE 
/* get it */); 
 804         // no system menu at all -- ok if we want to remove the close button 
 805         // anyhow, but bad if we want to show it 
 809     // enabling/disabling the close item from it also automatically 
 810     // disables/enables the close title bar button 
 811     if ( ::EnableMenuItem(hmenu
, SC_CLOSE
, 
 813                           (enable 
? MF_ENABLED 
: MF_GRAYED
)) == -1 ) 
 815         wxLogLastError(_T("EnableMenuItem(SC_CLOSE)")); 
 820     // update appearance immediately 
 821     if ( !::DrawMenuBar(GetHwnd()) ) 
 823         wxLogLastError(_T("DrawMenuBar")); 
 825 #endif // !__WXMICROWIN__ 
 830 bool wxTopLevelWindowMSW::SetShape(const wxRegion
& region
) 
 835     wxCHECK_MSG( HasFlag(wxFRAME_SHAPED
), FALSE
, 
 836                  _T("Shaped windows must be created with the wxFRAME_SHAPED style.")); 
 838     // The empty region signifies that the shape should be removed from the 
 840     if ( region
.IsEmpty() ) 
 842         if (::SetWindowRgn(GetHwnd(), NULL
, TRUE
) == 0) 
 844             wxLogLastError(_T("SetWindowRgn")); 
 850     // Windows takes ownership of the region, so 
 851     // we'll have to make a copy of the region to give to it. 
 852     DWORD noBytes 
= ::GetRegionData(GetHrgnOf(region
), 0, NULL
); 
 853     RGNDATA 
*rgnData 
= (RGNDATA
*) new char[noBytes
]; 
 854     ::GetRegionData(GetHrgnOf(region
), noBytes
, rgnData
); 
 855     HRGN hrgn 
= ::ExtCreateRegion(NULL
, noBytes
, rgnData
); 
 856     delete[] (char*) rgnData
; 
 858     // SetWindowRgn expects the region to be in coordinants 
 859     // relative to the window, not the client area.  Figure 
 860     // out the offset, if any. 
 862     DWORD dwStyle 
=   ::GetWindowLong(GetHwnd(), GWL_STYLE
); 
 863     DWORD dwExStyle 
= ::GetWindowLong(GetHwnd(), GWL_EXSTYLE
); 
 864     ::GetClientRect(GetHwnd(), &rect
); 
 865     ::AdjustWindowRectEx(&rect
, dwStyle
, FALSE
, dwExStyle
); 
 866     ::OffsetRgn(hrgn
, -rect
.left
, -rect
.top
); 
 868     // Now call the shape API with the new region. 
 869     if (::SetWindowRgn(GetHwnd(), hrgn
, TRUE
) == 0) 
 871         wxLogLastError(_T("SetWindowRgn")); 
 878 // ---------------------------------------------------------------------------- 
 879 // wxTopLevelWindow event handling 
 880 // ---------------------------------------------------------------------------- 
 882 // Default activation behaviour - set the focus for the first child 
 884 void wxTopLevelWindowMSW::OnActivate(wxActivateEvent
& event
) 
 886     if ( event
.GetActive() ) 
 888         // restore focus to the child which was last focused unless we already 
 890         wxLogTrace(_T("focus"), _T("wxTLW %08x activated."), (int) m_hWnd
); 
 892         wxWindow 
*winFocus 
= FindFocus(); 
 893         if ( !winFocus 
|| wxGetTopLevelParent(winFocus
) != this ) 
 895             wxWindow 
*parent 
= m_winLastFocused 
? m_winLastFocused
->GetParent() 
 902             wxSetFocusToChild(parent
, &m_winLastFocused
); 
 907         // remember the last focused child if it is our child 
 908         m_winLastFocused 
= FindFocus(); 
 910         if ( m_winLastFocused 
) 
 912             // let it know that it doesn't have focus any more 
 913             m_winLastFocused
->HandleKillFocus((WXHWND
)NULL
); 
 915             // and don't remember it if it's a child from some other frame 
 916             if ( wxGetTopLevelParent(m_winLastFocused
) != this ) 
 918                 m_winLastFocused 
= NULL
; 
 922         wxLogTrace(_T("focus"), 
 923                    _T("wxTLW %08x deactivated, last focused: %08x."), 
 925                    (int) (m_winLastFocused 
? GetHwndOf(m_winLastFocused
) 
 932 // the DialogProc for all wxWindows dialogs 
 933 LONG APIENTRY _EXPORT
 
 934 wxDlgProc(HWND 
WXUNUSED(hDlg
), 
 936           WPARAM 
WXUNUSED(wParam
), 
 937           LPARAM 
WXUNUSED(lParam
)) 
 942             // for this message, returning TRUE tells system to set focus to 
 943             // the first control in the dialog box, but as we set the focus 
 944             // ourselves, we return FALSE from here as well, so fall through 
 947             // for all the other ones, FALSE means that we didn't process the 
 953 // ============================================================================ 
 954 // wxTLWHiddenParentModule implementation 
 955 // ============================================================================ 
 957 HWND 
wxTLWHiddenParentModule::ms_hwnd 
= NULL
; 
 959 const wxChar 
*wxTLWHiddenParentModule::ms_className 
= NULL
; 
 961 bool wxTLWHiddenParentModule::OnInit() 
 969 void wxTLWHiddenParentModule::OnExit() 
 973         if ( !::DestroyWindow(ms_hwnd
) ) 
 975             wxLogLastError(_T("DestroyWindow(hidden TLW parent)")); 
 983         if ( !::UnregisterClass(ms_className
, wxGetInstance()) ) 
 985             wxLogLastError(_T("UnregisterClass(\"wxTLWHiddenParent\")")); 
 993 HWND 
wxTLWHiddenParentModule::GetHWND() 
 999             static const wxChar 
*HIDDEN_PARENT_CLASS 
= _T("wxTLWHiddenParent"); 
1002             wxZeroMemory(wndclass
); 
1004             wndclass
.lpfnWndProc   
= DefWindowProc
; 
1005             wndclass
.hInstance     
= wxGetInstance(); 
1006             wndclass
.lpszClassName 
= HIDDEN_PARENT_CLASS
; 
1008             if ( !::RegisterClass(&wndclass
) ) 
1010                 wxLogLastError(_T("RegisterClass(\"wxTLWHiddenParent\")")); 
1014                 ms_className 
= HIDDEN_PARENT_CLASS
; 
1018         ms_hwnd 
= ::CreateWindow(ms_className
, wxEmptyString
, 0, 0, 0, 0, 0, NULL
, 
1019                                  (HMENU
)NULL
, wxGetInstance(), NULL
); 
1022             wxLogLastError(_T("CreateWindow(hidden TLW parent)"));