1 ///////////////////////////////////////////////////////////////////////////// 
   2 // Name:        src/msw/mdi.cpp 
   3 // Purpose:     MDI classes for wxMSW 
   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" 
  27 #if wxUSE_MDI && !defined(__WXUNIVERSAL__) 
  35     #include "wx/dialog.h" 
  37         #include "wx/statusbr.h" 
  39     #include "wx/settings.h" 
  44 #include "wx/stockitem.h" 
  46 #include "wx/msw/private.h" 
  48 #if wxUSE_STATUSBAR && wxUSE_NATIVE_STATUSBAR 
  49     #include "wx/msw/statbr95.h" 
  53     #include "wx/toolbar.h" 
  54 #endif // wxUSE_TOOLBAR 
  58 // --------------------------------------------------------------------------- 
  60 // --------------------------------------------------------------------------- 
  62 extern wxMenu 
*wxCurrentPopupMenu
; 
  64 extern const wxChar 
*wxMDIFrameClassName
;   // from app.cpp 
  65 extern const wxChar 
*wxMDIChildFrameClassName
; 
  66 extern const wxChar 
*wxMDIChildFrameClassNameNoRedraw
; 
  67 extern void wxAssociateWinWithHandle(HWND hWnd
, wxWindow 
*win
); 
  68 extern void wxRemoveHandleAssociation(wxWindow 
*win
); 
  70 static HWND invalidHandle 
= 0; 
  72 // --------------------------------------------------------------------------- 
  74 // --------------------------------------------------------------------------- 
  76 static const int IDM_WINDOWTILEHOR  
= 4001; 
  77 static const int IDM_WINDOWCASCADE 
= 4002; 
  78 static const int IDM_WINDOWICONS 
= 4003; 
  79 static const int IDM_WINDOWNEXT 
= 4004; 
  80 static const int IDM_WINDOWTILEVERT 
= 4005; 
  81 static const int IDM_WINDOWPREV 
= 4006; 
  83 // This range gives a maximum of 500 MDI children. Should be enough :-) 
  84 static const int wxFIRST_MDI_CHILD 
= 4100; 
  85 static const int wxLAST_MDI_CHILD 
= 4600; 
  87 // --------------------------------------------------------------------------- 
  89 // --------------------------------------------------------------------------- 
  91 // set the MDI menus (by sending the WM_MDISETMENU message) and update the menu 
  92 // of the parent of win (which is supposed to be the MDI client window) 
  93 static void MDISetMenu(wxWindow 
*win
, HMENU hmenuFrame
, HMENU hmenuWindow
); 
  95 // insert the window menu (subMenu) into menu just before "Help" submenu or at 
  96 // the very end if not found 
  97 static void InsertWindowMenu(wxWindow 
*win
, WXHMENU menu
, HMENU subMenu
); 
  99 // Remove the window menu 
 100 static void RemoveWindowMenu(wxWindow 
*win
, WXHMENU menu
); 
 102 // is this an id of an MDI child? 
 103 inline bool IsMdiCommandId(int id
) 
 105     return (id 
>= wxFIRST_MDI_CHILD
) && (id 
<= wxLAST_MDI_CHILD
); 
 108 // unpack the parameters of WM_MDIACTIVATE message 
 109 static void UnpackMDIActivate(WXWPARAM wParam
, WXLPARAM lParam
, 
 110                               WXWORD 
*activate
, WXHWND 
*hwndAct
, WXHWND 
*hwndDeact
); 
 112 // return the HMENU of the MDI menu 
 113 static inline HMENU 
GetMDIWindowMenu(wxMDIParentFrame 
*frame
) 
 115     wxMenu 
*menu 
= frame
->GetWindowMenu(); 
 116     return menu 
? GetHmenuOf(menu
) : 0; 
 119 // =========================================================================== 
 121 // =========================================================================== 
 123 // --------------------------------------------------------------------------- 
 125 // --------------------------------------------------------------------------- 
 127 IMPLEMENT_DYNAMIC_CLASS(wxMDIParentFrame
, wxFrame
) 
 128 IMPLEMENT_DYNAMIC_CLASS(wxMDIChildFrame
, wxFrame
) 
 129 IMPLEMENT_DYNAMIC_CLASS(wxMDIClientWindow
, wxWindow
) 
 131 BEGIN_EVENT_TABLE(wxMDIParentFrame
, wxFrame
) 
 132     EVT_SIZE(wxMDIParentFrame::OnSize
) 
 133     EVT_ICONIZE(wxMDIParentFrame::OnIconized
) 
 134     EVT_SYS_COLOUR_CHANGED(wxMDIParentFrame::OnSysColourChanged
) 
 137 BEGIN_EVENT_TABLE(wxMDIChildFrame
, wxFrame
) 
 138     EVT_IDLE(wxMDIChildFrame::OnIdle
) 
 141 BEGIN_EVENT_TABLE(wxMDIClientWindow
, wxWindow
) 
 142     EVT_SCROLL(wxMDIClientWindow::OnScroll
) 
 145 // =========================================================================== 
 146 // wxMDIParentFrame: the frame which contains the client window which manages 
 148 // =========================================================================== 
 150 wxMDIParentFrame::wxMDIParentFrame() 
 152     m_clientWindow 
= NULL
; 
 153     m_currentChild 
= NULL
; 
 154     m_windowMenu 
= (wxMenu
*) NULL
; 
 155     m_parentFrameActive 
= true; 
 158 bool wxMDIParentFrame::Create(wxWindow 
*parent
, 
 160                               const wxString
& title
, 
 164                               const wxString
& name
) 
 166   m_clientWindow 
= NULL
; 
 167   m_currentChild 
