X-Git-Url: https://git.saurik.com/wxWidgets.git/blobdiff_plain/a58a6614646ea85d5cceb26f88e61a4a5a460149..da3f3e049a3bcd9db16d73dd83c007e12f6c95e1:/src/cocoa/frame.mm?ds=sidebyside diff --git a/src/cocoa/frame.mm b/src/cocoa/frame.mm index 3ab4609843..9277c357f6 100644 --- a/src/cocoa/frame.mm +++ b/src/cocoa/frame.mm @@ -9,15 +9,23 @@ // Licence: wxWindows license ///////////////////////////////////////////////////////////////////////////// -#include "wx/frame.h" -#include "wx/menu.h" -#include "wx/menuitem.h" -#include "wx/app.h" -#include "wx/log.h" -#include "wx/statusbr.h" +#include "wx/wxprec.h" +#ifndef WX_PRECOMP + #include "wx/log.h" + #include "wx/app.h" + #include "wx/frame.h" + #include "wx/menu.h" + #include "wx/toolbar.h" + #include "wx/statusbr.h" +#endif // WX_PRECOMP + +#include "wx/cocoa/autorelease.h" +#include "wx/cocoa/mbarman.h" #import #import +#import +#import // wxFrame @@ -28,6 +36,7 @@ IMPLEMENT_DYNAMIC_CLASS(wxFrame, wxTopLevelWindow) void wxFrame::Init() { + m_frameNSView = nil; } bool wxFrame::Create(wxWindow *parent, @@ -45,82 +54,242 @@ bool wxFrame::Create(wxWindow *parent, wxFrame::~wxFrame() { + [m_frameNSView release]; } -void wxFrame::Cocoa_wxMenuItemAction(wxMenuItem& item) -{ - Command(item.GetId()); -} - +// ------------------------------------------------------------------- +// Menubar void wxFrame::AttachMenuBar(wxMenuBar *mbar) { wxFrameBase::AttachMenuBar(mbar); - if(m_frameMenuBar) - { - wxLogDebug("Attached menu"); - [m_cocoaNSWindow setMenu:m_frameMenuBar->GetNSMenu()]; - } + wxMenuBarManager::GetInstance()->UpdateMenuBar(); } void wxFrame::DetachMenuBar() { - if(m_frameMenuBar) + wxFrameBase::DetachMenuBar(); + wxMenuBarManager::GetInstance()->UpdateMenuBar(); +} + +void wxFrame::SetMenuBar(wxMenuBar *menubar) +{ + if ( menubar == GetMenuBar() ) { - [m_cocoaNSWindow setMenu:nil]; + // nothing to do + return; } + wxFrameBase::DetachMenuBar(); + wxFrameBase::AttachMenuBar(menubar); + wxMenuBarManager::GetInstance()->UpdateMenuBar(); } -bool wxFrame::Show(bool show) +wxMenuBar* wxFrame::GetAppMenuBar(wxCocoaNSWindow *win) { - bool ret = wxFrameBase::Show(show); - if(show && GetMenuBar()) - [wxTheApp->GetNSApplication() setMenu:GetMenuBar()->GetNSMenu() ]; - return ret; + if(GetMenuBar()) + return GetMenuBar(); + return wxFrameBase::GetAppMenuBar(win); } -void wxFrame::Cocoa_FrameChanged(void) +void wxFrame::CocoaDelegate_wxMenuItemAction(WX_NSMenuItem menuItem) { - PositionStatusBar(); - wxFrameBase::Cocoa_FrameChanged(); + wxLogTrace(wxTRACE_COCOA,wxT("wxFrame::wxMenuItemAction")); + wxMenuItem *item = wxMenuItem::GetFromCocoa(menuItem); + wxCHECK_RET(item,wxT("wxMenuItemAction received but no wxMenuItem exists!")); + + wxMenu *menu = item->GetMenu(); + wxCHECK_RET(menu,wxT("wxMenuItemAction received but wxMenuItem is not in a wxMenu!")); + + // Since we're handling the delegate messages there's a very good chance + // we'll receive a menu action from an item with a nil target. + wxMenuBar *menubar = menu->GetMenuBar(); + if(menubar) + { + wxFrame *frame = menubar->GetFrame(); + wxASSERT_MSG(frame==this, wxT("Received wxMenuItemAction in NSWindow delegate from a menu item attached to a different frame.")); + frame->ProcessCommand(item->GetId()); + } + else + wxLogDebug(wxT("Received wxMenuItemAction in NSWindow delegate from an unknown menu item.")); } +bool wxFrame::CocoaDelegate_validateMenuItem(WX_NSMenuItem menuItem) +{ + SEL itemAction = [menuItem action]; + if(itemAction == @selector(wxMenuItemAction:)) + { + wxMenuItem *item = wxMenuItem::GetFromCocoa(menuItem); + wxCHECK_MSG(item,false,wxT("validateMenuItem received but no wxMenuItem exists!")); + // TODO: do more sanity checking + return item->IsEnabled(); + } + // TODO: else if cut/copy/paste + wxLogDebug(wxT("Asked to validate an unknown menu item")); + return false; +} + +// ------------------------------------------------------------------- +// Origin/Size wxPoint wxFrame::GetClientAreaOrigin() const { return wxPoint(0,0); } -void wxFrame::DoGetClientSize(int *width, int *height) const +void wxFrame::CocoaSetWxWindowSize(int width, int height) +{ + if(m_frameStatusBar) + height += m_frameStatusBar->GetSize().y; +#if wxUSE_TOOLBAR + if(m_frameToolBar) + height += m_frameToolBar->GetSize().y; +#endif //wxUSE_TOOLBAR + wxTopLevelWindow::CocoaSetWxWindowSize(width,height); +} + +// ------------------------------------------------------------------- +WX_NSView wxFrame::GetNonClientNSView() { - wxFrameBase::DoGetClientSize(width,height); - if(height) + if(m_frameNSView) + return m_frameNSView; + return GetNSViewForSuperview(); +} + +void wxFrame::CocoaReplaceView(WX_NSView oldView, WX_NSView newView) +{ + // If we have the additional toolbar/statbar view, then the + // default replaceSubview will work. Otherwise, the old view + // should be the content view and should be replaced that way + if(m_frameNSView) + wxWindow::CocoaReplaceView(oldView, newView); + else + wxTopLevelWindow::CocoaReplaceView(oldView, newView); +} + +void wxFrame::UpdateFrameNSView() +{ + if(!m_frameNSView) + { + m_frameNSView = [[NSView alloc] initWithFrame:[[m_cocoaNSWindow contentView] frame]]; + [m_cocoaNSWindow setContentView: m_frameNSView]; + [m_frameNSView addSubview:m_cocoaNSView]; + } + NSRect frameRect = [m_frameNSView frame]; + float tbarheight = 0.0; +#if wxUSE_TOOLBAR + if(m_frameToolBar) { - if(m_frameStatusBar && m_frameStatusBar->IsShown()) - *height -= m_frameStatusBar->GetSize().y; + NSView *tbarNSView = m_frameToolBar->GetNSViewForSuperview(); + if(![tbarNSView superview]) + [m_frameNSView addSubview: tbarNSView]; + // Do this after addSubView so that SetSize can work + m_frameToolBar->SetSize(m_frameToolBar->DoGetBestSize()); + NSRect tbarRect = [tbarNSView frame]; + tbarRect.size.width = frameRect.size.width; + tbarRect.origin.x = 0.0; + tbarRect.origin.y = frameRect.size.height - tbarRect.size.height; + [tbarNSView setFrame:tbarRect]; + // width expands, bottom margin expands + [tbarNSView setAutoresizingMask: NSViewWidthSizable|NSViewMinYMargin]; + tbarheight = tbarRect.size.height; } +#endif //wxUSE_TOOLBAR + float sbarheight = 0.0; + if(m_frameStatusBar) + { + NSView *sbarNSView = m_frameStatusBar->GetNSViewForSuperview(); + if(![sbarNSView superview]) + [m_frameNSView addSubview: sbarNSView]; + NSRect sbarRect = [sbarNSView frame]; + sbarRect.size.width = frameRect.size.width; + sbarRect.origin.x = 0.0; + sbarRect.origin.y = 0.0; + [sbarNSView setFrame:sbarRect]; + // width expands, top margin expands + [sbarNSView setAutoresizingMask: NSViewWidthSizable|NSViewMaxYMargin]; + sbarheight = sbarRect.size.height; + } + wxLogTrace(wxTRACE_COCOA,wxT("frame height=%f, tbar=%f, sbar=%f"),frameRect.size.height,tbarheight,sbarheight); + NSRect innerRect = [m_cocoaNSView frame]; + innerRect.size.height = frameRect.size.height - tbarheight - sbarheight; + innerRect.origin.y = sbarheight; + [m_cocoaNSView setFrame:innerRect]; + [m_cocoaNSView setAutoresizingMask: NSViewWidthSizable|NSViewHeightSizable]; + // Don't let the frame get smaller than the toolbar+statusbar height + NSRect frameMinRect = [NSWindow frameRectForContentRect: + NSMakeRect(0.0,0.0,0.0,tbarheight+sbarheight) + styleMask: [m_cocoaNSWindow styleMask]]; + [m_cocoaNSWindow setMinSize:frameMinRect.size]; } -void wxFrame::DoSetClientSize(int width, int height) +void wxFrame::SetStatusBar(wxStatusBar *statusbar) { - if(m_frameStatusBar && m_frameStatusBar->IsShown()) - height += m_frameStatusBar->GetSize().y; - wxFrameBase::DoSetClientSize(width,height); + if(m_frameStatusBar) + { + [m_frameStatusBar->GetNSViewForSuperview() removeFromSuperview]; + [m_frameStatusBar->GetNSViewForSuperview() setAutoresizingMask: NSViewMinYMargin]; + if(m_frameStatusBar->GetParent()) + m_frameStatusBar->GetParent()->CocoaAddChild(m_frameStatusBar); + } + m_frameStatusBar = statusbar; + if(m_frameStatusBar) + { + m_frameStatusBar->CocoaRemoveFromParent(); + } + UpdateFrameNSView(); } -void wxFrame::PositionStatusBar() +wxStatusBar* wxFrame::CreateStatusBar(int number, + long style, + wxWindowID winid, + const wxString& name) { - if( !m_frameStatusBar || !m_frameStatusBar->IsShown() ) - return; + wxAutoNSAutoreleasePool pool; + wxFrameBase::CreateStatusBar(number,style,winid,name); + if(m_frameStatusBar) + { + m_frameStatusBar->CocoaRemoveFromParent(); + } + UpdateFrameNSView(); + return m_frameStatusBar; +} - // Get the client size. Since it excludes the StatusBar area we want - // the top of the status bar to be directly under it (thus located at h) - // The width of the statusbar should then match the client width - int w, h; - GetClientSize(&w, &h); +#if wxUSE_TOOLBAR +void wxFrame::SetToolBar(wxToolBar *toolbar) +{ + if(m_frameToolBar) + { + m_frameToolBar->SetOwningFrame(NULL); + [m_frameToolBar->GetNSViewForSuperview() removeFromSuperview]; + [m_frameToolBar->GetNSViewForSuperview() setAutoresizingMask: NSViewMinYMargin]; + if(m_frameToolBar->GetParent()) + m_frameToolBar->GetParent()->CocoaAddChild(m_frameToolBar); + } + m_frameToolBar = toolbar; + if(m_frameToolBar) + { + m_frameToolBar->CocoaRemoveFromParent(); + m_frameToolBar->SetOwningFrame(this); + } + UpdateFrameNSView(); +} - int sh; - m_frameStatusBar->GetSize(NULL, &sh); +wxToolBar* wxFrame::CreateToolBar(long style, + wxWindowID winid, + const wxString& name) +{ + wxAutoNSAutoreleasePool pool; + wxFrameBase::CreateToolBar(style,winid,name); + if(m_frameToolBar) + { + m_frameToolBar->CocoaRemoveFromParent(); + m_frameToolBar->SetOwningFrame(this); + } + UpdateFrameNSView(); + return m_frameToolBar; +} +#endif // wxUSE_TOOLBAR - m_frameStatusBar->SetSize(0, h, w, sh); +void wxFrame::PositionStatusBar() +{ }