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/dialog.h" 
  35     #include "wx/string.h" 
  39     #include "wx/containr.h"        // wxSetFocusToChild() 
  42 #include "wx/module.h" 
  43 #include "wx/dynlib.h" 
  45 #include "wx/msw/private.h" 
  46 #if defined(__WXWINCE__) && !defined(__HANDHELDPC__) 
  49   // Standard SDK doesn't have aygshell.dll: see include/wx/msw/wince/libraries.h 
  50   #if _WIN32_WCE < 400 || !defined(__WINCE_STANDARDSDK__) 
  53 #include "wx/msw/wince/missing.h" 
  56 #include "wx/msw/missing.h" 
  57 #include "wx/msw/winundef.h" 
  59 #include "wx/display.h" 
  69 // ---------------------------------------------------------------------------- 
  70 // stubs for missing functions under MicroWindows 
  71 // ---------------------------------------------------------------------------- 
  75 // static inline bool IsIconic(HWND WXUNUSED(hwnd)) { return false; } 
  76 static inline bool IsZoomed(HWND 
WXUNUSED(hwnd
)) { return false; } 
  78 #endif // __WXMICROWIN__ 
  80 // NB: wxDlgProc must be defined here and not in dialog.cpp because the latter 
  81 //     is not included by wxUniv build which does need wxDlgProc 
  83 wxDlgProc(HWND hDlg
, UINT message
, WPARAM wParam
, LPARAM lParam
); 
  85 // ---------------------------------------------------------------------------- 
  87 // ---------------------------------------------------------------------------- 
  89 // the name of the default wxWidgets class 
  91 extern       wxChar 
*wxCanvasClassName
; 
  93 extern const wxChar 
*wxCanvasClassName
; 
  96 // ---------------------------------------------------------------------------- 
  97 // wxTLWHiddenParentModule: used to manage the hidden parent window (we need a 
  98 // module to ensure that the window is always deleted) 
  99 // ---------------------------------------------------------------------------- 
 101 class wxTLWHiddenParentModule 
: public wxModule
 
 104     // module init/finalize 
 105     virtual bool OnInit(); 
 106     virtual void OnExit(); 
 108     // get the hidden window (creates on demand) 
 109     static HWND 
GetHWND(); 
 112     // the HWND of the hidden parent 
 115     // the class used to create it 
 116     static const wxChar 
*ms_className
; 
 118     DECLARE_DYNAMIC_CLASS(wxTLWHiddenParentModule
) 
 121 IMPLEMENT_DYNAMIC_CLASS(wxTLWHiddenParentModule
, wxModule
) 
 123 // ============================================================================ 
 124 // wxTopLevelWindowMSW implementation 
 125 // ============================================================================ 
 127 BEGIN_EVENT_TABLE(wxTopLevelWindowMSW
, wxTopLevelWindowBase
) 
 128     EVT_ACTIVATE(wxTopLevelWindowMSW::OnActivate
) 
 131 // ---------------------------------------------------------------------------- 
 132 // wxTopLevelWindowMSW creation 
 133 // ---------------------------------------------------------------------------- 
 135 void wxTopLevelWindowMSW::Init() 
 138     m_maximizeOnShow 
= false; 
 140     // Data to save/restore when calling ShowFullScreen 
 142     m_fsOldWindowStyle 
= 0; 
 143     m_fsIsMaximized 
= false; 
 144     m_fsIsShowing 
= false; 
 146     m_winLastFocused 
= (wxWindow 
*)NULL
; 
 148 #ifdef __SMARTPHONE__ 
 153 WXDWORD 
wxTopLevelWindowMSW::MSWGetStyle(long style
, WXDWORD 
*exflags
) const 
 155     // let the base class deal with the common styles but fix the ones which 
 156     // don't make sense for us (we also deal with the borders ourselves) 
 157     WXDWORD msflags 
= wxWindow::MSWGetStyle
 
 159                         (style 
& ~wxBORDER_MASK
) | wxBORDER_NONE
, exflags
 
 160                       ) & ~WS_CHILD 
& ~WS_VISIBLE
; 
 162 #if defined(__WXWINCE__) && _WIN32_WCE < 400 
 163     msflags 
|= WS_VISIBLE
; 
 166     // first select the kind of window being created 
 168     // note that if we don't set WS_POPUP, Windows assumes WS_OVERLAPPED and 
 169     // creates a window with both caption and border, hence we also test it 
 170     // below in some other cases 
 171     if ( style 
& wxFRAME_TOOL_WINDOW 
) 
 175     //else: WS_OVERLAPPED is 0 anyhow, so it is on by default 
 177 #ifndef __SMARTPHONE__ 
 178     // border and caption styles 
 179     if ( style 
& wxRESIZE_BORDER 
) 
 180         msflags 
