1 /////////////////////////////////////////////////////////////////////////////
 
   3 // Purpose:     wxMDIParentFrame, wxMDIChildFrame, wxMDIClientWindow
 
   4 // Author:      David Elliott
 
   8 // Copyright:   (c) 2003 David Elliott
 
   9 // Licence:     wxWindows licence
 
  10 /////////////////////////////////////////////////////////////////////////////
 
  12 #include "wx/wxprec.h"
 
  18 // #include "wx/cocoa/autorelease.h"
 
  19 #include "wx/cocoa/mbarman.h"
 
  21 #import <AppKit/NSWindow.h>
 
  22 #import <Foundation/NSNotification.h>
 
  23 // #import <AppKit/NSApplication.h>
 
  24 // #import <AppKit/NSView.h>
 
  26 #include <wx/listimpl.cpp>
 
  27 WX_DEFINE_LIST(wxCocoaMDIChildFrameList);
 
  29 WX_DECLARE_HASH_MAP(int, wxMDIChildFrame*, wxIntegerHash, wxIntegerEqual, wxIntMDIChildFrameHashMap);
 
  31 // ============================================================================
 
  32 // wxMDIParentFrameObserver
 
  33 // ============================================================================
 
  34 @interface wxMDIParentFrameObserver : NSObject
 
  36     wxMDIParentFrame *m_mdiParent;
 
  40 - (id)initWithWxMDIParentFrame: (wxMDIParentFrame *)mdiParent;
 
  41 - (void)windowDidBecomeMain: (NSNotification *)notification;
 
  42 @end // interface wxMDIParentFrameObserver : NSObject
 
  44 @implementation wxMDIParentFrameObserver : NSObject
 
  47     wxFAIL_MSG(wxT("[wxMDIParentFrameObserver -init] should never be called!"));
 
  52 - (id)initWithWxMDIParentFrame: (wxMDIParentFrame *)mdiParent
 
  55     m_mdiParent = mdiParent;
 
  59 - (void)windowDidBecomeMain: (NSNotification *)notification
 
  61     wxASSERT(m_mdiParent);
 
  62     m_mdiParent->WindowDidBecomeMain(notification);
 
  65 @end // implementation wxMDIParentFrameObserver : NSObject
 
  67 // ========================================================================
 
  69 // ========================================================================
 
  70 IMPLEMENT_DYNAMIC_CLASS(wxMDIParentFrame,wxFrame)
 
  71 BEGIN_EVENT_TABLE(wxMDIParentFrame,wxFrame)
 
  74 void wxMDIParentFrame::Init()
 
  76     m_clientWindow = NULL;
 
  77     m_currentChild = NULL;
 
  78     m_observer = [[wxMDIParentFrameObserver alloc]
 
  79             initWithWxMDIParentFrame:this];
 
  80     [[NSNotificationCenter defaultCenter] addObserver:m_observer
 
  81             selector:@selector(windowDidBecomeMain:)
 
  82             name:NSWindowDidBecomeMainNotification object:nil];
 
  85 bool wxMDIParentFrame::Create(wxWindow *parent,
 
  86         wxWindowID winid, const wxString& title,
 
  87         const wxPoint& pos, const wxSize& size,
 
  88         long style, const wxString& name)
 
  90     bool success = wxFrame::Create(parent,winid,title,pos,size,style,name);
 
  96 wxMDIParentFrame::~wxMDIParentFrame()
 
  98     for(wxCocoaMDIChildFrameList::compatibility_iterator node =
 
  99             m_mdiChildren.GetFirst(); node; node = m_mdiChildren.GetFirst())
 
 101         wxMDIChildFrame *child = node->GetData();
 
 104         wxASSERT_MSG(!m_mdiChildren.Find(child),
 
 105             wxT("MDI child didn't remove itself using RemoveMDIChild()"));
 
 107     [m_observer release];
 
 110 void wxMDIParentFrame::AddMDIChild(wxMDIChildFrame *child)
 
 112     m_mdiChildren.Append(child);
 
 115 void wxMDIParentFrame::RemoveMDIChild(wxMDIChildFrame *child)
 
 117     m_mdiChildren.DeleteObject(child);
 
 118     if(child==m_currentChild)
 
 119         SetActiveChild(NULL);
 
 122 wxMDIChildFrame *wxMDIParentFrame::GetActiveChild() const
 
 124     return m_currentChild;
 
 127 void wxMDIParentFrame::SetActiveChild(wxMDIChildFrame *child)
 
 129     m_currentChild = child;
 
 130     wxMenuBarManager::GetInstance()->UpdateMenuBar();
 
 133 wxMDIClientWindow *wxMDIParentFrame::GetClientWindow() const
 
 135     return m_clientWindow;
 
 138 wxMDIClientWindow *wxMDIParentFrame::OnCreateClient()
 
 140     m_clientWindow = new wxMDIClientWindow( this );
 
 141     return m_clientWindow;
 
 144 void wxMDIParentFrame::ActivateNext()
 
 148 void wxMDIParentFrame::ActivatePrevious()
 
 152 wxMenuBar *wxMDIParentFrame::GetAppMenuBar(wxCocoaNSWindow *win)
 
 154     if(m_currentChild && (win==this))
 
 155         return m_currentChild->GetAppMenuBar(win);
 
 156     return wxFrame::GetAppMenuBar(win);
 
 159 void wxMDIParentFrame::CocoaDelegate_windowDidBecomeKey(void)
 
 161     wxLogTrace(wxTRACE_COCOA,wxT("wxMDIParentFrame=%p::CocoaDelegate_windowDidBecomeKey"),this);
 
 162     if(sm_cocoaDeactivateWindow && sm_cocoaDeactivateWindow==m_currentChild)
 
 164         sm_cocoaDeactivateWindow = NULL;
 
 167     else if(sm_cocoaDeactivateWindow == this)
 
 169         sm_cocoaDeactivateWindow = NULL;
 
 176             NSWindow *nswin = m_currentChild->GetNSWindow();
 
 177             if(![nswin isMainWindow])
 
 178                 [nswin makeMainWindow];
 
 180         wxFrame::CocoaDelegate_windowDidBecomeKey();
 
 184 void wxMDIParentFrame::CocoaDelegate_windowDidResignKey(void)
 
 186     wxLogTrace(wxTRACE_COCOA,wxT("wxMDIParentFrame=%p::CocoaDelegate_windowDidResignKey"),this);
 
 188         wxFrame::CocoaDelegate_windowDidResignKey();
 
 190         sm_cocoaDeactivateWindow = this;
 
 193 // We should not become the main window as we aren't a document window
 
 194 // MDI "Children" should be the main window
 
 195 bool wxMDIParentFrame::Cocoa_canBecomeMainWindow(bool &canBecome)
 
 197     canBecome = m_mdiChildren.IsEmpty(); return true;
 
 200 void wxMDIParentFrame::WindowDidBecomeMain(NSNotification *notification)
 
 202     // If we aren't the key window, we don't care
 
 203     if(![m_cocoaNSWindow isKeyWindow])
 
 205     wxCocoaNSWindow *win = wxCocoaNSWindow::GetFromCocoa([notification object]);
 
 206     // If we are key and becoming main, that's great
 
 209     // If one of our children is becoming main, also great
 
 210     for(wxCocoaMDIChildFrameList::compatibility_iterator node =
 
 211             m_mdiChildren.GetFirst(); node; node = node->GetNext())
 
 213         wxMDIChildFrame *child = node->GetData();
 
 217     // Some other window is becoming main, but we are key
 
 218     // Make the new main window the key window
 
 219     [[notification object] makeKeyWindow];
 
 222         wxIntMDIChildFrameHashMap hashmap;
 
 223         for(wxCocoaMDIChildFrameList::compatibility_iterator node =
 
 224                 m_mdiChildren.GetFirst(); node; node = node->GetNext())
 
 226             wxMDIChildFrame *child = node->GetData();
 
 227             hashmap.insert(wxIntMDIChildFrameHashMap::value_type([child->m_cocoaNSWindow windowNumber],child));
 
 232             NSCountWindows(&windowCount);
 
 233             wxASSERT(windowCount>0);
 
 234             int *windowList = new int[windowCount];
 
 235             NSWindowList(windowCount, windowList);
 
 236             wxIntMDIChildFrameHashMap::iterator iter = hashmap.end();
 
 237             for(int i=0; i<windowCount && iter == hashmap.end(); i++)
 
 238                 iter=hashmap.find(windowList[i]);
 
 239             if(iter != hashmap.end())
 
 240                 m_currentChild = iter->second;
 
 245 // ========================================================================
 
 247 // ========================================================================
 
 248 IMPLEMENT_DYNAMIC_CLASS(wxMDIChildFrame,wxFrame)
 
 249 BEGIN_EVENT_TABLE(wxMDIChildFrame,wxFrame)
 
 252 void wxMDIChildFrame::Init()
 
 257 bool wxMDIChildFrame::Create(wxMDIParentFrame *parent,
 
 258         wxWindowID winid, const wxString& title,
 
 259         const wxPoint& pos, const wxSize& size,
 
 260         long style, const wxString& name)
 
 262     bool success = wxFrame::Create(parent,winid,title,pos,size,style,name);
 
 265         m_mdiParent = parent;
 
 266         parent->AddMDIChild(this);
 
 271 wxMDIChildFrame::~wxMDIChildFrame()
 
 273     // Just in case Destroy() wasn't called
 
 274     m_mdiParent->RemoveMDIChild(this);
 
 277 void wxMDIChildFrame::Activate()
 
 281 void wxMDIChildFrame::CocoaDelegate_windowDidBecomeKey(void)
 
 283     wxLogTrace(wxTRACE_COCOA,wxT("wxMDIChildFrame=%p::CocoaDelegate_windowDidBecomeKey"),this);
 
 284     if(sm_cocoaDeactivateWindow && sm_cocoaDeactivateWindow==m_mdiParent)
 
 286         sm_cocoaDeactivateWindow = NULL;
 
 287         if(m_mdiParent->GetActiveChild() != this)
 
 288             sm_cocoaDeactivateWindow = m_mdiParent->GetActiveChild();
 
 290     m_mdiParent->SetActiveChild(this);
 
 291     wxFrame::CocoaDelegate_windowDidBecomeKey();
 
 294 void wxMDIChildFrame::CocoaDelegate_windowDidBecomeMain(void)
 
 296     m_mdiParent->SetActiveChild(this);
 
 297     wxFrame::CocoaDelegate_windowDidBecomeMain();
 
 300 void wxMDIChildFrame::CocoaDelegate_windowDidResignKey(void)
 
 302     wxLogTrace(wxTRACE_COCOA,wxT("wxMDIChildFrame=%p::CocoaDelegate_windowDidResignKey"),this);
 
 303     sm_cocoaDeactivateWindow = this;
 
 306 bool wxMDIChildFrame::Destroy()
 
 308     // It's good to do this here before we are really closed
 
 309     m_mdiParent->RemoveMDIChild(this);
 
 310     return wxFrame::Destroy();
 
 313 // ========================================================================
 
 315 // ========================================================================
 
 316 IMPLEMENT_DYNAMIC_CLASS(wxMDIClientWindow,wxWindow)
 
 318 wxMDIClientWindow::wxMDIClientWindow()
 
 322 wxMDIClientWindow::wxMDIClientWindow(wxMDIParentFrame *parent, long style)
 
 323 :   wxWindow(parent, -1)
 
 327 wxMDIClientWindow::~wxMDIClientWindow()
 
 331 bool wxMDIClientWindow::CreateClient( wxMDIParentFrame *parent, long style)