]> git.saurik.com Git - wxWidgets.git/blame_incremental - src/msw/mdi.cpp
* SPACE -> TAB in filelist.txt
[wxWidgets.git] / src / msw / mdi.cpp
... / ...
CommitLineData
1/////////////////////////////////////////////////////////////////////////////
2// Name: src/msw/mdi.cpp
3// Purpose: MDI classes for wxMSW
4// Author: Julian Smart
5// Modified by:
6// Created: 04/01/98
7// RCS-ID: $Id$
8// Copyright: (c) Julian Smart and Markus Holzem
9// Licence: wxWindows license
10/////////////////////////////////////////////////////////////////////////////
11
12// ===========================================================================
13// declarations
14// ===========================================================================
15
16// ---------------------------------------------------------------------------
17// headers
18// ---------------------------------------------------------------------------
19
20#ifdef __GNUG__
21 #pragma implementation "mdi.h"
22#endif
23
24// For compilers that support precompilation, includes "wx.h".
25#include "wx/wxprec.h"
26
27#ifdef __BORLANDC__
28 #pragma hdrstop
29#endif
30
31#ifndef WX_PRECOMP
32 #include "wx/setup.h"
33 #include "wx/frame.h"
34 #include "wx/menu.h"
35 #include "wx/app.h"
36 #include "wx/utils.h"
37 #include "wx/dialog.h"
38 #if wxUSE_STATUSBAR
39 #include "wx/statusbr.h"
40 #endif
41 #include "wx/settings.h"
42 #include "wx/intl.h"
43 #include "wx/log.h"
44#endif
45
46#include "wx/mdi.h"
47#include "wx/msw/private.h"
48
49#if wxUSE_STATUSBAR && wxUSE_NATIVE_STATUSBAR
50 #include "wx/msw/statbr95.h"
51#endif
52
53#if wxUSE_TOOLBAR
54 #include "wx/toolbar.h"
55#endif // wxUSE_TOOLBAR
56
57#include <string.h>
58
59// ---------------------------------------------------------------------------
60// global variables
61// ---------------------------------------------------------------------------
62
63extern wxWindowList wxModelessWindows; // from dialog.cpp
64extern wxMenu *wxCurrentPopupMenu;
65
66extern const wxChar *wxMDIFrameClassName; // from app.cpp
67extern const wxChar *wxMDIChildFrameClassName;
68extern const wxChar *wxMDIChildFrameClassNameNoRedraw;
69
70extern void wxAssociateWinWithHandle(HWND hWnd, wxWindow *win);
71extern void wxRemoveHandleAssociation(wxWindow *win);
72
73static HWND invalidHandle = 0;
74
75// ---------------------------------------------------------------------------
76// constants
77// ---------------------------------------------------------------------------
78
79static const int IDM_WINDOWTILE = 4001;
80static const int IDM_WINDOWTILEHOR = 4001;
81static const int IDM_WINDOWCASCADE = 4002;
82static const int IDM_WINDOWICONS = 4003;
83static const int IDM_WINDOWNEXT = 4004;
84static const int IDM_WINDOWTILEVERT = 4005;
85
86// This range gives a maximum of 500 MDI children. Should be enough :-)
87static const int wxFIRST_MDI_CHILD = 4100;
88static const int wxLAST_MDI_CHILD = 4600;
89
90// Status border dimensions
91static const int wxTHICK_LINE_BORDER = 3;
92static const int wxTHICK_LINE_WIDTH = 1;
93
94// ---------------------------------------------------------------------------
95// private functions
96// ---------------------------------------------------------------------------
97
98// set the MDI menus (by sending the WM_MDISETMENU message) and update the menu
99// of the parent of win (which is supposed to be the MDI client window)
100static void MDISetMenu(wxWindow *win, HMENU hmenuFrame, HMENU hmenuWindow);
101
102// insert the window menu (subMenu) into menu just before "Help" submenu or at
103// the very end if not found
104static void InsertWindowMenu(wxWindow *win, WXHMENU menu, HMENU subMenu);
105
106// Remove the window menu
107static void RemoveWindowMenu(wxWindow *win, WXHMENU menu);
108
109// is this an id of an MDI child?
110inline bool IsMdiCommandId(int id)
111{
112 return (id >= wxFIRST_MDI_CHILD) && (id <= wxLAST_MDI_CHILD);
113}
114
115// unpack the parameters of WM_MDIACTIVATE message
116static void UnpackMDIActivate(WXWPARAM wParam, WXLPARAM lParam,
117 WXWORD *activate, WXHWND *hwndAct, WXHWND *hwndDeact);
118
119// return the HMENU of the MDI menu
120static inline HMENU GetMDIWindowMenu(wxMDIParentFrame *frame)
121{
122 wxMenu *menu = frame->GetWindowMenu();
123 return menu ? GetHmenuOf(menu) : 0;
124}
125
126// ===========================================================================
127// implementation
128// ===========================================================================
129
130// ---------------------------------------------------------------------------
131// wxWin macros
132// ---------------------------------------------------------------------------
133
134IMPLEMENT_DYNAMIC_CLASS(wxMDIParentFrame, wxFrame)
135IMPLEMENT_DYNAMIC_CLASS(wxMDIChildFrame, wxFrame)
136IMPLEMENT_DYNAMIC_CLASS(wxMDIClientWindow, wxWindow)
137
138BEGIN_EVENT_TABLE(wxMDIParentFrame, wxFrame)
139 EVT_SIZE(wxMDIParentFrame::OnSize)
140 EVT_SYS_COLOUR_CHANGED(wxMDIParentFrame::OnSysColourChanged)
141END_EVENT_TABLE()
142
143BEGIN_EVENT_TABLE(wxMDIChildFrame, wxFrame)
144 EVT_IDLE(wxMDIChildFrame::OnIdle)
145END_EVENT_TABLE()
146
147BEGIN_EVENT_TABLE(wxMDIClientWindow, wxWindow)
148 EVT_SCROLL(wxMDIClientWindow::OnScroll)
149END_EVENT_TABLE()
150
151// ===========================================================================
152// wxMDIParentFrame: the frame which contains the client window which manages
153// the children
154// ===========================================================================
155
156wxMDIParentFrame::wxMDIParentFrame()
157{
158 m_clientWindow = NULL;
159 m_currentChild = NULL;
160 m_windowMenu = (wxMenu*) NULL;
161 m_parentFrameActive = TRUE;
162}
163
164bool wxMDIParentFrame::Create(wxWindow *parent,
165 wxWindowID id,
166 const wxString& title,
167 const wxPoint& pos,
168 const wxSize& size,
169 long style,
170 const wxString& name)
171{
172 m_clientWindow = NULL;
173 m_currentChild = NULL;
174
175 // this style can be used to prevent a window from having the standard MDI
176 // "Window" menu
177 if ( style & wxFRAME_NO_WINDOW_MENU )
178 {
179 m_windowMenu = (wxMenu *)NULL;
180 }
181 else // normal case: we have the window menu, so construct it
182 {
183 m_windowMenu = new wxMenu;
184
185 m_windowMenu->Append(IDM_WINDOWCASCADE, _("&Cascade"));
186 m_windowMenu->Append(IDM_WINDOWTILEHOR, _("Tile &Horizontally"));
187 m_windowMenu->Append(IDM_WINDOWTILEVERT, _("Tile &Vertically"));
188 m_windowMenu->AppendSeparator();
189 m_windowMenu->Append(IDM_WINDOWICONS, _("&Arrange Icons"));
190 m_windowMenu->Append(IDM_WINDOWNEXT, _("&Next"));
191 }
192
193 m_parentFrameActive = TRUE;
194
195 if (!parent)
196 wxTopLevelWindows.Append(this);
197
198 SetName(name);
199 m_windowStyle = style;
200
201 if ( parent )
202 parent->AddChild(this);
203
204 if ( id > -1 )
205 m_windowId = id;
206 else
207 m_windowId = NewControlId();
208
209 long exflags;
210 long msflags = MSWGetCreateWindowFlags(&exflags);
211
212 if ( !wxWindow::MSWCreate(wxMDIFrameClassName,
213 title,
214 pos, size,
215 msflags,
216 exflags) )
217 {
218 return FALSE;
219 }
220
221 wxModelessWindows.Append(this);
222
223 // unlike (almost?) all other windows, frames are created hidden
224 m_isShown = FALSE;
225
226 return TRUE;
227}
228
229wxMDIParentFrame::~wxMDIParentFrame()
230{
231 DestroyChildren();
232
233 // already delete by DestroyChildren()
234 m_frameToolBar = NULL;
235 m_frameStatusBar = NULL;
236
237 if (m_windowMenu)
238 {
239 delete m_windowMenu;
240 m_windowMenu = (wxMenu*) NULL;
241 }
242
243 // the MDI frame menubar is not automatically deleted by Windows unlike for
244 // the normal frames
245 if ( m_hMenu )
246 {
247 ::DestroyMenu((HMENU)m_hMenu);
248 m_hMenu = NULL;
249 }
250
251 if ( m_clientWindow )
252 {
253 if ( m_clientWindow->MSWGetOldWndProc() )
254 m_clientWindow->UnsubclassWin();
255
256 m_clientWindow->SetHWND(0);
257 delete m_clientWindow;
258 }
259}
260
261#if wxUSE_MENUS_NATIVE
262
263void wxMDIParentFrame::InternalSetMenuBar()
264{
265 m_parentFrameActive = TRUE;
266
267 InsertWindowMenu(GetClientWindow(), m_hMenu, GetMDIWindowMenu(this));
268}
269
270#endif // wxUSE_MENUS_NATIVE
271
272void wxMDIParentFrame::SetWindowMenu(wxMenu* menu)
273{
274 if (m_windowMenu)
275 {
276 if (GetMenuBar())
277 {
278 // Remove old window menu
279 RemoveWindowMenu(GetClientWindow(), m_hMenu);
280 }
281
282 delete m_windowMenu;
283 m_windowMenu = (wxMenu*) NULL;
284 }
285
286 if (menu)
287 {
288 m_windowMenu = menu;
289 if (GetMenuBar())
290 {
291 InsertWindowMenu(GetClientWindow(), m_hMenu,
292 GetHmenuOf(m_windowMenu));
293 }
294 }
295}
296
297void wxMDIParentFrame::OnSize(wxSizeEvent&)
298{
299 if ( GetClientWindow() )
300 {
301 int width, height;
302 GetClientSize(&width, &height);
303
304 GetClientWindow()->SetSize(0, 0, width, height);
305 }
306}
307
308// Returns the active MDI child window
309wxMDIChildFrame *wxMDIParentFrame::GetActiveChild() const
310{
311 HWND hWnd = (HWND)::SendMessage(GetWinHwnd(GetClientWindow()),
312 WM_MDIGETACTIVE, 0, 0L);
313 if ( hWnd == 0 )
314 return NULL;
315 else
316 return (wxMDIChildFrame *)wxFindWinFromHandle((WXHWND) hWnd);
317}
318
319// Create the client window class (don't Create the window, just return a new
320// class)
321wxMDIClientWindow *wxMDIParentFrame::OnCreateClient()
322{
323 return new wxMDIClientWindow;
324}
325
326// Responds to colour changes, and passes event on to children.
327void wxMDIParentFrame::OnSysColourChanged(wxSysColourChangedEvent& event)
328{
329 if ( m_clientWindow )
330 {
331 m_clientWindow->SetBackgroundColour(wxSystemSettings::GetColour(wxSYS_COLOUR_APPWORKSPACE));
332 m_clientWindow->Refresh();
333 }
334
335 event.Skip();
336}
337
338WXHICON wxMDIParentFrame::GetDefaultIcon() const
339{
340 return (WXHICON)(wxSTD_MDIPARENTFRAME_ICON ? wxSTD_MDIPARENTFRAME_ICON
341 : wxDEFAULT_MDIPARENTFRAME_ICON);
342}
343
344// ---------------------------------------------------------------------------
345// MDI operations
346// ---------------------------------------------------------------------------
347
348void wxMDIParentFrame::Cascade()
349{
350 ::SendMessage(GetWinHwnd(GetClientWindow()), WM_MDICASCADE, 0, 0);
351}
352
353// TODO: add a direction argument (hor/vert)
354void wxMDIParentFrame::Tile()
355{
356 ::SendMessage(GetWinHwnd(GetClientWindow()), WM_MDITILE, MDITILE_HORIZONTAL, 0);
357}
358
359void wxMDIParentFrame::ArrangeIcons()
360{
361 ::SendMessage(GetWinHwnd(GetClientWindow()), WM_MDIICONARRANGE, 0, 0);
362}
363
364void wxMDIParentFrame::ActivateNext()
365{
366 ::SendMessage(GetWinHwnd(GetClientWindow()), WM_MDINEXT, 0, 0);
367}
368
369void wxMDIParentFrame::ActivatePrevious()
370{
371 ::SendMessage(GetWinHwnd(GetClientWindow()), WM_MDINEXT, 0, 1);
372}
373
374// ---------------------------------------------------------------------------
375// the MDI parent frame window proc
376// ---------------------------------------------------------------------------
377
378long wxMDIParentFrame::MSWWindowProc(WXUINT message,
379 WXWPARAM wParam,
380 WXLPARAM lParam)
381{
382 long rc = 0;
383 bool processed = FALSE;
384
385 switch ( message )
386 {
387 case WM_ACTIVATE:
388 {
389 WXWORD state, minimized;
390 WXHWND hwnd;
391 UnpackActivate(wParam, lParam, &state, &minimized, &hwnd);
392
393 processed = HandleActivate(state, minimized != 0, hwnd);
394 }
395 break;
396
397 case WM_COMMAND:
398 {
399 WXWORD id, cmd;
400 WXHWND hwnd;
401 UnpackCommand(wParam, lParam, &id, &hwnd, &cmd);
402
403 (void)HandleCommand(id, cmd, hwnd);
404
405 // even if the frame didn't process it, there is no need to try it
406 // once again (i.e. call wxFrame::HandleCommand()) - we just dud it,
407 // so pretend we processed the message anyhow
408 processed = TRUE;
409 }
410
411 // always pass this message DefFrameProc(), otherwise MDI menu
412 // commands (and sys commands - more surprizingly!) won't work
413 MSWDefWindowProc(message, wParam, lParam);
414 break;
415
416 case WM_CREATE:
417 m_clientWindow = OnCreateClient();
418 // Uses own style for client style
419 if ( !m_clientWindow->CreateClient(this, GetWindowStyleFlag()) )
420 {
421 wxLogMessage(_("Failed to create MDI parent frame."));
422
423 rc = -1;
424 }
425
426 processed = TRUE;
427 break;
428
429 case WM_ERASEBKGND:
430 processed = TRUE;
431
432 // we erase background ourselves
433 rc = TRUE;
434 break;
435
436 case WM_MENUSELECT:
437 {
438 WXWORD item, flags;
439 WXHMENU hmenu;
440 UnpackMenuSelect(wParam, lParam, &item, &flags, &hmenu);
441
442 if ( m_parentFrameActive )
443 {
444 processed = HandleMenuSelect(item, flags, hmenu);
445 }
446 else if (m_currentChild)
447 {
448 processed = m_currentChild->
449 HandleMenuSelect(item, flags, hmenu);
450 }
451 }
452 break;
453
454 case WM_SIZE:
455 // as we don't (usually) resize the MDI client to exactly fit the
456 // client area (we put it below the toolbar, above statusbar &c),
457 // we should not pass this one to DefFrameProc
458 break;
459 }
460
461 if ( !processed )
462 rc = wxFrame::MSWWindowProc(message, wParam, lParam);
463
464 return rc;
465}
466
467bool wxMDIParentFrame::HandleActivate(int state, bool minimized, WXHWND activate)
468{
469 bool processed = FALSE;
470
471 if ( wxWindow::HandleActivate(state, minimized, activate) )
472 {
473 // already processed
474 processed = TRUE;
475 }
476
477 // If this window is an MDI parent, we must also send an OnActivate message
478 // to the current child.
479 if ( (m_currentChild != NULL) &&
480 ((state == WA_ACTIVE) || (state == WA_CLICKACTIVE)) )
481 {
482 wxActivateEvent event(wxEVT_ACTIVATE, TRUE, m_currentChild->GetId());
483 event.SetEventObject( m_currentChild );
484 if ( m_currentChild->GetEventHandler()->ProcessEvent(event) )
485 processed = TRUE;
486 }
487
488 return processed;
489}
490
491bool wxMDIParentFrame::HandleCommand(WXWORD id, WXWORD cmd, WXHWND hwnd)
492{
493 // In case it's e.g. a toolbar.
494 if ( hwnd )
495 {
496 wxWindow *win = wxFindWinFromHandle(hwnd);
497 if ( win )
498 return win->MSWCommand(cmd, id);
499 }
500
501 // is it one of standard MDI commands?
502 WXWPARAM wParam = 0;
503 int msg;
504 switch ( id )
505 {
506 case IDM_WINDOWCASCADE:
507 msg = WM_MDICASCADE;
508 wParam = MDITILE_SKIPDISABLED;
509 break;
510
511 case IDM_WINDOWTILEHOR:
512 wParam |= MDITILE_HORIZONTAL;
513 // fall through
514
515 case IDM_WINDOWTILEVERT:
516 if ( !wParam )
517 wParam = MDITILE_VERTICAL;
518 msg = WM_MDITILE;
519 wParam |= MDITILE_SKIPDISABLED;
520 break;
521
522 case IDM_WINDOWICONS:
523 msg = WM_MDIICONARRANGE;
524 break;
525
526 case IDM_WINDOWNEXT:
527 msg = WM_MDINEXT;
528 break;
529
530 default:
531 msg = 0;
532 }
533
534 if ( msg )
535 {
536 ::SendMessage(GetWinHwnd(GetClientWindow()), msg, wParam, 0);
537
538 return TRUE;
539 }
540
541 // FIXME VZ: what does this test do??
542 if (id >= 0xF000)
543 {
544 return FALSE; // Get WndProc to call default proc
545 }
546
547 if ( IsMdiCommandId(id) )
548 {
549 wxWindowList::Node* node = GetChildren().GetFirst();
550 while ( node )
551 {
552 wxWindow* child = node->GetData();
553 if ( child->GetHWND() )
554 {
555 long childId = wxGetWindowId(child->GetHWND());
556 if (childId == (long)id)
557 {
558 ::SendMessage( GetWinHwnd(GetClientWindow()),
559 WM_MDIACTIVATE,
560 (WPARAM)child->GetHWND(), 0);
561 return TRUE;
562 }
563 }
564 node = node->GetNext();
565 }
566 }
567 else if ( m_parentFrameActive )
568 {
569 return ProcessCommand(id);
570 }
571 else if ( m_currentChild )
572 {
573 return m_currentChild->HandleCommand(id, cmd, hwnd);
574 }
575 else
576 {
577 // this shouldn't happen because it means that our messages are being
578 // lost (they're not sent to the parent frame nor to the children)
579 wxFAIL_MSG(wxT("MDI parent frame is not active, yet there is no active MDI child?"));
580 }
581
582 return FALSE;
583}
584
585long wxMDIParentFrame::MSWDefWindowProc(WXUINT message,
586 WXWPARAM wParam,
587 WXLPARAM lParam)
588{
589 WXHWND clientWnd;
590 if ( GetClientWindow() )
591 clientWnd = GetClientWindow()->GetHWND();
592 else
593 clientWnd = 0;
594
595 return DefFrameProc(GetHwnd(), (HWND)clientWnd, message, wParam, lParam);
596}
597
598bool wxMDIParentFrame::MSWTranslateMessage(WXMSG* msg)
599{
600 MSG *pMsg = (MSG *)msg;
601
602 // first let the current child get it
603 if ( m_currentChild && m_currentChild->GetHWND() &&
604 m_currentChild->MSWTranslateMessage(msg) )
605 {
606 return TRUE;
607 }
608
609 // then try out accel table (will also check the menu accels)
610 if ( wxFrame::MSWTranslateMessage(msg) )
611 {
612 return TRUE;
613 }
614
615 // finally, check for MDI specific built in accel keys
616 if ( pMsg->message == WM_KEYDOWN || pMsg->message == WM_SYSKEYDOWN )
617 {
618 if ( ::TranslateMDISysAccel(GetWinHwnd(GetClientWindow()), pMsg))
619 return TRUE;
620 }
621
622 return FALSE;
623}
624
625// ===========================================================================
626// wxMDIChildFrame
627// ===========================================================================
628
629void wxMDIChildFrame::Init()
630{
631 m_needsResize = TRUE;
632}
633
634bool wxMDIChildFrame::Create(wxMDIParentFrame *parent,
635 wxWindowID id,
636 const wxString& title,
637 const wxPoint& pos,
638 const wxSize& size,
639 long style,
640 const wxString& name)
641{
642 SetName(name);
643 wxWindowBase::Show(TRUE); // MDI child frame starts off shown
644
645 if ( id > -1 )
646 m_windowId = id;
647 else
648 m_windowId = (int)NewControlId();
649
650 if ( parent )
651 {
652 parent->AddChild(this);
653 }
654
655 int x = pos.x;
656 int y = pos.y;
657 int width = size.x;
658 int height = size.y;
659
660 MDICREATESTRUCT mcs;
661
662 mcs.szClass = style & wxNO_FULL_REPAINT_ON_RESIZE
663 ? wxMDIChildFrameClassNameNoRedraw
664 : wxMDIChildFrameClassName;
665 mcs.szTitle = title;
666 mcs.hOwner = wxGetInstance();
667 if (x > -1)
668 mcs.x = x;
669 else
670 mcs.x = CW_USEDEFAULT;
671
672 if (y > -1)
673 mcs.y = y;
674 else
675 mcs.y = CW_USEDEFAULT;
676
677 if (width > -1)
678 mcs.cx = width;
679 else
680 mcs.cx = CW_USEDEFAULT;
681
682 if (height > -1)
683 mcs.cy = height;
684 else
685 mcs.cy = CW_USEDEFAULT;
686
687 DWORD msflags = WS_OVERLAPPED | WS_CLIPCHILDREN | WS_THICKFRAME | WS_VISIBLE ;
688 if (style & wxMINIMIZE_BOX)
689 msflags |= WS_MINIMIZEBOX;
690 if (style & wxMAXIMIZE_BOX)
691 msflags |= WS_MAXIMIZEBOX;
692 if (style & wxTHICK_FRAME)
693 msflags |= WS_THICKFRAME;
694 if (style & wxSYSTEM_MENU)
695 msflags |= WS_SYSMENU;
696 if ((style & wxMINIMIZE) || (style & wxICONIZE))
697 msflags |= WS_MINIMIZE;
698 if (style & wxMAXIMIZE)
699 msflags |= WS_MAXIMIZE;
700 if (style & wxCAPTION)
701 msflags |= WS_CAPTION;
702
703 mcs.style = msflags;
704
705 mcs.lParam = 0;
706
707 wxWindowCreationHook hook(this);
708
709 m_hWnd = (WXHWND)::SendMessage(GetWinHwnd(parent->GetClientWindow()),
710 WM_MDICREATE, 0, (LONG)(LPSTR)&mcs);
711
712 wxAssociateWinWithHandle((HWND) GetHWND(), this);
713
714 wxModelessWindows.Append(this);
715
716 return TRUE;
717}
718
719wxMDIChildFrame::~wxMDIChildFrame()
720{
721 DestroyChildren();
722
723 // already deleted by DestroyChildren()
724 m_frameToolBar = NULL;
725 m_frameStatusBar = NULL;
726
727 RemoveWindowMenu(NULL, m_hMenu);
728
729 MSWDestroyWindow();
730}
731
732// Set the client size (i.e. leave the calculation of borders etc.
733// to wxWindows)
734void wxMDIChildFrame::DoSetClientSize(int width, int height)
735{
736 HWND hWnd = GetHwnd();
737
738 RECT rect;
739 ::GetClientRect(hWnd, &rect);
740
741 RECT rect2;
742 GetWindowRect(hWnd, &rect2);
743
744 // Find the difference between the entire window (title bar and all)
745 // and the client area; add this to the new client size to move the
746 // window
747 int actual_width = rect2.right - rect2.left - rect.right + width;
748 int actual_height = rect2.bottom - rect2.top - rect.bottom + height;
749
750 if (GetStatusBar() && GetStatusBar()->IsShown())
751 {
752 int sx, sy;
753 GetStatusBar()->GetSize(&sx, &sy);
754 actual_height += sy;
755 }
756
757 POINT point;
758 point.x = rect2.left;
759 point.y = rect2.top;
760
761 // If there's an MDI parent, must subtract the parent's top left corner
762 // since MoveWindow moves relative to the parent
763 wxMDIParentFrame *mdiParent = (wxMDIParentFrame *)GetParent();
764 ::ScreenToClient((HWND) mdiParent->GetClientWindow()->GetHWND(), &point);
765
766 MoveWindow(hWnd, point.x, point.y, actual_width, actual_height, (BOOL)TRUE);
767
768 wxSizeEvent event(wxSize(width, height), m_windowId);
769 event.SetEventObject( this );
770 GetEventHandler()->ProcessEvent(event);
771}
772
773void wxMDIChildFrame::DoGetPosition(int *x, int *y) const
774{
775 RECT rect;
776 GetWindowRect(GetHwnd(), &rect);
777 POINT point;
778 point.x = rect.left;
779 point.y = rect.top;
780
781 // Since we now have the absolute screen coords,
782 // if there's a parent we must subtract its top left corner
783 wxMDIParentFrame *mdiParent = (wxMDIParentFrame *)GetParent();
784 ::ScreenToClient((HWND) mdiParent->GetClientWindow()->GetHWND(), &point);
785
786 *x = point.x;
787 *y = point.y;
788}
789
790void wxMDIChildFrame::InternalSetMenuBar()
791{
792 wxMDIParentFrame *parent = (wxMDIParentFrame *)GetParent();
793
794 InsertWindowMenu(parent->GetClientWindow(),
795 m_hMenu, GetMDIWindowMenu(parent));
796
797 parent->m_parentFrameActive = FALSE;
798}
799
800WXHICON wxMDIChildFrame::GetDefaultIcon() const
801{
802 return (WXHICON)(wxSTD_MDICHILDFRAME_ICON ? wxSTD_MDICHILDFRAME_ICON
803 : wxDEFAULT_MDICHILDFRAME_ICON);
804}
805
806// ---------------------------------------------------------------------------
807// MDI operations
808// ---------------------------------------------------------------------------
809
810void wxMDIChildFrame::Maximize(bool maximize)
811{
812 wxMDIParentFrame *parent = (wxMDIParentFrame *)GetParent();
813 if ( parent && parent->GetClientWindow() )
814 {
815 ::SendMessage(GetWinHwnd(parent->GetClientWindow()),
816 maximize ? WM_MDIMAXIMIZE : WM_MDIRESTORE,
817 (WPARAM)GetHwnd(), 0);
818 }
819}
820
821void wxMDIChildFrame::Restore()
822{
823 wxMDIParentFrame *parent = (wxMDIParentFrame *)GetParent();
824 if ( parent && parent->GetClientWindow() )
825 {
826 ::SendMessage(GetWinHwnd(parent->GetClientWindow()), WM_MDIRESTORE,
827 (WPARAM) GetHwnd(), 0);
828 }
829}
830
831void wxMDIChildFrame::Activate()
832{
833 wxMDIParentFrame *parent = (wxMDIParentFrame *)GetParent();
834 if ( parent && parent->GetClientWindow() )
835 {
836 ::SendMessage(GetWinHwnd(parent->GetClientWindow()), WM_MDIACTIVATE,
837 (WPARAM) GetHwnd(), 0);
838 }
839}
840
841// ---------------------------------------------------------------------------
842// MDI window proc and message handlers
843// ---------------------------------------------------------------------------
844
845long wxMDIChildFrame::MSWWindowProc(WXUINT message,
846 WXWPARAM wParam,
847 WXLPARAM lParam)
848{
849 long rc = 0;
850 bool processed = FALSE;
851
852 switch ( message )
853 {
854 case WM_COMMAND:
855 {
856 WORD id, cmd;
857 WXHWND hwnd;
858 UnpackCommand((WXWPARAM)wParam, (WXLPARAM)lParam,
859 &id, &hwnd, &cmd);
860
861 processed = HandleCommand(id, cmd, (WXHWND)hwnd);
862 }
863 break;
864
865 case WM_GETMINMAXINFO:
866 processed = HandleGetMinMaxInfo((MINMAXINFO *)lParam);
867 break;
868
869 case WM_MDIACTIVATE:
870 {
871 WXWORD act;
872 WXHWND hwndAct, hwndDeact;
873 UnpackMDIActivate(wParam, lParam, &act, &hwndAct, &hwndDeact);
874
875 processed = HandleMDIActivate(act, hwndAct, hwndDeact);
876 }
877 // fall through
878
879 case WM_MOVE:
880 // must pass WM_MOVE to DefMDIChildProc() to recalculate MDI client
881 // scrollbars if necessary
882
883 // fall through
884
885 case WM_SIZE:
886 // must pass WM_SIZE to DefMDIChildProc(), otherwise many weird
887 // things happen
888 MSWDefWindowProc(message, wParam, lParam);
889 break;
890
891 case WM_SYSCOMMAND:
892 // DefMDIChildProc handles SC_{NEXT/PREV}WINDOW here, so pass it
893 // the message (the base class version does not)
894 return MSWDefWindowProc(message, wParam, lParam);
895
896 case WM_WINDOWPOSCHANGING:
897 processed = HandleWindowPosChanging((LPWINDOWPOS)lParam);
898 break;
899 }
900
901 if ( !processed )
902 rc = wxFrame::MSWWindowProc(message, wParam, lParam);
903
904 return rc;
905}
906
907bool wxMDIChildFrame::HandleCommand(WXWORD id, WXWORD cmd, WXHWND hwnd)
908{
909 // In case it's e.g. a toolbar.
910 if ( hwnd )
911 {
912 wxWindow *win = wxFindWinFromHandle(hwnd);
913 if (win)
914 return win->MSWCommand(cmd, id);
915 }
916
917 if (wxCurrentPopupMenu)
918 {
919 wxMenu *popupMenu = wxCurrentPopupMenu;
920 wxCurrentPopupMenu = NULL;
921 if (popupMenu->MSWCommand(cmd, id))
922 return TRUE;
923 }
924
925 bool processed;
926 if (GetMenuBar() && GetMenuBar()->FindItem(id))
927 {
928 processed = ProcessCommand(id);
929 }
930 else
931 {
932 processed = FALSE;
933 }
934
935 return processed;
936}
937
938bool wxMDIChildFrame::HandleMDIActivate(long WXUNUSED(activate),
939 WXHWND hwndAct,
940 WXHWND hwndDeact)
941{
942 wxMDIParentFrame *parent = (wxMDIParentFrame *)GetParent();
943
944 HMENU menuToSet = 0;
945
946 bool activated;
947
948 if ( m_hWnd == hwndAct )
949 {
950 activated = TRUE;
951 parent->m_currentChild = this;
952
953 HMENU child_menu = (HMENU)GetWinMenu();
954 if ( child_menu )
955 {
956 parent->m_parentFrameActive = FALSE;
957
958 menuToSet = child_menu;
959 }
960 }
961 else if ( m_hWnd == hwndDeact )
962 {
963 wxASSERT_MSG( parent->m_currentChild == this,
964 wxT("can't deactivate MDI child which wasn't active!") );
965
966 activated = FALSE;
967 parent->m_currentChild = NULL;
968
969 HMENU parent_menu = (HMENU)parent->GetWinMenu();
970
971 // activate the the parent menu only when there is no other child
972 // that has been activated
973 if ( parent_menu && !hwndAct )
974 {
975 parent->m_parentFrameActive = TRUE;
976
977 menuToSet = parent_menu;
978 }
979 }
980 else
981 {
982 // we have nothing to do with it
983 return FALSE;
984 }
985
986 if ( menuToSet )
987 {
988 MDISetMenu(parent->GetClientWindow(),
989 menuToSet, GetMDIWindowMenu(parent));
990 }
991
992 wxActivateEvent event(wxEVT_ACTIVATE, activated, m_windowId);
993 event.SetEventObject( this );
994
995 ResetWindowStyle((void *)NULL);
996
997 return GetEventHandler()->ProcessEvent(event);
998}
999
1000bool wxMDIChildFrame::HandleWindowPosChanging(void *pos)
1001{
1002 WINDOWPOS *lpPos = (WINDOWPOS *)pos;
1003#if defined(__WIN95__)
1004 if (!(lpPos->flags & SWP_NOSIZE))
1005 {
1006 RECT rectClient;
1007 DWORD dwExStyle = ::GetWindowLong(GetHwnd(), GWL_EXSTYLE);
1008 DWORD dwStyle = ::GetWindowLong(GetHwnd(), GWL_STYLE);
1009 if (ResetWindowStyle((void *) & rectClient) && (dwStyle & WS_MAXIMIZE))
1010 {
1011 ::AdjustWindowRectEx(&rectClient, dwStyle, FALSE, dwExStyle);
1012 lpPos->x = rectClient.left;
1013 lpPos->y = rectClient.top;
1014 lpPos->cx = rectClient.right - rectClient.left;
1015 lpPos->cy = rectClient.bottom - rectClient.top;
1016 }
1017 wxMDIParentFrame* pFrameWnd = (wxMDIParentFrame *)GetParent();
1018 if (pFrameWnd && pFrameWnd->GetToolBar() && pFrameWnd->GetToolBar()->IsShown())
1019 {
1020 pFrameWnd->GetToolBar()->Refresh();
1021 }
1022 }
1023#endif // Win95
1024
1025 return FALSE;
1026}
1027
1028bool wxMDIChildFrame::HandleGetMinMaxInfo(void *mmInfo)
1029{
1030 MINMAXINFO *info = (MINMAXINFO *)mmInfo;
1031
1032 // let the default window proc calculate the size of MDI children
1033 // frames because it is based on the size of the MDI client window,
1034 // not on the values specified in wxWindow m_max variables
1035 bool processed = MSWDefWindowProc(WM_GETMINMAXINFO, 0, (LPARAM)mmInfo) != 0;
1036
1037 // but allow GetSizeHints() to set the min size
1038 if ( m_minWidth != -1 )
1039 {
1040 info->ptMinTrackSize.x = m_minWidth;
1041
1042 processed = TRUE;
1043 }
1044
1045 if ( m_minHeight != -1 )
1046 {
1047 info->ptMinTrackSize.y = m_minHeight;
1048
1049 processed = TRUE;
1050 }
1051
1052 return TRUE;
1053}
1054
1055// ---------------------------------------------------------------------------
1056// MDI specific message translation/preprocessing
1057// ---------------------------------------------------------------------------
1058
1059long wxMDIChildFrame::MSWDefWindowProc(WXUINT message, WXUINT wParam, WXLPARAM lParam)
1060{
1061 return DefMDIChildProc(GetHwnd(),
1062 (UINT)message, (WPARAM)wParam, (LPARAM)lParam);
1063}
1064
1065bool wxMDIChildFrame::MSWTranslateMessage(WXMSG* msg)
1066{
1067 return wxFrame::MSWTranslateMessage(msg);
1068}
1069
1070// ---------------------------------------------------------------------------
1071// misc
1072// ---------------------------------------------------------------------------
1073
1074void wxMDIChildFrame::MSWDestroyWindow()
1075{
1076 invalidHandle = GetHwnd();
1077
1078 wxMDIParentFrame *parent = (wxMDIParentFrame *)GetParent();
1079
1080 // Must make sure this handle is invalidated (set to NULL) since all sorts
1081 // of things could happen after the child client is destroyed, but before
1082 // the wxFrame is destroyed.
1083
1084 HWND oldHandle = (HWND)GetHWND();
1085 SendMessage(GetWinHwnd(parent->GetClientWindow()), WM_MDIDESTROY,
1086 (WPARAM)oldHandle, 0);
1087
1088 if (parent->GetActiveChild() == (wxMDIChildFrame*) NULL)
1089 ResetWindowStyle((void*) NULL);
1090
1091 invalidHandle = 0;
1092
1093 if (m_hMenu)
1094 {
1095 ::DestroyMenu((HMENU) m_hMenu);
1096 m_hMenu = 0;
1097 }
1098 wxRemoveHandleAssociation(this);
1099 m_hWnd = 0;
1100}
1101
1102// Change the client window's extended style so we don't get a client edge
1103// style when a child is maximised (a double border looks silly.)
1104bool wxMDIChildFrame::ResetWindowStyle(void *vrect)
1105{
1106#if defined(__WIN95__)
1107 RECT *rect = (RECT *)vrect;
1108 wxMDIParentFrame* pFrameWnd = (wxMDIParentFrame *)GetParent();
1109 wxMDIChildFrame* pChild = pFrameWnd->GetActiveChild();
1110 if (!pChild || (pChild == this))
1111 {
1112 HWND hwndClient = GetWinHwnd(pFrameWnd->GetClientWindow());
1113 DWORD dwStyle = ::GetWindowLong(hwndClient, GWL_EXSTYLE);
1114
1115 // we want to test whether there is a maximized child, so just set
1116 // dwThisStyle to 0 if there is no child at all
1117 DWORD dwThisStyle = pChild
1118 ? ::GetWindowLong(GetWinHwnd(pChild), GWL_STYLE) : 0;
1119 DWORD dwNewStyle = dwStyle;
1120 if ( dwThisStyle & WS_MAXIMIZE )
1121 dwNewStyle &= ~(WS_EX_CLIENTEDGE);
1122 else
1123 dwNewStyle |= WS_EX_CLIENTEDGE;
1124
1125 if (dwStyle != dwNewStyle)
1126 {
1127 // force update of everything
1128 ::RedrawWindow(hwndClient, NULL, NULL,
1129 RDW_INVALIDATE | RDW_ALLCHILDREN);
1130 ::SetWindowLong(hwndClient, GWL_EXSTYLE, dwNewStyle);
1131 ::SetWindowPos(hwndClient, NULL, 0, 0, 0, 0,
1132 SWP_FRAMECHANGED | SWP_NOACTIVATE |
1133 SWP_NOMOVE | SWP_NOSIZE | SWP_NOZORDER |
1134 SWP_NOCOPYBITS);
1135 if (rect)
1136 ::GetClientRect(hwndClient, rect);
1137
1138 return TRUE;
1139 }
1140 }
1141#endif // Win95
1142
1143 return FALSE;
1144}
1145
1146// ===========================================================================
1147// wxMDIClientWindow: the window of predefined (by Windows) class which
1148// contains the child frames
1149// ===========================================================================
1150
1151bool wxMDIClientWindow::CreateClient(wxMDIParentFrame *parent, long style)
1152{
1153 m_backgroundColour = wxSystemSettings::GetColour(wxSYS_COLOUR_APPWORKSPACE);
1154
1155 CLIENTCREATESTRUCT ccs;
1156 m_windowStyle = style;
1157 m_parent = parent;
1158
1159 ccs.hWindowMenu = GetMDIWindowMenu(parent);
1160 ccs.idFirstChild = wxFIRST_MDI_CHILD;
1161
1162 DWORD msStyle = MDIS_ALLCHILDSTYLES | WS_VISIBLE | WS_CHILD |
1163 WS_CLIPCHILDREN | WS_CLIPSIBLINGS;
1164
1165 if ( style & wxHSCROLL )
1166 msStyle |= WS_HSCROLL;
1167 if ( style & wxVSCROLL )
1168 msStyle |= WS_VSCROLL;
1169
1170#if defined(__WIN95__)
1171 DWORD exStyle = WS_EX_CLIENTEDGE;
1172#else
1173 DWORD exStyle = 0;
1174#endif
1175
1176 wxWindowCreationHook hook(this);
1177 m_hWnd = (WXHWND)::CreateWindowEx
1178 (
1179 exStyle,
1180 wxT("MDICLIENT"),
1181 NULL,
1182 msStyle,
1183 0, 0, 0, 0,
1184 GetWinHwnd(parent),
1185 NULL,
1186 wxGetInstance(),
1187 (LPSTR)(LPCLIENTCREATESTRUCT)&ccs);
1188 if ( !m_hWnd )
1189 {
1190 wxLogLastError(wxT("CreateWindowEx(MDI client)"));
1191
1192 return FALSE;
1193 }
1194
1195 SubclassWin(m_hWnd);
1196
1197 return TRUE;
1198}
1199
1200// Explicitly call default scroll behaviour
1201void wxMDIClientWindow::OnScroll(wxScrollEvent& event)
1202{
1203 // Note: for client windows, the scroll position is not set in
1204 // WM_HSCROLL, WM_VSCROLL, so we can't easily determine what
1205 // scroll position we're at.
1206 // This makes it hard to paint patterns or bitmaps in the background,
1207 // and have the client area scrollable as well.
1208
1209 if ( event.GetOrientation() == wxHORIZONTAL )
1210 m_scrollX = event.GetPosition(); // Always returns zero!
1211 else
1212 m_scrollY = event.GetPosition(); // Always returns zero!
1213
1214 event.Skip();
1215}
1216
1217void wxMDIClientWindow::DoSetSize(int x, int y, int width, int height, int sizeFlags)
1218{
1219 // Try to fix a problem whereby if you show an MDI child frame, then reposition the
1220 // client area, you can end up with a non-refreshed portion in the client window
1221 // (see OGL studio sample). So check if the position is changed and if so,
1222 // redraw the MDI child frames.
1223
1224 wxPoint oldPos = GetPosition();
1225
1226 wxWindow::DoSetSize(x, y, width, height, sizeFlags);
1227
1228 wxPoint newPos = GetPosition();
1229
1230 if ((newPos.x != oldPos.x) || (newPos.y != oldPos.y))
1231 {
1232 if (GetParent())
1233 {
1234 wxNode* node = GetParent()->GetChildren().First();
1235 while (node)
1236 {
1237 wxWindow* child = (wxWindow*) node->Data();
1238 if (child->IsKindOf(CLASSINFO(wxMDIChildFrame)))
1239 {
1240 HWND hWnd = (HWND) child->GetHWND();
1241 ::RedrawWindow(hWnd, NULL, NULL, RDW_FRAME|RDW_ALLCHILDREN|RDW_INVALIDATE );
1242 }
1243 node = node->Next();
1244 }
1245 }
1246 }
1247}
1248
1249void wxMDIChildFrame::OnIdle(wxIdleEvent& event)
1250{
1251 // MDI child frames get their WM_SIZE when they're constructed but at this
1252 // moment they don't have any children yet so all child windows will be
1253 // positioned incorrectly when they are added later - to fix this, we
1254 // generate an artificial size event here
1255 if ( m_needsResize )
1256 {
1257 m_needsResize = FALSE; // avoid any possibility of recursion
1258
1259 SendSizeEvent();
1260 }
1261
1262 event.Skip();
1263}
1264
1265// ---------------------------------------------------------------------------
1266// non member functions
1267// ---------------------------------------------------------------------------
1268
1269static void MDISetMenu(wxWindow *win, HMENU hmenuFrame, HMENU hmenuWindow)
1270{
1271 ::SendMessage(GetWinHwnd(win), WM_MDISETMENU,
1272#ifdef __WIN32__
1273 (WPARAM)hmenuFrame, (LPARAM)hmenuWindow
1274#else
1275 0, MAKELPARAM(hmenuFrame, hmenuWindow)
1276#endif
1277 );
1278
1279 // update menu bar of the parent window
1280 wxWindow *parent = win->GetParent();
1281 wxCHECK_RET( parent, wxT("MDI client without parent frame? weird...") );
1282
1283#ifndef __WIN16__
1284 ::SendMessage(GetWinHwnd(win), WM_MDIREFRESHMENU, 0, 0L);
1285#endif
1286
1287 ::DrawMenuBar(GetWinHwnd(parent));
1288}
1289
1290static void InsertWindowMenu(wxWindow *win, WXHMENU menu, HMENU subMenu)
1291{
1292 // Try to insert Window menu in front of Help, otherwise append it.
1293 HMENU hmenu = (HMENU)menu;
1294
1295 if (subMenu)
1296 {
1297 int N = GetMenuItemCount(hmenu);
1298 bool success = FALSE;
1299 for ( int i = 0; i < N; i++ )
1300 {
1301 wxChar buf[256];
1302 int chars = GetMenuString(hmenu, i, buf, WXSIZEOF(buf), MF_BYPOSITION);
1303 if ( chars == 0 )
1304 {
1305 wxLogLastError(wxT("GetMenuString"));
1306
1307 continue;
1308 }
1309
1310 if ( wxStripMenuCodes(wxString(buf)).IsSameAs(_("Help")) )
1311 {
1312 success = TRUE;
1313 ::InsertMenu(hmenu, i, MF_BYPOSITION | MF_POPUP | MF_STRING,
1314 (UINT)subMenu, _("&Window"));
1315 break;
1316 }
1317 }
1318
1319 if ( !success )
1320 {
1321 ::AppendMenu(hmenu, MF_POPUP, (UINT)subMenu, _("&Window"));
1322 }
1323 }
1324
1325 MDISetMenu(win, hmenu, subMenu);
1326}
1327
1328static void RemoveWindowMenu(wxWindow *win, WXHMENU menu)
1329{
1330 HMENU hMenu = (HMENU)menu;
1331
1332 if ( hMenu )
1333 {
1334 wxChar buf[1024];
1335
1336 int N = ::GetMenuItemCount(hMenu);
1337 for ( int i = 0; i < N; i++ )
1338 {
1339 if ( !::GetMenuString(hMenu, i, buf, WXSIZEOF(buf), MF_BYPOSITION) )
1340 {
1341 wxLogLastError(wxT("GetMenuString"));
1342
1343 continue;
1344 }
1345
1346 if ( wxStrcmp(buf, _("&Window")) == 0 )
1347 {
1348 if ( !::RemoveMenu(hMenu, i, MF_BYPOSITION) )
1349 {
1350 wxLogLastError(wxT("RemoveMenu"));
1351 }
1352
1353 break;
1354 }
1355 }
1356 }
1357
1358 if ( win )
1359 {
1360 // we don't change the windows menu, but we update the main one
1361 MDISetMenu(win, hMenu, NULL);
1362 }
1363}
1364
1365static void UnpackMDIActivate(WXWPARAM wParam, WXLPARAM lParam,
1366 WXWORD *activate, WXHWND *hwndAct, WXHWND *hwndDeact)
1367{
1368#ifdef __WIN32__
1369 *activate = TRUE;
1370 *hwndAct = (WXHWND)lParam;
1371 *hwndDeact = (WXHWND)wParam;
1372#else // Win16
1373 *activate = (WXWORD)wParam;
1374 *hwndAct = (WXHWND)LOWORD(lParam);
1375 *hwndDeact = (WXHWND)HIWORD(lParam);
1376#endif // Win32/Win16
1377}