|= WS_THICKFRAME
; 
 181     else if ( exflags 
&& ((style 
& wxBORDER_DOUBLE
) || (style 
& wxBORDER_RAISED
)) ) 
 182         *exflags 
|= WS_EX_DLGMODALFRAME
; 
 183     else if ( !(style 
& wxBORDER_NONE
) ) 
 184         msflags 
|= WS_BORDER
; 
 189     // normally we consider that all windows without caption must be popups, 
 190     // but CE is an exception: there windows normally do not have the caption 
 191     // but shouldn't be made popups as popups can't have menus and don't look 
 192     // like normal windows anyhow 
 193     if ( style 
& wxCAPTION 
) 
 194         msflags 
|= WS_CAPTION
; 
 198 #endif // !__WXWINCE__ 
 200     // next translate the individual flags 
 201     if ( style 
& wxMINIMIZE_BOX 
) 
 202         msflags 
|= WS_MINIMIZEBOX
; 
 203     if ( style 
& wxMAXIMIZE_BOX 
) 
 204         msflags 
|= WS_MAXIMIZEBOX
; 
 205     if ( style 
& wxSYSTEM_MENU 
) 
 206         msflags 
|= WS_SYSMENU
; 
 208     // NB: under CE these 2 styles are not supported currently, we should 
 209     //     call Minimize()/Maximize() "manually" if we want to support them 
 210     if ( style 
& wxMINIMIZE 
) 
 211         msflags 
|= WS_MINIMIZE
; 
 212     if ( style 
& wxMAXIMIZE 
) 
 213         msflags 
|= WS_MAXIMIZE
; 
 215     // Keep this here because it saves recoding this function in wxTinyFrame 
 216     if ( style 
& (wxTINY_CAPTION_VERT 
| wxTINY_CAPTION_HORIZ
) ) 
 217         msflags 
|= WS_CAPTION
; 
 221         // there is no taskbar under CE, so omit all this 
 222 #if !defined(__WXWINCE__) 
 223         if ( !(GetExtraStyle() & wxTOPLEVEL_EX_DIALOG
) ) 
 225             if ( style 
& wxFRAME_TOOL_WINDOW 
) 
 227                 // create the palette-like window 
 228                 *exflags 
|= WS_EX_TOOLWINDOW
; 
 230                 // tool windows shouldn't appear on the taskbar (as documented) 
 231                 style 
|= wxFRAME_NO_TASKBAR
; 
 234             // We have to solve 2 different problems here: 
 236             // 1. frames with wxFRAME_NO_TASKBAR flag shouldn't appear in the 
 237             //    taskbar even if they don't have a parent 
 239             // 2. frames without this style should appear in the taskbar even 
 240             //    if they're owned (Windows only puts non owned windows into 
 241             //    the taskbar normally) 
 243             // The second one is solved here by using WS_EX_APPWINDOW flag, the 
 244             // first one is dealt with in our MSWGetParent() method 
 246             if ( !(style 
& wxFRAME_NO_TASKBAR
) && GetParent() ) 
 248                 // need to force the frame to appear in the taskbar 
 249                 *exflags 
|= WS_EX_APPWINDOW
; 
 251             //else: nothing to do [here] 
 253 #endif // !__WXWINCE__ 
 255         if ( style 
& wxSTAY_ON_TOP 
) 
 256             *exflags 
|= WS_EX_TOPMOST
; 
 258         if ( GetExtraStyle() & wxFRAME_EX_CONTEXTHELP 
) 
 259             *exflags 
|= WS_EX_CONTEXTHELP
; 
 265 WXHWND 
wxTopLevelWindowMSW::MSWGetParent() const 
 267     // for the frames without wxFRAME_FLOAT_ON_PARENT style we should use NULL 
 268     // parent HWND or it would be always on top of its parent which is not what 
 269     // we usually want (in fact, we only want it for frames with the 
 270     // wxFRAME_FLOAT_ON_PARENT flag) 
 271     HWND hwndParent 
= NULL
; 
 272     if ( HasFlag(wxFRAME_FLOAT_ON_PARENT
) ) 
 274         const wxWindow 
*parent 
= GetParent(); 
 278             // this flag doesn't make sense then and will be ignored 
 279             wxFAIL_MSG( _T("wxFRAME_FLOAT_ON_PARENT but no parent?") ); 
 283             hwndParent 