= NULL
; 
 169   // this style can be used to prevent a window from having the standard MDI 
 171   if ( style 
& wxFRAME_NO_WINDOW_MENU 
) 
 173       m_windowMenu 
= (wxMenu 
*)NULL
; 
 175   else // normal case: we have the window menu, so construct it 
 177       m_windowMenu 
= new wxMenu
; 
 179       m_windowMenu
->Append(IDM_WINDOWCASCADE
, _("&Cascade")); 
 180       m_windowMenu
->Append(IDM_WINDOWTILEHOR
, _("Tile &Horizontally")); 
 181       m_windowMenu
->Append(IDM_WINDOWTILEVERT
, _("Tile &Vertically")); 
 182       m_windowMenu
->AppendSeparator(); 
 183       m_windowMenu
->Append(IDM_WINDOWICONS
, _("&Arrange Icons")); 
 184       m_windowMenu
->Append(IDM_WINDOWNEXT
, _("&Next")); 
 185       m_windowMenu
->Append(IDM_WINDOWPREV
, _("&Previous")); 
 188   m_parentFrameActive 
= true; 
 191     wxTopLevelWindows
.Append(this); 
 194   m_windowStyle 
= style
; 
 197       parent
->AddChild(this); 
 199   if ( id 
!= wxID_ANY 
) 
 202     m_windowId 
= NewControlId(); 
 205   WXDWORD msflags 
= MSWGetCreateWindowFlags(&exflags
); 
 206   msflags 
&= ~WS_VSCROLL
; 
 207   msflags 
