1 /////////////////////////////////////////////////////////////////////////////
2 // Name: src/osx/carbon/toolbar.cpp
4 // Author: Stefan Csomor
7 // RCS-ID: $Id: toolbar.cpp 54954 2008-08-03 11:27:03Z VZ $
8 // Copyright: (c) Stefan Csomor
9 // Licence: wxWindows licence
10 /////////////////////////////////////////////////////////////////////////////
12 #include "wx/wxprec.h"
16 #include "wx/toolbar.h"
23 #include "wx/osx/private.h"
24 #include "wx/geometry.h"
25 #include "wx/sysopt.h"
27 const short kwxMacToolBarToolDefaultWidth = 16;
28 const short kwxMacToolBarToolDefaultHeight = 16;
29 const short kwxMacToolBarTopMargin = 4;
30 const short kwxMacToolBarLeftMargin = 4;
31 const short kwxMacToolBorder = 0;
32 const short kwxMacToolSpacing = 6;
34 BEGIN_EVENT_TABLE(wxToolBar, wxToolBarBase)
35 EVT_PAINT( wxToolBar::OnPaint )
40 #pragma mark Tool Implementation
42 // ----------------------------------------------------------------------------
44 // ----------------------------------------------------------------------------
46 // We have a dual implementation for each tool, WXWidget and NSToolbarItem*
48 // when embedding native controls in the native toolbar we must make sure the
49 // control does not get deleted behind our backs, so the retain count gets increased
50 // (after creation it is 1), first be the creation of the custom NSToolbarItem wrapper
51 // object, and second by the code 'creating' the custom HIView (which is the same as the
52 // already existing native control, therefore we just increase the ref count)
53 // when this view is removed from the native toolbar its count gets decremented again
54 // and when the HITooolbarItem wrapper object gets destroyed it is decremented as well
55 // so in the end the control lives with a refcount of one and can be disposed of by the
56 // wxControl code. For embedded controls on a non-native toolbar this ref count is less
57 // so we can only test against a range, not a specific value of the refcount.
59 class wxToolBarTool : public wxToolBarToolBase
65 const wxString& label,
66 const wxBitmap& bmpNormal,
67 const wxBitmap& bmpDisabled,
70 const wxString& shortHelp,
71 const wxString& longHelp );
73 wxToolBarTool(wxToolBar *tbar, wxControl *control, const wxString& label)
74 : wxToolBarToolBase(tbar, control, label)
78 SetControlHandle( (WXWidget) control->GetHandle() );
81 virtual ~wxToolBarTool()
86 WXWidget GetControlHandle()
88 return (WXWidget) m_controlHandle;
91 void SetControlHandle( WXWidget handle )
93 m_controlHandle = handle;
96 void SetPosition( const wxPoint& position );
100 if ( m_controlHandle )
104 [m_controlHandle retain];
108 // the embedded control is not under the responsibility of the tool, it gets disposed of in the
109 // proper wxControl destructor
111 m_controlHandle = NULL ;
114 #if wxOSX_USE_NATIVE_TOOLBAR
117 [m_toolbarItem release];
118 m_toolbarItem = NULL;
120 #endif // wxOSX_USE_NATIVE_TOOLBAR
123 wxSize GetSize() const
129 curSize = GetControl()->GetSize();
131 else if ( IsButton() )
133 curSize = GetToolBar()->GetToolSize();
138 curSize = GetToolBar()->GetToolSize();
139 if ( GetToolBar()->GetWindowStyleFlag() & (wxTB_LEFT|wxTB_RIGHT) )
148 wxPoint GetPosition() const
150 return wxPoint( m_x, m_y );
153 bool DoEnable( bool enable );
157 void UpdateToggleImage( bool toggle );
161 #if wxOSX_USE_NATIVE_TOOLBAR
164 // strip mnemonics from the label for compatibility with the usual
165 // labels in wxStaticText sense
166 wxString labelStr = wxStripMenuCodes(m_label);
167 wxCFStringRef l(labelStr, GetToolBarFontEncoding());
169 [m_toolbarItem setLabel:l.AsNSString()];
171 wxCFStringRef sh( GetShortHelp(), GetToolBarFontEncoding() );
172 [m_toolbarItem setToolTip:sh.AsNSString()];
179 wxToolBar *tbar = (wxToolBar*) GetToolBar();
184 shouldToggle = !IsToggled();
185 tbar->ToggleTool( GetId(), shouldToggle );
188 tbar->OnLeftClick( GetId(), IsToggled() );
191 #if wxOSX_USE_NATIVE_TOOLBAR
192 void SetToolbarItemRef( NSToolbarItem* ref )
194 if ( m_controlHandle )
195 [m_controlHandle setHidden:YES];
197 [m_toolbarItem release];
202 NSToolbarItem* GetToolbarItemRef() const
204 return m_toolbarItem;
207 void SetIndex( CFIndex idx )
212 CFIndex GetIndex() const
217 virtual void SetLabel(const wxString& label)
219 wxToolBarToolBase::SetLabel(label);
222 #endif // wxOSX_USE_NATIVE_TOOLBAR
225 #if wxOSX_USE_NATIVE_TOOLBAR
226 wxFontEncoding GetToolBarFontEncoding() const
230 f = GetToolBar()->GetFont();
231 return f.IsOk() ? f.GetEncoding() : wxFont::GetDefaultEncoding();
233 #endif // wxOSX_USE_NATIVE_TOOLBAR
237 m_controlHandle = NULL;
239 #if wxOSX_USE_NATIVE_TOOLBAR
240 m_toolbarItem = NULL;
245 WXWidget m_controlHandle;
248 wxBitmap m_alternateBitmap;
250 #if wxOSX_USE_NATIVE_TOOLBAR
251 NSToolbarItem* m_toolbarItem;
252 // position in its toolbar, -1 means not inserted
257 #if wxOSX_USE_NATIVE_TOOLBAR
259 @interface wxNSToolbarItem : NSToolbarItem
264 - (id) initWithItemIdentifier: (NSString*) identifier;
265 - (void)setImplementation: (wxToolBarTool *) theImplementation;
266 - (wxToolBarTool*) implementation;
267 - (void) clickedAction: (id) sender;
272 @interface wxNSToolbarDelegate : NSObject
276 - (NSToolbarItem *)toolbar:(NSToolbar *)toolbar itemForItemIdentifier:(NSString *)itemIdentifier willBeInsertedIntoToolbar:(BOOL)flag;
278 - (NSArray *)toolbarDefaultItemIdentifiers:(NSToolbar*)toolbar;
280 - (NSArray *)toolbarAllowedItemIdentifiers:(NSToolbar*)toolbar;
282 - (NSArray *)toolbarSelectableItemIdentifiers:(NSToolbar *)toolbar;
290 @interface wxNSToolBarButton : NSButton
295 - (id)initWithFrame:(NSRect)frame;
296 - (void) clickedAction: (id) sender;
297 - (void)setImplementation: (wxToolBarTool *) theImplementation;
298 - (wxToolBarTool*) implementation;
303 #if wxOSX_USE_NATIVE_TOOLBAR
305 @implementation wxNSToolbarItem
307 - (id)initWithItemIdentifier: (NSString*) identifier
309 [super initWithItemIdentifier:identifier];
311 [self setTarget: self];
312 [self setAction: @selector(clickedAction:)];
316 - (void) clickedAction: (id) sender
325 - (void)setImplementation: (wxToolBarTool *) theImplementation
327 impl = theImplementation;
330 - (wxToolBarTool*) implementation
337 @implementation wxNSToolbarDelegate
339 - (NSArray *)toolbarDefaultItemIdentifiers:(NSToolbar*)toolbar
341 wxUnusedVar(toolbar);
345 - (NSArray *)toolbarAllowedItemIdentifiers:(NSToolbar*)toolbar
347 wxUnusedVar(toolbar);
351 - (NSArray *)toolbarSelectableItemIdentifiers:(NSToolbar *)toolbar
353 wxUnusedVar(toolbar);
357 - (NSToolbarItem*) toolbar:(NSToolbar*) toolbar itemForItemIdentifier:(NSString*) itemIdentifier willBeInsertedIntoToolbar:(BOOL) flag
359 wxUnusedVar(toolbar);
361 wxToolBarTool* tool = (wxToolBarTool*) [itemIdentifier longLongValue];
363 wxToolBarTool* tool = (wxToolBarTool*) [itemIdentifier intValue];
367 wxNSToolbarItem* item = (wxNSToolbarItem*) tool->GetToolbarItemRef();
368 if ( flag && tool->IsControl() )
370 NSView* view = tool->GetControl()->GetHandle();
371 [view removeFromSuperview];
373 wxSize sz = tool->GetControl()->GetSize();
374 NSSize size = NSMakeSize((float)sz.x, (float)sz.y);
375 [item setMaxSize:size];
376 [item setMinSize:size];
388 @implementation wxNSToolBarButton
390 - (id)initWithFrame:(NSRect)frame
392 [super initWithFrame:frame];
394 [self setTarget: self];
395 [self setAction: @selector(clickedAction:)];
399 - (void) clickedAction: (id) sender
408 - (void)setImplementation: (wxToolBarTool *) theImplementation
410 impl = theImplementation;
413 - (wxToolBarTool*) implementation
425 bool wxToolBarTool::DoEnable( bool enable )
429 GetControl()->Enable( enable );
431 else if ( IsButton() )
433 #if wxOSX_USE_NATIVE_TOOLBAR
434 if ( m_toolbarItem != NULL )
435 [m_toolbarItem setEnabled:enable];
438 if ( m_controlHandle != NULL )
439 [(NSControl*)m_controlHandle setEnabled:enable];
445 void wxToolBarTool::SetPosition( const wxPoint& position )
450 int mac_x = position.x;
451 int mac_y = position.y;
455 NSRect frame = [m_controlHandle frame];
456 if ( frame.origin.x != mac_x || frame.origin.y != mac_y )
458 frame.origin.x = mac_x;
459 frame.origin.y = mac_y;
460 [m_controlHandle setFrame:frame];
463 else if ( IsControl() )
465 // embedded native controls are moved by the OS
466 #if wxOSX_USE_NATIVE_TOOLBAR
467 if ( ((wxToolBar*)GetToolBar())->MacWantsNativeToolbar() == false )
470 GetControl()->Move( position );
475 NSRect frame = [m_controlHandle frame];
476 if ( frame.origin.x != mac_x || frame.origin.y != mac_y )
478 frame.origin.x = mac_x;
479 frame.origin.y = mac_y;
480 [m_controlHandle setFrame:frame];
485 void wxToolBarTool::UpdateImages()
487 [(NSButton*) m_controlHandle setImage:m_bmpNormal.GetNSImage()];
489 if ( CanBeToggled() )
491 int w = m_bmpNormal.GetWidth();
492 int h = m_bmpNormal.GetHeight();
493 m_alternateBitmap = wxBitmap( w, h );
496 dc.SelectObject( m_alternateBitmap );
497 dc.SetPen( wxPen(*wxBLACK) );
498 dc.SetBrush( wxBrush( *wxLIGHT_GREY ));
499 dc.DrawRectangle( 0, 0, w, h );
500 dc.DrawBitmap( m_bmpNormal, 0, 0, true );
501 dc.SelectObject( wxNullBitmap );
503 [(NSButton*) m_controlHandle setAlternateImage:m_alternateBitmap.GetNSImage()];
505 UpdateToggleImage( CanBeToggled() && IsToggled() );
508 void wxToolBarTool::UpdateToggleImage( bool toggle )
510 #if wxOSX_USE_NATIVE_TOOLBAR
511 if (m_toolbarItem != NULL )
513 // the native toolbar item only has a 'selected' state (one for one toolbar)
514 // so we emulate the toggle here
515 if ( CanBeToggled() && toggle )
516 [m_toolbarItem setImage:m_alternateBitmap.GetNSImage()];
518 [m_toolbarItem setImage:m_bmpNormal.GetNSImage()];
523 wxToolBarTool::wxToolBarTool(
526 const wxString& label,
527 const wxBitmap& bmpNormal,
528 const wxBitmap& bmpDisabled,
530 wxObject *clientData,
531 const wxString& shortHelp,
532 const wxString& longHelp )
535 tbar, id, label, bmpNormal, bmpDisabled, kind,
536 clientData, shortHelp, longHelp )
542 #pragma mark Toolbar Implementation
544 wxToolBarToolBase *wxToolBar::CreateTool(
546 const wxString& label,
547 const wxBitmap& bmpNormal,
548 const wxBitmap& bmpDisabled,
550 wxObject *clientData,
551 const wxString& shortHelp,
552 const wxString& longHelp )
554 return new wxToolBarTool(
555 this, id, label, bmpNormal, bmpDisabled, kind,
556 clientData, shortHelp, longHelp );
560 wxToolBar::CreateTool(wxControl *control, const wxString& label)
562 return new wxToolBarTool(this, control, label);
565 void wxToolBar::Init()
569 m_defaultWidth = kwxMacToolBarToolDefaultWidth;
570 m_defaultHeight = kwxMacToolBarToolDefaultHeight;
572 #if wxOSX_USE_NATIVE_TOOLBAR
574 m_macUsesNativeToolbar = false;
578 // also for the toolbar we have the dual implementation:
579 // only when MacInstallNativeToolbar is called is the native toolbar set as the window toolbar
581 bool wxToolBar::Create(
587 const wxString& name )
589 if ( !wxToolBarBase::Create( parent, id, pos, size, style, wxDefaultValidator, name ) )
594 OSStatus err = noErr;
596 #if wxOSX_USE_NATIVE_TOOLBAR
598 if (parent->IsKindOf(CLASSINFO(wxFrame)) && wxSystemOptions::GetOptionInt(wxT("mac.toolbar.no-native")) != 1)
600 static wxNSToolbarDelegate* controller = nil;
602 if ( controller == nil )
603 controller = [[wxNSToolbarDelegate alloc] init];
604 wxString identifier = wxString::Format( wxT("%p"), this );
605 wxCFStringRef cfidentifier(identifier);
606 NSToolbar* tb = [[NSToolbar alloc] initWithIdentifier:cfidentifier.AsNSString()];
610 if (m_macToolbar != NULL)
612 [tb setDelegate:controller];
614 NSToolbarDisplayMode mode = NSToolbarDisplayModeDefault;
615 NSToolbarSizeMode displaySize = NSToolbarSizeModeSmall;
617 if ( style & wxTB_NOICONS )
618 mode = NSToolbarDisplayModeLabelOnly;
619 else if ( style & wxTB_TEXT )
620 mode = NSToolbarDisplayModeIconAndLabel;
622 mode = NSToolbarDisplayModeIconOnly;
624 [tb setDisplayMode:mode];
625 [tb setSizeMode:displaySize];
628 #endif // wxOSX_USE_NATIVE_TOOLBAR
630 return (err == noErr);
633 wxToolBar::~wxToolBar()
635 #if wxOSX_USE_NATIVE_TOOLBAR
636 if (m_macToolbar != NULL)
638 // if this is the installed toolbar, then deinstall it
639 if (m_macUsesNativeToolbar)
640 MacInstallNativeToolbar( false );
642 [(NSToolbar*)m_macToolbar setDelegate:nil];
643 [(NSToolbar*)m_macToolbar release];
649 bool wxToolBar::Show( bool show )
651 WXWindow tlw = MacGetTopLevelWindowRef();
652 bool bResult = (tlw != NULL);
656 #if wxOSX_USE_NATIVE_TOOLBAR
657 bool ownToolbarInstalled = false;
658 MacTopLevelHasNativeToolbar( &ownToolbarInstalled );
659 if (ownToolbarInstalled)
661 bResult = ([(NSToolbar*)m_macToolbar isVisible] != show);
663 [(NSToolbar*)m_macToolbar setVisible:show];
666 bResult = wxToolBarBase::Show( show );
669 bResult = wxToolBarBase::Show( show );
676 bool wxToolBar::IsShown() const
680 #if wxOSX_USE_NATIVE_TOOLBAR
681 bool ownToolbarInstalled;
683 MacTopLevelHasNativeToolbar( &ownToolbarInstalled );
684 if (ownToolbarInstalled)
686 bResult = [(NSToolbar*)m_macToolbar isVisible];
689 bResult = wxToolBarBase::IsShown();
692 bResult = wxToolBarBase::IsShown();
698 void wxToolBar::DoGetSize( int *width, int *height ) const
700 #if wxOSX_USE_NATIVE_TOOLBAR
701 bool ownToolbarInstalled;
703 MacTopLevelHasNativeToolbar( &ownToolbarInstalled );
704 if ( ownToolbarInstalled )
706 WXWindow tlw = MacGetTopLevelWindowRef();
707 float toolbarHeight = 0.0;
708 NSRect windowFrame = NSMakeRect(0, 0, 0, 0);
710 if(m_macToolbar && [(NSToolbar*)m_macToolbar isVisible])
712 windowFrame = [NSWindow contentRectForFrameRect:[tlw frame]
713 styleMask:[tlw styleMask]];
714 toolbarHeight = NSHeight(windowFrame)
715 - NSHeight([[tlw contentView] frame]);
719 *width = (int)windowFrame.size.width;
720 if ( height != NULL )
721 *height = (int)toolbarHeight;
724 wxToolBarBase::DoGetSize( width, height );
727 wxToolBarBase::DoGetSize( width, height );
731 wxSize wxToolBar::DoGetBestSize() const
735 DoGetSize( &width, &height );
737 return wxSize( width, height );
740 void wxToolBar::SetWindowStyleFlag( long style )
742 wxToolBarBase::SetWindowStyleFlag( style );
744 #if wxOSX_USE_NATIVE_TOOLBAR
745 if (m_macToolbar != NULL)
747 NSToolbarDisplayMode mode = NSToolbarDisplayModeDefault;
749 if ( style & wxTB_NOICONS )
750 mode = NSToolbarDisplayModeLabelOnly;
751 else if ( style & wxTB_TEXT )
752 mode = NSToolbarDisplayModeIconAndLabel;
754 mode = NSToolbarDisplayModeIconOnly;
756 [(NSToolbar*) m_macToolbar setDisplayMode:mode];
761 #if wxOSX_USE_NATIVE_TOOLBAR
762 bool wxToolBar::MacWantsNativeToolbar()
764 return m_macUsesNativeToolbar;
767 bool wxToolBar::MacTopLevelHasNativeToolbar(bool *ownToolbarInstalled) const
769 bool bResultV = false;
771 if (ownToolbarInstalled != NULL)
772 *ownToolbarInstalled = false;
774 WXWindow tlw = MacGetTopLevelWindowRef();
777 NSToolbar* curToolbarRef = [tlw toolbar];
778 bResultV = (curToolbarRef != NULL);
779 if (bResultV && (ownToolbarInstalled != NULL))
780 *ownToolbarInstalled = (curToolbarRef == m_macToolbar);
786 bool wxToolBar::MacInstallNativeToolbar(bool usesNative)
788 bool bResult = false;
790 if (usesNative && (m_macToolbar == NULL))
793 if (usesNative && ((GetWindowStyleFlag() & (wxTB_LEFT|wxTB_RIGHT|wxTB_BOTTOM)) != 0))
796 WXWindow tlw = MacGetTopLevelWindowRef();
800 // check the existing toolbar
801 NSToolbar* curToolbarRef = [tlw toolbar];
803 m_macUsesNativeToolbar = usesNative;
805 if (m_macUsesNativeToolbar)
807 // only install toolbar if there isn't one installed already
808 if (curToolbarRef == NULL)
811 [tlw setToolbar:(NSToolbar*) m_macToolbar];
812 [(NSToolbar*) m_macToolbar setVisible:YES];
814 m_peer->Move(0,0,0,0 );
815 SetSize( wxSIZE_AUTO_WIDTH, 0 );
816 m_peer->SetVisibility( false );
817 wxToolBarBase::Show( false );
822 // only deinstall toolbar if this is the installed one
823 if (m_macToolbar == curToolbarRef)
826 [(NSToolbar*) m_macToolbar setVisible:NO];
827 [tlw setToolbar:nil];
828 m_peer->SetVisibility( true );
833 InvalidateBestSize();
835 // wxLogDebug( wxT(" --> [%lx] - result [%s]"), (long)this, bResult ? wxT("T") : wxT("F") );
840 bool wxToolBar::Realize()
842 if ( !wxToolBarBase::Realize() )
848 int maxToolWidth = 0;
849 int maxToolHeight = 0;
851 int x = m_xMargin + kwxMacToolBarLeftMargin;
852 int y = m_yMargin + kwxMacToolBarTopMargin;
857 // find the maximum tool width and height
859 wxToolBarToolsList::compatibility_iterator node = m_tools.GetFirst();
862 tool = (wxToolBarTool *) node->GetData();
865 wxSize sz = tool->GetSize();
867 if ( sz.x > maxToolWidth )
869 if ( sz.y > maxToolHeight )
870 maxToolHeight = sz.y;
873 node = node->GetNext();
876 bool lastIsRadio = false;
877 bool curIsRadio = false;
879 #if wxOSX_USE_NATIVE_TOOLBAR
880 CFIndex currentPosition = 0;
881 bool insertAll = false;
883 NSToolbar* refTB = (NSToolbar*)m_macToolbar;
888 enc = f.GetEncoding();
890 enc = wxFont::GetDefaultEncoding();
893 node = m_tools.GetFirst();
896 tool = (wxToolBarTool*) node->GetData();
899 node = node->GetNext();
903 // set tool position:
904 // for the moment just perform a single row/column alignment
905 wxSize cursize = tool->GetSize();
906 if ( x + cursize.x > maxWidth )
907 maxWidth = x + cursize.x;
908 if ( y + cursize.y > maxHeight )
909 maxHeight = y + cursize.y;
911 if ( GetWindowStyleFlag() & (wxTB_LEFT|wxTB_RIGHT) )
913 int x1 = x + ( maxToolWidth - cursize.x ) / 2;
914 tool->SetPosition( wxPoint(x1, y) );
918 int y1 = y + ( maxToolHeight - cursize.y ) / 2;
919 tool->SetPosition( wxPoint(x, y1) );
922 // update the item positioning state
923 if ( GetWindowStyleFlag() & (wxTB_LEFT|wxTB_RIGHT) )
924 y += cursize.y + kwxMacToolSpacing;
926 x += cursize.x + kwxMacToolSpacing;
928 #if wxOSX_USE_NATIVE_TOOLBAR
929 // install in native NSToolbar
932 NSToolbarItem* hiItemRef = tool->GetToolbarItemRef();
933 if ( hiItemRef != NULL )
935 // since setting the help texts is non-virtual we have to update
937 wxCFStringRef sh( tool->GetShortHelp(), enc);
938 [hiItemRef setToolTip:sh.AsNSString()];
940 if ( insertAll || (tool->GetIndex() != currentPosition) )
946 // if this is the first tool that gets newly inserted or repositioned
947 // first remove all 'old' tools from here to the right, because of this
948 // all following tools will have to be reinserted (insertAll).
949 for ( wxToolBarToolsList::compatibility_iterator node2 = m_tools.GetLast();
951 node2 = node2->GetPrevious() )
953 wxToolBarTool *tool2 = (wxToolBarTool*) node2->GetData();
955 const long idx = tool2->GetIndex();
958 [refTB removeItemAtIndex:idx];
963 wxString identifier = wxString::Format( wxT("%ld"), (long) tool );
964 wxCFStringRef cfidentifier(identifier);
966 [refTB insertItemWithItemIdentifier:cfidentifier.AsNSString() atIndex:currentPosition];
967 tool->SetIndex( currentPosition );
975 // update radio button (and group) state
976 lastIsRadio = curIsRadio;
977 curIsRadio = ( tool->IsButton() && (tool->GetKind() == wxITEM_RADIO) );
981 if ( tool->IsToggled() )
982 DoToggleTool( tool, true );
988 if ( tool->Toggle( true ) )
990 DoToggleTool( tool, true );
993 else if ( tool->IsToggled() )
995 if ( tool->IsToggled() )
996 DoToggleTool( tool, true );
998 wxToolBarToolsList::compatibility_iterator nodePrev = node->GetPrevious();
1001 wxToolBarToolBase *toggleTool = nodePrev->GetData();
1002 if ( (toggleTool == NULL) || !toggleTool->IsButton() || (toggleTool->GetKind() != wxITEM_RADIO) )
1005 if ( toggleTool->Toggle( false ) )
1006 DoToggleTool( toggleTool, false );
1008 nodePrev = nodePrev->GetPrevious();
1013 node = node->GetNext();
1016 if ( GetWindowStyleFlag() & (wxTB_TOP|wxTB_BOTTOM) )
1018 // if not set yet, only one row
1019 if ( m_maxRows <= 0 )
1022 m_minWidth = maxWidth;
1024 maxHeight += m_yMargin + kwxMacToolBarTopMargin;
1025 m_minHeight = m_maxHeight = maxHeight;
1029 // if not set yet, have one column
1030 if ( (GetToolsCount() > 0) && (m_maxRows <= 0) )
1031 SetRows( GetToolsCount() );
1033 m_minHeight = maxHeight;
1035 maxWidth += m_xMargin + kwxMacToolBarLeftMargin;
1036 m_minWidth = m_maxWidth = maxWidth;
1040 // FIXME: should this be OSX-only?
1042 bool wantNativeToolbar, ownToolbarInstalled;
1044 // attempt to install the native toolbar
1045 wantNativeToolbar = ((GetWindowStyleFlag() & (wxTB_LEFT|wxTB_BOTTOM|wxTB_RIGHT)) == 0);
1046 MacInstallNativeToolbar( wantNativeToolbar );
1047 (void)MacTopLevelHasNativeToolbar( &ownToolbarInstalled );
1048 if (!ownToolbarInstalled)
1050 SetSize( maxWidth, maxHeight );
1051 InvalidateBestSize();
1055 SetSize( maxWidth, maxHeight );
1056 InvalidateBestSize();
1064 void wxToolBar::SetToolBitmapSize(const wxSize& size)
1066 m_defaultWidth = size.x + kwxMacToolBorder;
1067 m_defaultHeight = size.y + kwxMacToolBorder;
1069 #if wxOSX_USE_NATIVE_TOOLBAR
1070 if (m_macToolbar != NULL)
1072 int maxs = wxMax( size.x, size.y );
1073 NSToolbarSizeMode sizeSpec;
1075 sizeSpec = NSToolbarSizeModeRegular;
1076 else if ( maxs > 24 )
1077 sizeSpec = NSToolbarSizeModeDefault;
1079 sizeSpec = NSToolbarSizeModeSmall;
1081 [(NSToolbar*) m_macToolbar setSizeMode:sizeSpec ];
1086 // The button size is bigger than the bitmap size
1087 wxSize wxToolBar::GetToolSize() const
1089 return wxSize(m_defaultWidth + kwxMacToolBorder, m_defaultHeight + kwxMacToolBorder);
1092 void wxToolBar::SetRows(int nRows)
1094 // avoid resizing the frame uselessly
1095 if ( nRows != m_maxRows )
1099 void wxToolBar::MacSuperChangedPosition()
1101 wxWindow::MacSuperChangedPosition();
1103 #if wxOSX_USE_NATIVE_TOOLBAR
1104 if (! m_macUsesNativeToolbar )
1112 void wxToolBar::SetToolNormalBitmap( int id, const wxBitmap& bitmap )
1114 wxToolBarTool* tool = static_cast<wxToolBarTool*>(FindById(id));
1117 wxCHECK_RET( tool->IsButton(), wxT("Can only set bitmap on button tools."));
1119 tool->SetNormalBitmap(bitmap);
1121 // a side-effect of the UpdateToggleImage function is that it always changes the bitmap used on the button.
1122 tool->UpdateImages();
1126 void wxToolBar::SetToolDisabledBitmap( int id, const wxBitmap& bitmap )
1128 wxToolBarTool* tool = static_cast<wxToolBarTool*>(FindById(id));
1131 wxCHECK_RET( tool->IsButton(), wxT("Can only set bitmap on button tools."));
1133 tool->SetDisabledBitmap(bitmap);
1135 // TODO: what to do for this one?
1139 wxToolBarToolBase *wxToolBar::FindToolForPosition(wxCoord x, wxCoord y) const
1141 wxToolBarTool *tool;
1142 wxToolBarToolsList::compatibility_iterator node = m_tools.GetFirst();
1145 tool = (wxToolBarTool *)node->GetData();
1148 wxRect2DInt r( tool->GetPosition(), tool->GetSize() );
1149 if ( r.Contains( wxPoint( x, y ) ) )
1153 node = node->GetNext();
1159 wxString wxToolBar::MacGetToolTipString( wxPoint &pt )
1161 wxToolBarToolBase *tool = FindToolForPosition( pt.x, pt.y );
1163 return tool->GetShortHelp();
1165 return wxEmptyString;
1168 void wxToolBar::DoEnableTool(wxToolBarToolBase *t, bool enable)
1171 ((wxToolBarTool*)t)->DoEnable( enable );
1174 void wxToolBar::DoToggleTool(wxToolBarToolBase *t, bool toggle)
1176 wxToolBarTool *tool = (wxToolBarTool *)t;
1177 if ( ( tool != NULL ) && tool->IsButton() )
1178 tool->UpdateToggleImage( toggle );
1181 bool wxToolBar::DoInsertTool(size_t WXUNUSED(pos), wxToolBarToolBase *toolBase)
1183 wxToolBarTool *tool = static_cast< wxToolBarTool*>(toolBase );
1187 wxSize toolSize = GetToolSize();
1188 WXWidget controlHandle = NULL;
1189 NSRect toolrect = NSMakeRect(0, 0, toolSize.x, toolSize.y );
1191 #if wxOSX_USE_NATIVE_TOOLBAR
1192 wxString label = tool->GetLabel();
1193 if (m_macToolbar && !label.empty() )
1195 // strip mnemonics from the label for compatibility
1196 // with the usual labels in wxStaticText sense
1197 label = wxStripMenuCodes(label);
1199 #endif // wxOSX_USE_NATIVE_TOOLBAR
1201 switch (tool->GetStyle())
1203 case wxTOOL_STYLE_SEPARATOR:
1205 wxASSERT( tool->GetControlHandle() == NULL );
1208 if ( GetWindowStyleFlag() & (wxTB_LEFT|wxTB_RIGHT) )
1209 toolrect.size.height = toolSize.y;
1211 toolrect.size.width = toolSize.x;
1213 // in flat style we need a visual separator
1214 #if wxOSX_USE_NATIVE_TOOLBAR
1215 if (m_macToolbar != NULL)
1217 NSToolbarItem* item = [[NSToolbarItem alloc] initWithItemIdentifier:NSToolbarSeparatorItemIdentifier];
1218 tool->SetToolbarItemRef( item );
1220 #endif // wxOSX_USE_NATIVE_TOOLBAR
1222 NSBox* box = [[NSBox alloc] initWithFrame:toolrect];
1223 [box setBoxType:NSBoxSeparator];
1224 controlHandle = box;
1225 tool->SetControlHandle( controlHandle );
1229 case wxTOOL_STYLE_BUTTON:
1231 wxASSERT( tool->GetControlHandle() == NULL );
1233 wxNSToolBarButton* v = [[wxNSToolBarButton alloc] initWithFrame:toolrect];
1235 [v setBezelStyle:NSRegularSquareBezelStyle];
1237 [v setButtonType: ( tool->CanBeToggled() ? NSOnOffButton : NSMomentaryPushInButton )];
1238 [v setImplementation:tool];
1242 #if wxOSX_USE_NATIVE_TOOLBAR
1243 if (m_macToolbar != NULL)
1245 wxString identifier = wxString::Format(wxT("%ld"), (long) tool);
1246 wxCFStringRef cfidentifier( identifier, wxFont::GetDefaultEncoding() );
1247 wxNSToolbarItem* item = [[wxNSToolbarItem alloc] initWithItemIdentifier:cfidentifier.AsNSString() ];
1248 [item setImplementation:tool];
1249 tool->SetToolbarItemRef( item );
1252 #endif // wxOSX_USE_NATIVE_TOOLBAR
1253 tool->SetControlHandle( controlHandle );
1254 tool->UpdateImages();
1255 tool->UpdateLabel();
1257 SetBevelButtonTextPlacement( m_controlHandle, kControlBevelButtonPlaceBelowGraphic );
1258 SetControlTitleWithCFString( m_controlHandle , wxCFStringRef( label, wxFont::GetDefaultEncoding() );
1261 InstallControlEventHandler(
1262 (WXWidget) controlHandle, GetwxMacToolBarToolEventHandlerUPP(),
1263 GetEventTypeCount(eventList), eventList, tool, NULL );
1268 case wxTOOL_STYLE_CONTROL:
1270 #if wxOSX_USE_NATIVE_TOOLBAR
1271 if (m_macToolbar != NULL)
1273 WXWidget view = (WXWidget) tool->GetControl()->GetHandle() ;
1274 wxCHECK_MSG( view, false, wxT("control must be non-NULL") );
1276 wxString identifier = wxString::Format(wxT("%ld"), (long) tool);
1277 wxCFStringRef cfidentifier( identifier, wxFont::GetDefaultEncoding() );
1278 wxNSToolbarItem* item = [[wxNSToolbarItem alloc] initWithItemIdentifier:cfidentifier.AsNSString() ];
1279 [item setImplementation:tool];
1280 tool->SetToolbarItemRef( item );
1283 // right now there's nothing to do here
1285 tool->UpdateLabel();
1292 if ( controlHandle )
1294 WXWidget container = (WXWidget) GetHandle();
1295 wxASSERT_MSG( container != NULL, wxT("No valid Mac container control") );
1297 // SetControlVisibility( controlHandle, true, true );
1298 [container addSubview:controlHandle];
1301 // nothing special to do here - we relayout in Realize() later
1302 InvalidateBestSize();
1308 void wxToolBar::DoSetToggle(wxToolBarToolBase *WXUNUSED(tool), bool WXUNUSED(toggle))
1310 wxFAIL_MSG( wxT("not implemented") );
1313 bool wxToolBar::DoDeleteTool(size_t WXUNUSED(pos), wxToolBarToolBase *toolbase)
1315 wxToolBarTool* tool = static_cast< wxToolBarTool*>(toolbase );
1316 wxToolBarToolsList::compatibility_iterator node;
1317 for ( node = m_tools.GetFirst(); node; node = node->GetNext() )
1319 wxToolBarToolBase *tool2 = node->GetData();
1320 if ( tool2 == tool )
1322 // let node point to the next node in the list
1323 node = node->GetNext();
1329 wxSize sz = ((wxToolBarTool*)tool)->GetSize();
1331 #if wxOSX_USE_NATIVE_TOOLBAR
1332 CFIndex removeIndex = tool->GetIndex();
1335 #if wxOSX_USE_NATIVE_TOOLBAR
1336 if (m_macToolbar != NULL)
1338 if ( removeIndex != -1 && m_macToolbar )
1340 [(NSToolbar*) m_macToolbar removeItemAtIndex:removeIndex];
1341 tool->SetIndex( -1 );
1346 tool->ClearControl();
1348 // and finally reposition all the controls after this one
1350 for ( /* node -> first after deleted */; node; node = node->GetNext() )
1352 wxToolBarTool *tool2 = (wxToolBarTool*) node->GetData();
1353 wxPoint pt = tool2->GetPosition();
1355 if ( GetWindowStyleFlag() & (wxTB_LEFT|wxTB_RIGHT) )
1360 tool2->SetPosition( pt );
1362 #if wxOSX_USE_NATIVE_TOOLBAR
1363 if (m_macToolbar != NULL)
1365 if ( removeIndex != -1 && tool2->GetIndex() > removeIndex )
1366 tool2->SetIndex( tool2->GetIndex() - 1 );
1371 InvalidateBestSize();
1376 #include <Carbon/Carbon.h>
1378 void wxToolBar::OnPaint(wxPaintEvent& event)
1380 #if wxOSX_USE_NATIVE_TOOLBAR
1381 if ( m_macUsesNativeToolbar )
1393 bool drawMetalTheme = MacGetTopLevelWindow()->GetExtraStyle() & wxFRAME_EX_METAL;
1395 if ( !drawMetalTheme )
1397 HIThemePlacardDrawInfo info;
1398 memset( &info, 0, sizeof(info) );
1400 info.state = IsEnabled() ? kThemeStateActive : kThemeStateInactive;
1402 CGContextRef cgContext = (CGContextRef) MacGetCGContextRef();
1403 HIRect rect = CGRectMake( 0, 0, w, h );
1404 HIThemeDrawPlacard( &rect, &info, cgContext, kHIThemeOrientationNormal );
1408 // leave the background as it is (striped or metal)
1414 #endif // wxUSE_TOOLBAR