= GetHwndOf(parent
); 
 286     //else: don't float on parent, must not be owned 
 288     // now deal with the 2nd taskbar-related problem (see comments above in 
 290     if ( HasFlag(wxFRAME_NO_TASKBAR
) && !hwndParent 
) 
 293         hwndParent 
= wxTLWHiddenParentModule::GetHWND(); 
 296     return (WXHWND
)hwndParent
; 
 299 bool wxTopLevelWindowMSW::CreateDialog(const void *dlgTemplate
, 
 300                                        const wxString
& title
, 
 304 #ifdef __WXMICROWIN__ 
 305     // no dialogs support under MicroWin yet 
 306     return CreateFrame(title
, pos
, size
); 
 307 #else // !__WXMICROWIN__ 
 308     wxWindow 
*parent 
= GetParent(); 
 310     // for the dialogs without wxDIALOG_NO_PARENT style, use the top level 
 311     // app window as parent - this avoids creating modal dialogs without 
 313     if ( !parent 
&& !(GetWindowStyleFlag() & wxDIALOG_NO_PARENT
) ) 
 315         parent 
= wxTheApp
->GetTopWindow(); 
 319             // don't use transient windows as parents, this is dangerous as it 
 320             // can lead to a crash if the parent is destroyed before the child 
 322             // also don't use the window which is currently hidden as then the 
 323             // dialog would be hidden as well 
 324             if ( (parent
->GetExtraStyle() & wxWS_EX_TRANSIENT
) || 
 332     m_hWnd 
= (WXHWND
)::CreateDialogIndirect
 
 335                         (DLGTEMPLATE
*)dlgTemplate
, 
 336                         parent 
? GetHwndOf(parent
) : NULL
, 
 342         wxFAIL_MSG(wxT("Failed to create dialog. Incorrect DLGTEMPLATE?")); 
 344         wxLogSysError(wxT("Can't create dialog using memory template")); 
 350     (void)MSWGetCreateWindowFlags(&exflags
); 
 354         ::SetWindowLong(GetHwnd(), GWL_EXSTYLE
, exflags
); 
 355         ::SetWindowPos(GetHwnd(), 
 356                        exflags 
& WS_EX_TOPMOST 
? HWND_TOPMOST 
: 0, 
 360                        (exflags 
& WS_EX_TOPMOST 
? 0 : SWP_NOZORDER
) | 
 364 #if defined(__WIN95__) 
 365     // For some reason, the system menu is activated when we use the 
 366     // WS_EX_CONTEXTHELP style, so let's set a reasonable icon 
 367     if ( exflags 
& WS_EX_CONTEXTHELP 
) 
 369         wxFrame 
*winTop 
= wxDynamicCast(wxTheApp
->GetTopWindow(), wxFrame
); 
 372             wxIcon icon 
= winTop
->GetIcon(); 
 375                 ::SendMessage(GetHwnd(), WM_SETICON
, 
 377                               (LPARAM
)GetHiconOf(icon
)); 
 383     // move the dialog to its initial position without forcing repainting 
 385     (void)MSWGetCreateWindowCoords(pos
, size
, x
, y
, w
, h
); 
 387     if ( x 
== (int)CW_USEDEFAULT 
) 
 389         // centre it on the screen - what else can we do? 
 390         wxSize sizeDpy 
= wxGetDisplaySize(); 
 392         x 
= (sizeDpy
.x 
- w
) / 2; 
 393         y 
= (sizeDpy
.y 
- h
) / 2; 
 396 #if !defined(__WXWINCE__) || defined(__WINCE_STANDARDSDK__) 
 397     if ( !::MoveWindow(GetHwnd(), x
, y
, w
, h
, FALSE
) ) 
 399         wxLogLastError(wxT("MoveWindow")); 
 403     if ( !title
.empty() ) 
 405         ::SetWindowText(GetHwnd(), title
); 
 411 #endif // __WXMICROWIN__/!__WXMICROWIN__ 
 414 bool wxTopLevelWindowMSW::CreateFrame(const wxString
& title
, 
 419     WXDWORD flags 
= MSWGetCreateWindowFlags(&exflags
); 
 421 #if !defined(__HANDHELDPC__) && ((defined(_WIN32_WCE) && _WIN32_WCE < 400) || \ 
 422     defined(__POCKETPC__) || \ 
 423     defined(__SMARTPHONE__)) 
 424     // Always expand to fit the screen in PocketPC or SmartPhone 
 425     wxSize 
sz(wxDefaultSize
); 
 427 #else // other (including normal desktop) Windows 
 431     return MSWCreate(wxCanvasClassName
, title
, pos
, sz
, flags
, exflags
); 
 434 bool wxTopLevelWindowMSW::Create(wxWindow 
*parent
, 
 436                                  const wxString
& title
, 
 440                                  const wxString
& name
) 
 442     bool ret 
wxDUMMY_INITIALIZE(false); 
 447     wxSize sizeReal 
= size
; 
 448     if ( !sizeReal
.IsFullySpecified() ) 
 450         sizeReal
.SetDefaults(GetDefaultSize()); 
 453     m_windowStyle 
= style
; 
 457     m_windowId 
= id 
== wxID_ANY 
? NewControlId() : id
; 
 459     wxTopLevelWindows
.Append(this); 
 462         parent
->AddChild(this); 
 464     if ( GetExtraStyle() & wxTOPLEVEL_EX_DIALOG 
) 
 466         // we have different dialog templates to allows creation of dialogs 
 467         // with & without captions under MSWindows, resizeable or not (but a 
 468         // resizeable dialog always has caption - otherwise it would look too 
 471         // we need 3 additional WORDs for dialog menu, class and title (as we 
 472         // don't use DS_SETFONT we don't need the fourth WORD for the font) 
 473         static const int dlgsize 
= sizeof(DLGTEMPLATE
) + (sizeof(WORD
) * 3); 
 474         DLGTEMPLATE 
*dlgTemplate 
= (DLGTEMPLATE 
*)malloc(dlgsize
); 
 475         memset(dlgTemplate
, 0, dlgsize
); 
 477         // these values are arbitrary, they won't be used normally anyhow 
 480         dlgTemplate
->cx 
= 144; 
 481         dlgTemplate
->cy 
= 75; 
 483         // reuse the code in MSWGetStyle() but correct the results slightly for 
 485         dlgTemplate
->style 
= MSWGetStyle(style
, NULL
); 
 487         // all dialogs are popups 
 488         dlgTemplate
->style 
|= WS_POPUP
; 
 490         // force 3D-look if necessary, it looks impossibly ugly otherwise 
 491         if ( style 
& (wxRESIZE_BORDER 
| wxCAPTION
) ) 
 492             dlgTemplate
->style 
|= DS_MODALFRAME
; 
 494         ret 
= CreateDialog(dlgTemplate
, title
, pos
, sizeReal
); 
 499         ret 
= CreateFrame(title
, pos
, sizeReal
); 
 502     if ( ret 
&& !(GetWindowStyleFlag() & wxCLOSE_BOX
) ) 
 504         EnableCloseButton(false); 
 507     // for some reason we need to manually send ourselves this message as 
 508     // otherwise the mnemonics are always shown -- even if they're configured 
 509     // to be hidden until "Alt" is pressed in the control panel 
 511     // this could indicate a bug somewhere else but for now this is the only 
 519             MAKEWPARAM(UIS_INITIALIZE
, UISF_HIDEFOCUS 
| UISF_HIDEACCEL
), 
 524     // Native look is full screen window on Smartphones and Standard SDK 
 525 #if defined(__WXWINCE__) 
 526     if ( style 
& wxMAXIMIZE 
) 
 532 #ifdef __SMARTPHONE__ 
 533     SetRightMenu(); // to nothing for initialization 
 539 wxTopLevelWindowMSW::~wxTopLevelWindowMSW() 
 541     // after destroying an owned window, Windows activates the next top level 
 542     // window in Z order but it may be different from our owner (to reproduce 
 543     // this simply Alt-TAB to another application and back before closing the 
 544     // owned frame) whereas we always want to yield activation to our parent 
 545     if ( HasFlag(wxFRAME_FLOAT_ON_PARENT
) ) 
 547         wxWindow 
*parent 
= GetParent(); 
 550             ::BringWindowToTop(GetHwndOf(parent
)); 
 555 // ---------------------------------------------------------------------------- 
 556 // wxTopLevelWindowMSW showing 
 557 // ---------------------------------------------------------------------------- 
 559 void wxTopLevelWindowMSW::DoShowWindow(int nShowCmd
) 
 561     ::ShowWindow(GetHwnd(), nShowCmd
); 
 563     m_iconized 
= nShowCmd 
== SW_MINIMIZE
; 
 566 bool wxTopLevelWindowMSW::Show(bool show
) 
 568     // don't use wxWindow version as we want to call DoShowWindow() ourselves 
 569     if ( !wxWindowBase::Show(show
) ) 
 575         if ( m_maximizeOnShow 
) 
 578             nShowCmd 
= SW_MAXIMIZE
; 
 580             // This is necessary, or no window appears 
 581 #ifdef __WINCE_STANDARDSDK__ 
 582             DoShowWindow(SW_SHOW
); 
 585             m_maximizeOnShow 
= false; 
 589            if ( GetWindowStyle() & wxFRAME_TOOL_WINDOW 
) 
 590                nShowCmd 
= SW_SHOWNA
; 
 600     DoShowWindow(nShowCmd
); 
 602 #if defined(__WXWINCE__) && (_WIN32_WCE >= 400 && !defined(__POCKETPC__) && !defined(__SMARTPHONE__)) 
 603     // Addornments have to be added when the frame is the correct size 
 604     wxFrame
* frame 
= wxDynamicCast(this, wxFrame
); 
 605     if (frame 
&& frame
->GetMenuBar()) 
 606         frame
->GetMenuBar()->AddAdornments(GetWindowStyleFlag()); 
 611         ::BringWindowToTop(GetHwnd()); 
 613         wxActivateEvent 
event(wxEVT_ACTIVATE
, true, m_windowId
); 
 614         event
.SetEventObject( this ); 
 615         GetEventHandler()->ProcessEvent(event
); 
 619         // Try to highlight the correct window (the parent) 
 622             HWND hWndParent 
= GetHwndOf(GetParent()); 
 624                 ::BringWindowToTop(hWndParent
); 
 631 // ---------------------------------------------------------------------------- 
 632 // wxTopLevelWindowMSW maximize/minimize 
 633 // ---------------------------------------------------------------------------- 
 635 void wxTopLevelWindowMSW::Maximize(bool maximize
) 
 639         // just maximize it directly 
 640         DoShowWindow(maximize 
? SW_MAXIMIZE 
: SW_RESTORE
); 
 644         // we can't maximize the hidden frame because it shows it as well, so 
 645         // just remember that we should do it later in this case 
 646         m_maximizeOnShow 
= maximize
; 
 650 bool wxTopLevelWindowMSW::IsMaximized() const 
 655     return ::IsZoomed(GetHwnd()) != 0; 
 659 void wxTopLevelWindowMSW::Iconize(bool iconize
) 
 661     DoShowWindow(iconize 
? SW_MINIMIZE 
: SW_RESTORE
); 
 664 bool wxTopLevelWindowMSW::IsIconized() const 
 669     // also update the current state 
 670     ((wxTopLevelWindowMSW 
*)this)->m_iconized 
= ::IsIconic(GetHwnd()) != 0; 
 676 void wxTopLevelWindowMSW::Restore() 
 678     DoShowWindow(SW_RESTORE
); 
 681 // ---------------------------------------------------------------------------- 
 682 // wxTopLevelWindowMSW fullscreen 
 683 // ---------------------------------------------------------------------------- 
 685 bool wxTopLevelWindowMSW::ShowFullScreen(bool show
, long style
) 
 687     if ( show 
== IsFullScreen() ) 
 693     m_fsIsShowing 
= show
; 
 699         // zap the frame borders 
 701         // save the 'normal' window style 
 702         m_fsOldWindowStyle 
= GetWindowLong(GetHwnd(), GWL_STYLE
); 
 704         // save the old position, width & height, maximize state 
 705         m_fsOldSize 
= GetRect(); 
 706         m_fsIsMaximized 
= IsMaximized(); 
 708         // decide which window style flags to turn off 
 709         LONG newStyle 
= m_fsOldWindowStyle
; 
 712         if (style 
& wxFULLSCREEN_NOBORDER
) 
 714             offFlags 
|= WS_BORDER
; 
 716             offFlags 
|= WS_THICKFRAME
; 
 719         if (style 
& wxFULLSCREEN_NOCAPTION
) 
 720             offFlags 
|= WS_CAPTION 
| WS_SYSMENU
; 
 722         newStyle 
&= ~offFlags
; 
 724         // change our window style to be compatible with full-screen mode 
 725         ::SetWindowLong(GetHwnd(), GWL_STYLE
, newStyle
); 
 729         // resize to the size of the display containing us 
 730         int dpy 
= wxDisplay::GetFromWindow(this); 
 731         if ( dpy 
!= wxNOT_FOUND 
) 
 733             rect 
= wxDisplay(dpy
).GetGeometry(); 
 735         else // fall back to the main desktop 
 736 #else // wxUSE_DISPLAY 
 738             // resize to the size of the desktop 
 739             wxCopyRECTToRect(wxGetWindowRect(::GetDesktopWindow()), rect
); 
 741             // FIXME: size of the bottom menu (toolbar) 
 742             // should be taken in account 
 743             rect
.height 
+= rect
.y
; 
 747 #endif // wxUSE_DISPLAY 
 751         // now flush the window style cache and actually go full-screen 
 752         long flags 
= SWP_FRAMECHANGED
; 
 754         // showing the frame full screen should also show it if it's still 
 758             // don't call wxWindow version to avoid flicker from calling 
 759             // ::ShowWindow() -- we're going to show the window at the correct 
 760             // location directly below -- but do call the wxWindowBase version 
 761             // to sync the internal m_isShown flag 
 762             wxWindowBase::Show(); 
 764             flags 
|= SWP_SHOWWINDOW
; 
 767         SetWindowPos(GetHwnd(), HWND_TOP
, 
 768                      rect
.x
, rect
.y
, rect
.width
, rect
.height
, 
 771 #if !defined(__HANDHELDPC__) && (defined(__WXWINCE__) && (_WIN32_WCE < 400)) 
 772         ::SHFullScreen(GetHwnd(), SHFS_HIDETASKBAR 
| SHFS_HIDESIPBUTTON
); 
 775         // finally send an event allowing the window to relayout itself &c 
 776         wxSizeEvent 
event(rect
.GetSize(), GetId()); 
 777         GetEventHandler()->ProcessEvent(event
); 
 779     else // stop showing full screen 
 781 #if !defined(__HANDHELDPC__) && (defined(__WXWINCE__) && (_WIN32_WCE < 400)) 
 782         ::SHFullScreen(GetHwnd(), SHFS_SHOWTASKBAR 
| SHFS_SHOWSIPBUTTON
); 
 784         Maximize(m_fsIsMaximized
); 
 785         SetWindowLong(GetHwnd(),GWL_STYLE
, m_fsOldWindowStyle
); 
 786         SetWindowPos(GetHwnd(),HWND_TOP
,m_fsOldSize
.x
, m_fsOldSize
.y
, 
 787             m_fsOldSize
.width
, m_fsOldSize
.height
, SWP_FRAMECHANGED
); 
 793 // ---------------------------------------------------------------------------- 
 794 // wxTopLevelWindowMSW misc 
 795 // ---------------------------------------------------------------------------- 
 797 void wxTopLevelWindowMSW::SetIcon(const wxIcon
& icon
) 
 799     SetIcons( wxIconBundle( icon 
) ); 
 802 void wxTopLevelWindowMSW::SetIcons(const wxIconBundle
& icons
) 
 804     wxTopLevelWindowBase::SetIcons(icons
); 
 806 #if defined(__WIN95__) && !defined(__WXMICROWIN__) 
 807     const wxIcon
& sml 
= icons
.GetIcon( wxSize( 16, 16 ) ); 
 808     if( sml
.Ok() && sml
.GetWidth() == 16 && sml
.GetHeight() == 16 ) 
 810         ::SendMessage( GetHwndOf( this ), WM_SETICON
, ICON_SMALL
, 
 811                        (LPARAM
)GetHiconOf(sml
) ); 
 814     const wxIcon
& big 
= icons
.GetIcon( wxSize( 32, 32 ) ); 
 815     if( big
.Ok() && big
.GetWidth() == 32 && big
.GetHeight() == 32 ) 
 817         ::SendMessage( GetHwndOf( this ), WM_SETICON
, ICON_BIG
, 
 818                        (LPARAM
)GetHiconOf(big
) ); 
 823 bool wxTopLevelWindowMSW::EnableCloseButton(bool enable
) 
 825 #if !defined(__WXMICROWIN__) 
 826     // get system (a.k.a. window) menu 
 827     HMENU hmenu 
= GetSystemMenu(GetHwnd(), FALSE 
/* get it */); 
 830         // no system menu at all -- ok if we want to remove the close button 
 831         // anyhow, but bad if we want to show it 
 835     // enabling/disabling the close item from it also automatically 
 836     // disables/enables the close title bar button 
 837     if ( ::EnableMenuItem(hmenu
, SC_CLOSE
, 
 839                           (enable 
? MF_ENABLED 
: MF_GRAYED
)) == -1 ) 
 841         wxLogLastError(_T("EnableMenuItem(SC_CLOSE)")); 
 846     // update appearance immediately 
 847     if ( !::DrawMenuBar(GetHwnd()) ) 
 849         wxLogLastError(_T("DrawMenuBar")); 
 852 #endif // !__WXMICROWIN__ 
 859 bool wxTopLevelWindowMSW::SetShape(const wxRegion
& region
) 
 861     wxCHECK_MSG( HasFlag(wxFRAME_SHAPED
), false, 
 862                  _T("Shaped windows must be created with the wxFRAME_SHAPED style.")); 
 864     // The empty region signifies that the shape should be removed from the 
 866     if ( region
.IsEmpty() ) 
 868         if (::SetWindowRgn(GetHwnd(), NULL
, TRUE
) == 0) 
 870             wxLogLastError(_T("SetWindowRgn")); 
 876     // Windows takes ownership of the region, so 
 877     // we'll have to make a copy of the region to give to it. 
 878     DWORD noBytes 
= ::GetRegionData(GetHrgnOf(region
), 0, NULL
); 
 879     RGNDATA 
*rgnData 
= (RGNDATA
*) new char[noBytes
]; 
 880     ::GetRegionData(GetHrgnOf(region
), noBytes
, rgnData
); 
 881     HRGN hrgn 
= ::ExtCreateRegion(NULL
, noBytes
, rgnData
); 
 882     delete[] (char*) rgnData
; 
 884     // SetWindowRgn expects the region to be in coordinants 
 885     // relative to the window, not the client area.  Figure 
 886     // out the offset, if any. 
 888     DWORD dwStyle 
=   ::GetWindowLong(GetHwnd(), GWL_STYLE
); 
 889     DWORD dwExStyle 
= ::GetWindowLong(GetHwnd(), GWL_EXSTYLE
); 
 890     ::GetClientRect(GetHwnd(), &rect
); 
 891     ::AdjustWindowRectEx(&rect
, dwStyle
, FALSE
, dwExStyle
); 
 892     ::OffsetRgn(hrgn
, -rect
.left
, -rect
.top
); 
 894     // Now call the shape API with the new region. 
 895     if (::SetWindowRgn(GetHwnd(), hrgn
, TRUE
) == 0) 
 897         wxLogLastError(_T("SetWindowRgn")); 
 903 #endif // !__WXWINCE__ 
 905 void wxTopLevelWindowMSW::RequestUserAttention(int flags
) 
 907     // check if we can use FlashWindowEx(): unfortunately an explicit test for 
 908     // FLASHW_STOP, for example, doesn't work because MSVC6 headers do #define 
 909     // it but don't provide FlashWindowEx() declaration 
 911     // available in the headers, check if it is supported by the system 
 912     typedef BOOL (WINAPI 
*FlashWindowEx_t
)(FLASHWINFO 
*pfwi
); 
 913     FlashWindowEx_t s_pfnFlashWindowEx 
= NULL
; 
 914     if ( !s_pfnFlashWindowEx 
) 
 916         wxDynamicLibrary 
dllUser32(_T("user32.dll")); 
 917         s_pfnFlashWindowEx 
= (FlashWindowEx_t
) 
 918                                 dllUser32
.GetSymbol(_T("FlashWindowEx")); 
 920         // we can safely unload user32.dll here, it's goign to remain loaded as 
 921         // long as the program is running anyhow 
 924     if ( s_pfnFlashWindowEx 
) 
 926         WinStruct
<FLASHWINFO
> fwi
; 
 927         fwi
.hwnd 
= GetHwnd(); 
 928         fwi
.dwFlags 
= FLASHW_ALL
; 
 929         if ( flags 
& wxUSER_ATTENTION_INFO 
) 
 931             // just flash a few times 
 934         else // wxUSER_ATTENTION_ERROR 
 936             // flash until the user notices it 
 937             fwi
.dwFlags 
|= FLASHW_TIMERNOFG
; 
 940         s_pfnFlashWindowEx(&fwi
); 
 942     else // FlashWindowEx() not available 
 943 #endif // FlashWindowEx() defined 
 947         ::FlashWindow(GetHwnd(), TRUE
); 
 948 #endif // __WXWINCE__ 
 952 // ---------------------------------------------------------------------------- 
 953 // wxTopLevelWindow event handling 
 954 // ---------------------------------------------------------------------------- 
 956 // Default activation behaviour - set the focus for the first child 
 958 void wxTopLevelWindowMSW::OnActivate(wxActivateEvent
& event
) 
 960     if ( event
.GetActive() ) 
 962         // restore focus to the child which was last focused unless we already 
 964         wxLogTrace(_T("focus"), _T("wxTLW %08x activated."), (int) m_hWnd
); 
 966         wxWindow 
*winFocus 
= FindFocus(); 
 967         if ( !winFocus 
|| wxGetTopLevelParent(winFocus
) != this ) 
 969             wxWindow 
*parent 
= m_winLastFocused 
? m_winLastFocused
->GetParent() 
 976             wxSetFocusToChild(parent
, &m_winLastFocused
); 
 981         // remember the last focused child if it is our child 
 982         m_winLastFocused 
= FindFocus(); 
 984         if ( m_winLastFocused 
) 
 986             // let it know that it doesn't have focus any more 
 987             m_winLastFocused
->HandleKillFocus((WXHWND
)NULL
); 
 989             // and don't remember it if it's a child from some other frame 
 990             if ( wxGetTopLevelParent(m_winLastFocused
) != this ) 
 992                 m_winLastFocused 
= NULL
; 
 996         wxLogTrace(_T("focus"), 
 997                    _T("wxTLW %08x deactivated, last focused: %08x."), 
 999                    (int) (m_winLastFocused 
? GetHwndOf(m_winLastFocused
) 
1006 // the DialogProc for all wxWidgets dialogs 
1007 LONG APIENTRY _EXPORT
 
1008 wxDlgProc(HWND hDlg
, 
1010           WPARAM 
WXUNUSED(wParam
), 
1011           LPARAM 
WXUNUSED(lParam
)) 
1013     if ( message 
== WM_INITDIALOG 
) 
1015         // under CE, add a "Ok" button in the dialog title bar and make it full 
1018         // VZ: we should probably allow for overriding this, e.g. by including 
1019         //     MAXIMIZED flag in the dialog style by default and doing this 
1020         //     only if it is present... 
1022         // Standard SDK doesn't have aygshell.dll: see 
1023         // include/wx/msw/wince/libraries.h 
1024 #if defined(__WXWINCE__) && !defined(__WINCE_STANDARDSDK__) && !defined(__HANDHELDPC__) 
1025         SHINITDLGINFO shidi
; 
1026         shidi
.dwMask 
= SHIDIM_FLAGS
; 
1027         shidi
.dwFlags 
= SHIDIF_SIZEDLGFULLSCREEN
 
1028 #ifndef __SMARTPHONE__ 
1033         SHInitDialog( &shidi 
); 
1034 #else // no SHInitDialog() 
1039     // for almost all messages, returning FALSE means that we didn't process 
1042     // for WM_INITDIALOG, returning TRUE tells system to set focus to 
1043     // the first control in the dialog box, but as we set the focus 
1044     // ourselves, we return FALSE for it as well 
1048 // ============================================================================ 
1049 // wxTLWHiddenParentModule implementation 
1050 // ============================================================================ 
1052 HWND 
wxTLWHiddenParentModule::ms_hwnd 
= NULL
; 
1054 const wxChar 
*wxTLWHiddenParentModule::ms_className 
= NULL
; 
1056 bool wxTLWHiddenParentModule::OnInit() 
1059     ms_className 
= NULL
; 
1064 void wxTLWHiddenParentModule::OnExit() 
1068         if ( !::DestroyWindow(ms_hwnd
) ) 
1070             wxLogLastError(_T("DestroyWindow(hidden TLW parent)")); 
1078         if ( !::UnregisterClass(ms_className
, wxGetInstance()) ) 
1080             wxLogLastError(_T("UnregisterClass(\"wxTLWHiddenParent\")")); 
1083         ms_className 
= NULL
; 
1088 HWND 
wxTLWHiddenParentModule::GetHWND() 
1092         if ( !ms_className 
) 
1094             static const wxChar 
*HIDDEN_PARENT_CLASS 
= _T("wxTLWHiddenParent"); 
1097             wxZeroMemory(wndclass
); 
1099             wndclass
.lpfnWndProc   
= DefWindowProc
; 
1100             wndclass
.hInstance     
= wxGetInstance(); 
1101             wndclass
.lpszClassName 
= HIDDEN_PARENT_CLASS
; 
1103             if ( !::RegisterClass(&wndclass
) ) 
1105                 wxLogLastError(_T("RegisterClass(\"wxTLWHiddenParent\")")); 
1109                 ms_className 
= HIDDEN_PARENT_CLASS
; 
1113         ms_hwnd 
= ::CreateWindow(ms_className
, wxEmptyString
, 0, 0, 0, 0, 0, NULL
, 
1114                                  (HMENU
)NULL
, wxGetInstance(), NULL
); 
1117             wxLogLastError(_T("CreateWindow(hidden TLW parent)"));