&= ~WS_HSCROLL
; 
 209   if ( !wxWindow::MSWCreate(wxMDIFrameClassName
, 
 218   // unlike (almost?) all other windows, frames are created hidden 
 224 wxMDIParentFrame::~wxMDIParentFrame() 
 226     // see comment in ~wxMDIChildFrame 
 228     m_frameToolBar 
= NULL
; 
 231     m_frameStatusBar 
= NULL
; 
 232 #endif // wxUSE_STATUSBAR 
 239         m_windowMenu 
= (wxMenu
*) NULL
; 
 242     // the MDI frame menubar is not automatically deleted by Windows unlike for 
 246         ::DestroyMenu((HMENU
)m_hMenu
); 
 247         m_hMenu 
= (WXHMENU
)NULL
; 
 250     if ( m_clientWindow 
) 
 252         if ( m_clientWindow
->MSWGetOldWndProc() ) 
 253             m_clientWindow
->UnsubclassWin(); 
 255         m_clientWindow
->SetHWND(0); 
 256         delete m_clientWindow
; 
 260 #if wxUSE_MENUS_NATIVE 
 262 void wxMDIParentFrame::InternalSetMenuBar() 
 264     m_parentFrameActive 
= true; 
 266     InsertWindowMenu(GetClientWindow(), m_hMenu
, GetMDIWindowMenu(this)); 
 269 #endif // wxUSE_MENUS_NATIVE 
 271 void wxMDIParentFrame::SetWindowMenu(wxMenu
* menu
) 
 277             // Remove old window menu 
 278             RemoveWindowMenu(GetClientWindow(), m_hMenu
); 
 282         m_windowMenu 
= (wxMenu
*) NULL
; 
 290             InsertWindowMenu(GetClientWindow(), m_hMenu
, 
 291                              GetHmenuOf(m_windowMenu
)); 
 296 void wxMDIParentFrame::DoMenuUpdates(wxMenu
* menu
) 
 298     wxMDIChildFrame 
*child 
= GetActiveChild(); 
 301         wxEvtHandler
* source 
= child
->GetEventHandler(); 
 302         wxMenuBar
* bar 
= child
->GetMenuBar(); 
 306             menu
->UpdateUI(source
); 
 312                 int nCount 
= bar
->GetMenuCount(); 
 313                 for (int n 
= 0; n 
< nCount
; n
++) 
 314                     bar
->GetMenu(n
)->UpdateUI(source
); 
 320         wxFrameBase::DoMenuUpdates(menu
); 
 324 void wxMDIParentFrame::UpdateClientSize() 
 326     if ( GetClientWindow() ) 
 329         GetClientSize(&width
, &height
); 
 331         GetClientWindow()->SetSize(0, 0, width
, height
); 
 335 void wxMDIParentFrame::OnSize(wxSizeEvent
& WXUNUSED(event
)) 
 339     // do not call event.Skip() here, it somehow messes up MDI client window 
 342 void wxMDIParentFrame::OnIconized(wxIconizeEvent
& event
) 
 346     if ( !event
.Iconized() ) 
 352 // Returns the active MDI child window 
 353 wxMDIChildFrame 
*wxMDIParentFrame::GetActiveChild() const 
 355     HWND hWnd 
= (HWND
)::SendMessage(GetWinHwnd(GetClientWindow()), 
 356                                     WM_MDIGETACTIVE
, 0, 0L); 
 360         return (wxMDIChildFrame 
*)wxFindWinFromHandle((WXHWND
) hWnd
); 
 363 // Create the client window class (don't Create the window, just return a new 
 365 wxMDIClientWindow 
*wxMDIParentFrame::OnCreateClient() 
 367     return new wxMDIClientWindow
; 
 370 // Responds to colour changes, and passes event on to children. 
 371 void wxMDIParentFrame::OnSysColourChanged(wxSysColourChangedEvent
& event
) 
 373     if ( m_clientWindow 
) 
 375         m_clientWindow
->SetBackgroundColour(wxSystemSettings::GetColour(wxSYS_COLOUR_APPWORKSPACE
)); 
 376         m_clientWindow
->Refresh(); 
 382 WXHICON 
wxMDIParentFrame::GetDefaultIcon() const 
 384     // we don't have any standard icons (any more) 
 388 // --------------------------------------------------------------------------- 
 390 // --------------------------------------------------------------------------- 
 392 void wxMDIParentFrame::Cascade() 
 394     ::SendMessage(GetWinHwnd(GetClientWindow()), WM_MDICASCADE
, 0, 0); 
 397 void wxMDIParentFrame::Tile(wxOrientation orient
) 
 399     wxASSERT_MSG( orient 
== wxHORIZONTAL 
|| orient 
== wxVERTICAL
, 
 400                   _T("invalid orientation value") ); 
 402     ::SendMessage(GetWinHwnd(GetClientWindow()), WM_MDITILE
, 
 403                   orient 
== wxHORIZONTAL 
? MDITILE_HORIZONTAL
 
 404                                          : MDITILE_VERTICAL
, 0); 
 407 void wxMDIParentFrame::ArrangeIcons() 
 409     ::SendMessage(GetWinHwnd(GetClientWindow()), WM_MDIICONARRANGE
, 0, 0); 
 412 void wxMDIParentFrame::ActivateNext() 
 414     ::SendMessage(GetWinHwnd(GetClientWindow()), WM_MDINEXT
, 0, 0); 
 417 void wxMDIParentFrame::ActivatePrevious() 
 419     ::SendMessage(GetWinHwnd(GetClientWindow()), WM_MDINEXT
, 0, 1); 
 422 // --------------------------------------------------------------------------- 
 423 // the MDI parent frame window proc 
 424 // --------------------------------------------------------------------------- 
 426 WXLRESULT 
wxMDIParentFrame::MSWWindowProc(WXUINT message
, 
 431     bool processed 
= false; 
 437                 WXWORD state
, minimized
; 
 439                 UnpackActivate(wParam
, lParam
, &state
, &minimized
, &hwnd
); 
 441                 processed 
= HandleActivate(state
, minimized 
!= 0, hwnd
); 
 449                 UnpackCommand(wParam
, lParam
, &id
, &hwnd
, &cmd
); 
 451                 (void)HandleCommand(id
, cmd
, hwnd
); 
 453                 // even if the frame didn't process it, there is no need to try it 
 454                 // once again (i.e. call wxFrame::HandleCommand()) - we just did it, 
 455                 // so pretend we processed the message anyhow 
 459             // always pass this message DefFrameProc(), otherwise MDI menu 
 460             // commands (and sys commands - more surprisingly!) won't work 
 461             MSWDefWindowProc(message
, wParam
, lParam
); 
 465             m_clientWindow 
= OnCreateClient(); 
 466             // Uses own style for client style 
 467             if ( !m_clientWindow
->CreateClient(this, GetWindowStyleFlag()) ) 
 469                 wxLogMessage(_("Failed to create MDI parent frame.")); 
 480             // we erase background ourselves 
 488                 UnpackMenuSelect(wParam
, lParam
, &item
, &flags
, &hmenu
); 
 490                 if ( m_parentFrameActive 
) 
 492                     processed 
= HandleMenuSelect(item
, flags
, hmenu
); 
 494                 else if (m_currentChild
) 
 496                     processed 
= m_currentChild
-> 
 497                         HandleMenuSelect(item
, flags
, hmenu
); 
 503             // though we don't (usually) resize the MDI client to exactly fit the 
 504             // client area we need to pass this one to DefFrameProc to allow the children to show 
 509         rc 
= wxFrame::MSWWindowProc(message
, wParam
, lParam
); 
 514 bool wxMDIParentFrame::HandleActivate(int state
, bool minimized
, WXHWND activate
) 
 516     bool processed 
= false; 
 518     if ( wxWindow::HandleActivate(state
, minimized
, activate
) ) 
 524     // If this window is an MDI parent, we must also send an OnActivate message 
 525     // to the current child. 
 526     if ( (m_currentChild 
!= NULL
) && 
 527          ((state 
== WA_ACTIVE
) || (state 
== WA_CLICKACTIVE
)) ) 
 529         wxActivateEvent 
event(wxEVT_ACTIVATE
, true, m_currentChild
->GetId()); 
 530         event
.SetEventObject( m_currentChild 
); 
 531         if ( m_currentChild
->GetEventHandler()->ProcessEvent(event
) ) 
 538 bool wxMDIParentFrame::HandleCommand(WXWORD id
, WXWORD cmd
, WXHWND hwnd
) 
 540     // In case it's e.g. a toolbar. 
 543         wxWindow 
*win 
= wxFindWinFromHandle(hwnd
); 
 545             return win
->MSWCommand(cmd
, id
); 
 548     if (wxCurrentPopupMenu
) 
 550         wxMenu 
*popupMenu 
= wxCurrentPopupMenu
; 
 551         wxCurrentPopupMenu 
= NULL
; 
 552         if (popupMenu
->MSWCommand(cmd
, id
)) 
 556     // is it one of standard MDI commands? 
 562         case IDM_WINDOWCASCADE
: 
 564             wParam 
= MDITILE_SKIPDISABLED
; 
 567         case IDM_WINDOWTILEHOR
: 
 568             wParam 
|= MDITILE_HORIZONTAL
; 
 571         case IDM_WINDOWTILEVERT
: 
 573                 wParam 
= MDITILE_VERTICAL
; 
 575             wParam 
|= MDITILE_SKIPDISABLED
; 
 578         case IDM_WINDOWICONS
: 
 579             msg 
= WM_MDIICONARRANGE
; 
 584             lParam 
= 0;         // next child 
 589             lParam 
= 1;         // previous child 
 598         ::SendMessage(GetWinHwnd(GetClientWindow()), msg
, wParam
, lParam
); 
 603     // FIXME VZ: what does this test do?? 
 606         return false; // Get WndProc to call default proc 
 609     if ( IsMdiCommandId(id
) ) 
 611         wxWindowList::compatibility_iterator node 
= GetChildren().GetFirst(); 
 614             wxWindow 
*child 
= node
->GetData(); 
 615             if ( child
->GetHWND() ) 
 617                 long childId 
= wxGetWindowId(child
->GetHWND()); 
 618                 if (childId 
== (long)id
) 
 620                     ::SendMessage( GetWinHwnd(GetClientWindow()), 
 622                                    (WPARAM
)child
->GetHWND(), 0); 
 626             node 
= node
->GetNext(); 
 629     else if ( m_parentFrameActive 
) 
 631         return ProcessCommand(id
); 
 633     else if ( m_currentChild 
) 
 635         return m_currentChild
->HandleCommand(id
, cmd
, hwnd
); 
 639         // this shouldn't happen because it means that our messages are being 
 640         // lost (they're not sent to the parent frame nor to the children) 
 641         wxFAIL_MSG(wxT("MDI parent frame is not active, yet there is no active MDI child?")); 
 647 WXLRESULT 
wxMDIParentFrame::MSWDefWindowProc(WXUINT message
, 
 652     if ( GetClientWindow() ) 
 653         clientWnd 
= GetClientWindow()->GetHWND(); 
 657     return DefFrameProc(GetHwnd(), (HWND
)clientWnd
, message
, wParam
, lParam
); 
 660 bool wxMDIParentFrame::MSWTranslateMessage(WXMSG
* msg
) 
 662     MSG 
*pMsg 
= (MSG 
*)msg
; 
 664     // first let the current child get it 
 665     if ( m_currentChild 
&& m_currentChild
->GetHWND() && 
 666          m_currentChild
->MSWTranslateMessage(msg
) ) 
 671     // then try out accel table (will also check the menu accels) 
 672     if ( wxFrame::MSWTranslateMessage(msg
) ) 
 677     // finally, check for MDI specific built in accel keys 
 678     if ( pMsg
->message 
== WM_KEYDOWN 
|| pMsg
->message 
== WM_SYSKEYDOWN 
) 
 680         if ( ::TranslateMDISysAccel(GetWinHwnd(GetClientWindow()), pMsg
)) 
 687 // =========================================================================== 
 689 // =========================================================================== 
 691 void wxMDIChildFrame::Init() 
 693     m_needsResize 
= true; 
 694     m_needsInitialShow 
= true; 
 697 bool wxMDIChildFrame::Create(wxMDIParentFrame 
*parent
, 
 699                              const wxString
& title
, 
 703                              const wxString
& name
) 
 707   if ( id 
!= wxID_ANY 
) 
 710     m_windowId 
= (int)NewControlId(); 
 714       parent
->AddChild(this); 
 724   mcs
.szClass 
= style 
& wxFULL_REPAINT_ON_RESIZE
 
 725                     ? wxMDIChildFrameClassName
 
 726                     : wxMDIChildFrameClassNameNoRedraw
; 
 728   mcs
.hOwner 
= wxGetInstance(); 
 729   if (x 
!= wxDefaultCoord
) 
 732       mcs
.x 
= CW_USEDEFAULT
; 
 734   if (y 
!= wxDefaultCoord
) 
 737       mcs
.y 
= CW_USEDEFAULT
; 
 739   if (width 
!= wxDefaultCoord
) 
 742       mcs
.cx 
= CW_USEDEFAULT
; 
 744   if (height 
!= wxDefaultCoord
) 
 747       mcs
.cy 
= CW_USEDEFAULT
; 
 749   DWORD msflags 
= WS_OVERLAPPED 
| WS_CLIPCHILDREN
; 
 750   if (style 
& wxMINIMIZE_BOX
) 
 751     msflags 
|= WS_MINIMIZEBOX
; 
 752   if (style 
& wxMAXIMIZE_BOX
) 
 753     msflags 
|= WS_MAXIMIZEBOX
; 
 754   if (style 
& wxTHICK_FRAME
) 
 755     msflags 
|= WS_THICKFRAME
; 
 756   if (style 
& wxSYSTEM_MENU
) 
 757     msflags 
|= WS_SYSMENU
; 
 758   if ((style 
& wxMINIMIZE
) || (style 
& wxICONIZE
)) 
 759     msflags 
|= WS_MINIMIZE
; 
 760   if (style 
& wxMAXIMIZE
) 
 761     msflags 
|= WS_MAXIMIZE
; 
 762   if (style 
& wxCAPTION
) 
 763     msflags 
|= WS_CAPTION
; 
 769   wxWindowCreationHook 
hook(this); 
 771   m_hWnd 
= (WXHWND
)::SendMessage(GetWinHwnd(parent
->GetClientWindow()), 
 772                                  WM_MDICREATE
, 0, (LONG
)(LPSTR
)&mcs
); 
 774   wxAssociateWinWithHandle((HWND
) GetHWND(), this); 
 779 wxMDIChildFrame::~wxMDIChildFrame() 
 781     // will be destroyed by DestroyChildren() but reset them before calling it 
 782     // to avoid using dangling pointers if a callback comes in the meanwhile 
 784     m_frameToolBar 
= NULL
; 
 787     m_frameStatusBar 
= NULL
; 
 788 #endif // wxUSE_STATUSBAR 
 792     RemoveWindowMenu(NULL
, m_hMenu
); 
 797 bool wxMDIChildFrame::Show(bool show
) 
 799     m_needsInitialShow 
= false; 
 801     if (!wxFrame::Show(show
)) 
 804     // KH: Without this call, new MDI children do not become active. 
 805     // This was added here after the same BringWindowToTop call was 
 806     // removed from wxTopLevelWindow::Show (November 2005) 
 808         ::BringWindowToTop(GetHwnd()); 
 810     // we need to refresh the MDI frame window menu to include (or exclude if 
 811     // we've been hidden) this frame 
 812     wxMDIParentFrame 
*parent 
= (wxMDIParentFrame 
*)GetParent(); 
 813     MDISetMenu(parent
->GetClientWindow(), NULL
, NULL
); 
 818 // Set the client size (i.e. leave the calculation of borders etc. 
 820 void wxMDIChildFrame::DoSetClientSize(int width
, int height
) 
 822   HWND hWnd 
= GetHwnd(); 
 825   ::GetClientRect(hWnd
, &rect
); 
 828   GetWindowRect(hWnd
, &rect2
); 
 830   // Find the difference between the entire window (title bar and all) 
 831   // and the client area; add this to the new client size to move the 
 833   int actual_width 
= rect2
.right 
- rect2
.left 
- rect
.right 
+ width
; 
 834   int actual_height 
= rect2
.bottom 
- rect2
.top 
- rect
.bottom 
+ height
; 
 837   if (GetStatusBar() && GetStatusBar()->IsShown()) 
 840     GetStatusBar()->GetSize(&sx
, &sy
); 
 843 #endif // wxUSE_STATUSBAR 
 846   point
.x 
= rect2
.left
; 
 849   // If there's an MDI parent, must subtract the parent's top left corner 
 850   // since MoveWindow moves relative to the parent 
 851   wxMDIParentFrame 
*mdiParent 
= (wxMDIParentFrame 
*)GetParent(); 
 852   ::ScreenToClient((HWND
) mdiParent
->GetClientWindow()->GetHWND(), &point
); 
 854   MoveWindow(hWnd
, point
.x
, point
.y
, actual_width
, actual_height
, (BOOL
)true); 
 856   wxSize 
size(width
, height
); 
 857   wxSizeEvent 
event(size
, m_windowId
); 
 858   event
.SetEventObject( this ); 
 859   GetEventHandler()->ProcessEvent(event
); 
 862 void wxMDIChildFrame::DoGetPosition(int *x
, int *y
) const 
 865   GetWindowRect(GetHwnd(), &rect
); 
 870   // Since we now have the absolute screen coords, 
 871   // if there's a parent we must subtract its top left corner 
 872   wxMDIParentFrame 
*mdiParent 
= (wxMDIParentFrame 
*)GetParent(); 
 873   ::ScreenToClient((HWND
) mdiParent
->GetClientWindow()->GetHWND(), &point
); 
 881 void wxMDIChildFrame::InternalSetMenuBar() 
 883     wxMDIParentFrame 
*parent 
= (wxMDIParentFrame 
*)GetParent(); 
 885     InsertWindowMenu(parent
->GetClientWindow(), 
 886                      m_hMenu
, GetMDIWindowMenu(parent
)); 
 888     parent
->m_parentFrameActive 
= false; 
 891 void wxMDIChildFrame::DetachMenuBar() 
 893     RemoveWindowMenu(NULL
, m_hMenu
); 
 894     wxFrame::DetachMenuBar(); 
 897 WXHICON 
wxMDIChildFrame::GetDefaultIcon() const 
 899     // we don't have any standard icons (any more) 
 903 // --------------------------------------------------------------------------- 
 905 // --------------------------------------------------------------------------- 
 907 void wxMDIChildFrame::Maximize(bool maximize
) 
 909     wxMDIParentFrame 
*parent 
= (wxMDIParentFrame 
*)GetParent(); 
 910     if ( parent 
&& parent
->GetClientWindow() ) 
 912         ::SendMessage(GetWinHwnd(parent
->GetClientWindow()), 
 913                       maximize 
? WM_MDIMAXIMIZE 
: WM_MDIRESTORE
, 
 914                       (WPARAM
)GetHwnd(), 0); 
 918 void wxMDIChildFrame::Restore() 
 920     wxMDIParentFrame 
*parent 
= (wxMDIParentFrame 
*)GetParent(); 
 921     if ( parent 
&& parent
->GetClientWindow() ) 
 923         ::SendMessage(GetWinHwnd(parent
->GetClientWindow()), WM_MDIRESTORE
, 
 924                       (WPARAM
) GetHwnd(), 0); 
 928 void wxMDIChildFrame::Activate() 
 930     wxMDIParentFrame 
*parent 
= (wxMDIParentFrame 
*)GetParent(); 
 931     if ( parent 
&& parent
->GetClientWindow() ) 
 933         ::SendMessage(GetWinHwnd(parent
->GetClientWindow()), WM_MDIACTIVATE
, 
 934                       (WPARAM
) GetHwnd(), 0); 
 938 // --------------------------------------------------------------------------- 
 939 // MDI window proc and message handlers 
 940 // --------------------------------------------------------------------------- 
 942 WXLRESULT 
wxMDIChildFrame::MSWWindowProc(WXUINT message
, 
 947     bool processed 
= false; 
 955                 UnpackCommand((WXWPARAM
)wParam
, (WXLPARAM
)lParam
, 
 958                 processed 
= HandleCommand(id
, cmd
, (WXHWND
)hwnd
); 
 962         case WM_GETMINMAXINFO
: 
 963             processed 
= HandleGetMinMaxInfo((MINMAXINFO 
*)lParam
); 
 969                 WXHWND hwndAct
, hwndDeact
; 
 970                 UnpackMDIActivate(wParam
, lParam
, &act
, &hwndAct
, &hwndDeact
); 
 972                 processed 
= HandleMDIActivate(act
, hwndAct
, hwndDeact
); 
 977             // must pass WM_MOVE to DefMDIChildProc() to recalculate MDI client 
 978             // scrollbars if necessary 
 983             // must pass WM_SIZE to DefMDIChildProc(), otherwise many weird 
 985             MSWDefWindowProc(message
, wParam
, lParam
); 
 989             // DefMDIChildProc handles SC_{NEXT/PREV}WINDOW here, so pass it 
 990             // the message (the base class version does not) 
 991             return MSWDefWindowProc(message
, wParam
, lParam
); 
 993         case WM_WINDOWPOSCHANGING
: 
 994             processed 
= HandleWindowPosChanging((LPWINDOWPOS
)lParam
); 
 999         rc 
= wxFrame::MSWWindowProc(message
, wParam
, lParam
); 
1004 bool wxMDIChildFrame::HandleCommand(WXWORD id
, WXWORD cmd
, WXHWND hwnd
) 
1006     // In case it's e.g. a toolbar. 
1009         wxWindow 
*win 
= wxFindWinFromHandle(hwnd
); 
1011             return win
->MSWCommand(cmd
, id
); 
1014     if (wxCurrentPopupMenu
) 
1016         wxMenu 
*popupMenu 
= wxCurrentPopupMenu
; 
1017         wxCurrentPopupMenu 
= NULL
; 
1018         if (popupMenu
->MSWCommand(cmd
, id
)) 
1023     if (GetMenuBar() && GetMenuBar()->FindItem(id
)) 
1025         processed 
= ProcessCommand(id
); 
1035 bool wxMDIChildFrame::HandleMDIActivate(long WXUNUSED(activate
), 
1039     wxMDIParentFrame 
*parent 
= (wxMDIParentFrame 
*)GetParent(); 
1041     HMENU menuToSet 
= 0; 
1045     if ( m_hWnd 
== hwndAct 
) 
1048         parent
->m_currentChild 
= this; 
1050         HMENU child_menu 
= (HMENU
)GetWinMenu(); 
1053             parent
->m_parentFrameActive 
= false; 
1055             menuToSet 
= child_menu
; 
1058     else if ( m_hWnd 
== hwndDeact 
) 
1060         wxASSERT_MSG( parent
->m_currentChild 
== this, 
1061                       wxT("can't deactivate MDI child which wasn't active!") ); 
1064         parent
->m_currentChild 
= NULL
; 
1066         HMENU parent_menu 
= (HMENU
)parent
->GetWinMenu(); 
1068         // activate the the parent menu only when there is no other child 
1069         // that has been activated 
1070         if ( parent_menu 
&& !hwndAct 
) 
1072             parent
->m_parentFrameActive 
= true; 
1074             menuToSet 
= parent_menu
; 
1079         // we have nothing to do with it 
1085         MDISetMenu(parent
->GetClientWindow(), 
1086                    menuToSet
, GetMDIWindowMenu(parent
)); 
1089     wxActivateEvent 
event(wxEVT_ACTIVATE
, activated
, m_windowId
); 
1090     event
.SetEventObject( this ); 
1092     ResetWindowStyle((void *)NULL
); 
1094     return GetEventHandler()->ProcessEvent(event
); 
1097 bool wxMDIChildFrame::HandleWindowPosChanging(void *pos
) 
1099     WINDOWPOS 
*lpPos 
= (WINDOWPOS 
*)pos
; 
1101     if (!(lpPos
->flags 
& SWP_NOSIZE
)) 
1104         DWORD dwExStyle 
= ::GetWindowLong(GetHwnd(), GWL_EXSTYLE
); 
1105         DWORD dwStyle 
= ::GetWindowLong(GetHwnd(), GWL_STYLE
); 
1106         if (ResetWindowStyle((void *) & rectClient
) && (dwStyle 
& WS_MAXIMIZE
)) 
1108             ::AdjustWindowRectEx(&rectClient
, dwStyle
, false, dwExStyle
); 
1109             lpPos
->x 
= rectClient
.left
; 
1110             lpPos
->y 
= rectClient
.top
; 
1111             lpPos
->cx 
= rectClient
.right 
- rectClient
.left
; 
1112             lpPos
->cy 
= rectClient
.bottom 
- rectClient
.top
; 
1115         wxMDIParentFrame
* pFrameWnd 
= (wxMDIParentFrame 
*)GetParent(); 
1116         if (pFrameWnd 
&& pFrameWnd
->GetToolBar() && pFrameWnd
->GetToolBar()->IsShown()) 
1118             pFrameWnd
->GetToolBar()->Refresh(); 
1126 bool wxMDIChildFrame::HandleGetMinMaxInfo(void *mmInfo
) 
1128     MINMAXINFO 
*info 
= (MINMAXINFO 
*)mmInfo
; 
1130     // let the default window proc calculate the size of MDI children 
1131     // frames because it is based on the size of the MDI client window, 
1132     // not on the values specified in wxWindow m_max variables 
1133     bool processed 
= MSWDefWindowProc(WM_GETMINMAXINFO
, 0, (LPARAM
)mmInfo
) != 0; 
1135     int minWidth 
= GetMinWidth(), 
1136         minHeight 
= GetMinHeight(); 
1138     // but allow GetSizeHints() to set the min size 
1139     if ( minWidth 
!= wxDefaultCoord 
) 
1141         info
->ptMinTrackSize
.x 
= minWidth
; 
1146     if ( minHeight 
!= wxDefaultCoord 
) 
1148         info
->ptMinTrackSize
.y 
= minHeight
; 
1156 // --------------------------------------------------------------------------- 
1157 // MDI specific message translation/preprocessing 
1158 // --------------------------------------------------------------------------- 
1160 WXLRESULT 
wxMDIChildFrame::MSWDefWindowProc(WXUINT message
, WXWPARAM wParam
, WXLPARAM lParam
) 
1162     return DefMDIChildProc(GetHwnd(), 
1163                            (UINT
)message
, (WPARAM
)wParam
, (LPARAM
)lParam
); 
1166 bool wxMDIChildFrame::MSWTranslateMessage(WXMSG
* msg
) 
1168     return wxFrame::MSWTranslateMessage(msg
); 
1171 // --------------------------------------------------------------------------- 
1173 // --------------------------------------------------------------------------- 
1175 void wxMDIChildFrame::MSWDestroyWindow() 
1177     invalidHandle 
= GetHwnd(); 
1179     wxMDIParentFrame 
*parent 
= (wxMDIParentFrame 
*)GetParent(); 
1181     // Must make sure this handle is invalidated (set to NULL) since all sorts 
1182     // of things could happen after the child client is destroyed, but before 
1183     // the wxFrame is destroyed. 
1185     HWND oldHandle 
= (HWND
)GetHWND(); 
1186     SendMessage(GetWinHwnd(parent
->GetClientWindow()), WM_MDIDESTROY
, 
1187                 (WPARAM
)oldHandle
, 0); 
1189     if (parent
->GetActiveChild() == (wxMDIChildFrame
*) NULL
) 
1190         ResetWindowStyle((void*) NULL
); 
1196         ::DestroyMenu((HMENU
) m_hMenu
); 
1199     wxRemoveHandleAssociation(this); 
1203 // Change the client window's extended style so we don't get a client edge 
1204 // style when a child is maximised (a double border looks silly.) 
1205 bool wxMDIChildFrame::ResetWindowStyle(void *vrect
) 
1207     RECT 
*rect 
= (RECT 
*)vrect
; 
1208     wxMDIParentFrame
* pFrameWnd 
= (wxMDIParentFrame 
*)GetParent(); 
1209     wxMDIChildFrame
* pChild 
= pFrameWnd
->GetActiveChild(); 
1211     if (!pChild 
|| (pChild 
== this)) 
1213         HWND hwndClient 
= GetWinHwnd(pFrameWnd
->GetClientWindow()); 
1214         DWORD dwStyle 
= ::GetWindowLong(hwndClient
, GWL_EXSTYLE
); 
1216         // we want to test whether there is a maximized child, so just set 
1217         // dwThisStyle to 0 if there is no child at all 
1218         DWORD dwThisStyle 
= pChild
 
1219             ? ::GetWindowLong(GetWinHwnd(pChild
), GWL_STYLE
) : 0; 
1220         DWORD dwNewStyle 
= dwStyle
; 
1221         if ( dwThisStyle 
& WS_MAXIMIZE 
) 
1222             dwNewStyle 
&= ~(WS_EX_CLIENTEDGE
); 
1224             dwNewStyle 
|= WS_EX_CLIENTEDGE
; 
1226         if (dwStyle 
!= dwNewStyle
) 
1228             // force update of everything 
1229             ::RedrawWindow(hwndClient
, NULL
, NULL
, 
1230                            RDW_INVALIDATE 
| RDW_ALLCHILDREN
); 
1231             ::SetWindowLong(hwndClient
, GWL_EXSTYLE
, dwNewStyle
); 
1232             ::SetWindowPos(hwndClient
, NULL
, 0, 0, 0, 0, 
1233                            SWP_FRAMECHANGED 
| SWP_NOACTIVATE 
| 
1234                            SWP_NOMOVE 
| SWP_NOSIZE 
| SWP_NOZORDER 
| 
1237                 ::GetClientRect(hwndClient
, rect
); 
1246 // =========================================================================== 
1247 // wxMDIClientWindow: the window of predefined (by Windows) class which 
1248 // contains the child frames 
1249 // =========================================================================== 
1251 bool wxMDIClientWindow::CreateClient(wxMDIParentFrame 
*parent
, long style
) 
1253     m_backgroundColour 
= wxSystemSettings::GetColour(wxSYS_COLOUR_APPWORKSPACE
); 
1255     CLIENTCREATESTRUCT ccs
; 
1256     m_windowStyle 
= style
; 
1259     ccs
.hWindowMenu 
= GetMDIWindowMenu(parent
); 
1260     ccs
.idFirstChild 
= wxFIRST_MDI_CHILD
; 
1262     DWORD msStyle 
= MDIS_ALLCHILDSTYLES 
| WS_VISIBLE 
| WS_CHILD 
| 
1263                     WS_CLIPCHILDREN 
| WS_CLIPSIBLINGS
; 
1265     if ( style 
& wxHSCROLL 
) 
1266         msStyle 
|= WS_HSCROLL
; 
1267     if ( style 
& wxVSCROLL 
) 
1268         msStyle 
|= WS_VSCROLL
; 
1270     DWORD exStyle 
= WS_EX_CLIENTEDGE
; 
1272     wxWindowCreationHook 
hook(this); 
1273     m_hWnd 
= (WXHWND
)::CreateWindowEx
 
1283                         (LPSTR
)(LPCLIENTCREATESTRUCT
)&ccs
); 
1286         wxLogLastError(wxT("CreateWindowEx(MDI client)")); 
1291     SubclassWin(m_hWnd
); 
1296 // Explicitly call default scroll behaviour 
1297 void wxMDIClientWindow::OnScroll(wxScrollEvent
& event
) 
1299     // Note: for client windows, the scroll position is not set in 
1300     // WM_HSCROLL, WM_VSCROLL, so we can't easily determine what 
1301     // scroll position we're at. 
1302     // This makes it hard to paint patterns or bitmaps in the background, 
1303     // and have the client area scrollable as well. 
1305     if ( event
.GetOrientation() == wxHORIZONTAL 
) 
1306         m_scrollX 
= event
.GetPosition(); // Always returns zero! 
1308         m_scrollY 
= event
.GetPosition(); // Always returns zero! 
1313 void wxMDIClientWindow::DoSetSize(int x
, int y
, int width
, int height
, int sizeFlags
) 
1315     // Try to fix a problem whereby if you show an MDI child frame, then reposition the 
1316     // client area, you can end up with a non-refreshed portion in the client window 
1317     // (see OGL studio sample). So check if the position is changed and if so, 
1318     // redraw the MDI child frames. 
1320     const wxPoint oldPos 
= GetPosition(); 
1322     wxWindow::DoSetSize(x
, y
, width
, height
, sizeFlags 
| wxSIZE_FORCE
); 
1324     const wxPoint newPos 
= GetPosition(); 
1326     if ((newPos
.x 
!= oldPos
.x
) || (newPos
.y 
!= oldPos
.y
)) 
1330             wxWindowList::compatibility_iterator node 
= GetParent()->GetChildren().GetFirst(); 
1333                 wxWindow 
*child 
= node
->GetData(); 
1334                 if (child
->IsKindOf(CLASSINFO(wxMDIChildFrame
))) 
1336                    ::RedrawWindow(GetHwndOf(child
), 
1343                 node 
= node
->GetNext(); 
1349 void wxMDIChildFrame::OnIdle(wxIdleEvent
& event
) 
1351     // wxMSW prior to 2.5.3 created MDI child frames as visible, which resulted 
1352     // in flicker e.g. when the frame contained controls with non-trivial 
1353     // layout. Since 2.5.3, the frame is created hidden as all other top level 
1354     // windows. In order to maintain backward compatibility, the frame is shown 
1355     // in OnIdle, unless Show(false) was called by the programmer before. 
1356     if ( m_needsInitialShow 
) 
1361     // MDI child frames get their WM_SIZE when they're constructed but at this 
1362     // moment they don't have any children yet so all child windows will be 
1363     // positioned incorrectly when they are added later - to fix this, we 
1364     // generate an artificial size event here 
1365     if ( m_needsResize 
) 
1367         m_needsResize 
= false; // avoid any possibility of recursion 
1375 // --------------------------------------------------------------------------- 
1376 // non member functions 
1377 // --------------------------------------------------------------------------- 
1379 static void MDISetMenu(wxWindow 
*win
, HMENU hmenuFrame
, HMENU hmenuWindow
) 
1381     if ( hmenuFrame 
|| hmenuWindow 
) 
1383         if ( !::SendMessage(GetWinHwnd(win
), 
1386                             (LPARAM
)hmenuWindow
) ) 
1388             wxLogLastError(_T("SendMessage(WM_MDISETMENU)")); 
1392     // update menu bar of the parent window 
1393     wxWindow 
*parent 
= win
->GetParent(); 
1394     wxCHECK_RET( parent
, wxT("MDI client without parent frame? weird...") ); 
1396     ::SendMessage(GetWinHwnd(win
), WM_MDIREFRESHMENU
, 0, 0L); 
1398     ::DrawMenuBar(GetWinHwnd(parent
)); 
1401 static void InsertWindowMenu(wxWindow 
*win
, WXHMENU menu
, HMENU subMenu
) 
1403     // Try to insert Window menu in front of Help, otherwise append it. 
1404     HMENU hmenu 
= (HMENU
)menu
; 
1408         int N 
= GetMenuItemCount(hmenu
); 
1409         bool success 
= false; 
1410         for ( int i 
= 0; i 
< N
; i
++ ) 
1413             int chars 
= GetMenuString(hmenu
, i
, buf
, WXSIZEOF(buf
), MF_BYPOSITION
); 
1416                 wxLogLastError(wxT("GetMenuString")); 
1421             wxString 
strBuf(buf
); 
1422             if ( wxStripMenuCodes(strBuf
) == wxGetStockLabel(wxID_HELP
,false) ) 
1425                 ::InsertMenu(hmenu
, i
, MF_BYPOSITION 
| MF_POPUP 
| MF_STRING
, 
1426                              (UINT
)subMenu
, _("&Window")); 
1433             ::AppendMenu(hmenu
, MF_POPUP
, (UINT
)subMenu
, _("&Window")); 
1437     MDISetMenu(win
, hmenu
, subMenu
); 
1440 static void RemoveWindowMenu(wxWindow 
*win
, WXHMENU menu
) 
1442     HMENU hMenu 
= (HMENU
)menu
; 
1448         int N 
= ::GetMenuItemCount(hMenu
); 
1449         for ( int i 
= 0; i 
< N
; i
++ ) 
1451             if ( !::GetMenuString(hMenu
, i
, buf
, WXSIZEOF(buf
), MF_BYPOSITION
) ) 
1453                 // Ignore successful read of menu string with length 0 which 
1454                 // occurs, for example, for a maximized MDI childs system menu 
1455                 if ( ::GetLastError() != 0 ) 
1457                     wxLogLastError(wxT("GetMenuString")); 
1463             if ( wxStrcmp(buf
, _("&Window")) == 0 ) 
1465                 if ( !::RemoveMenu(hMenu
, i
, MF_BYPOSITION
) ) 
1467                     wxLogLastError(wxT("RemoveMenu")); 
1477         // we don't change the windows menu, but we update the main one 
1478         MDISetMenu(win
, hMenu
, NULL
); 
1482 static void UnpackMDIActivate(WXWPARAM wParam
, WXLPARAM lParam
, 
1483                               WXWORD 
*activate
, WXHWND 
*hwndAct
, WXHWND 
*hwndDeact
) 
1486     *hwndAct 
= (WXHWND
)lParam
; 
1487     *hwndDeact 
= (WXHWND
)wParam
; 
1490 #endif // wxUSE_MDI && !defined(__WXUNIVERSAL__)