1 /////////////////////////////////////////////////////////////////////////////
2 // Name: src/common/tbarbase.cpp
3 // Purpose: wxToolBarBase implementation
4 // Author: Julian Smart
5 // Modified by: VZ at 11.12.99 (wxScrollableToolBar split off)
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"
29 #include "wx/toolbar.h"
32 #include "wx/control.h"
34 #include "wx/settings.h"
35 #if WXWIN_COMPATIBILITY_2_8
37 #endif // WXWIN_COMPATIBILITY_2_8
41 // ----------------------------------------------------------------------------
43 // ----------------------------------------------------------------------------
45 BEGIN_EVENT_TABLE(wxToolBarBase
, wxControl
)
48 #include "wx/listimpl.cpp"
50 WX_DEFINE_LIST(wxToolBarToolsList
)
52 // ============================================================================
54 // ============================================================================
56 // ----------------------------------------------------------------------------
58 // ----------------------------------------------------------------------------
60 IMPLEMENT_DYNAMIC_CLASS(wxToolBarToolBase
, wxObject
)
62 wxToolBarToolBase::~wxToolBarToolBase()
64 delete m_dropdownMenu
;
66 GetControl()->Destroy();
70 bool wxToolBarToolBase::Enable(bool enable
)
72 if ( m_enabled
== enable
)
80 bool wxToolBarToolBase::Toggle(bool toggle
)
82 wxASSERT_MSG( CanBeToggled(), wxT("can't toggle this tool") );
84 if ( m_toggled
== toggle
)
92 bool wxToolBarToolBase::SetToggle(bool toggle
)
94 wxItemKind kind
= toggle
? wxITEM_CHECK
: wxITEM_NORMAL
;
103 bool wxToolBarToolBase::SetShortHelp(const wxString
& help
)
105 if ( m_shortHelpString
== help
)
108 m_shortHelpString
= help
;
113 bool wxToolBarToolBase::SetLongHelp(const wxString
& help
)
115 if ( m_longHelpString
== help
)
118 m_longHelpString
= help
;
124 void wxToolBarToolBase::SetDropdownMenu(wxMenu
* menu
)
126 delete m_dropdownMenu
;
127 m_dropdownMenu
= menu
;
131 // ----------------------------------------------------------------------------
132 // wxToolBarBase adding/deleting items
133 // ----------------------------------------------------------------------------
135 wxToolBarBase::wxToolBarBase()
137 // the list owns the pointers
138 m_xMargin
= m_yMargin
= 0;
139 m_maxRows
= m_maxCols
= 0;
140 m_toolPacking
= m_toolSeparation
= 0;
142 m_defaultHeight
= 15;
145 void wxToolBarBase::FixupStyle()
147 if ( !HasFlag(wxTB_TOP
| wxTB_LEFT
| wxTB_RIGHT
| wxTB_BOTTOM
) )
149 // this is the default
150 m_windowStyle
|= wxTB_TOP
;
154 wxToolBarToolBase
*wxToolBarBase::DoAddTool(int id
,
155 const wxString
& label
,
156 const wxBitmap
& bitmap
,
157 const wxBitmap
& bmpDisabled
,
159 const wxString
& shortHelp
,
160 const wxString
& longHelp
,
161 wxObject
*clientData
,
162 wxCoord
WXUNUSED(xPos
),
163 wxCoord
WXUNUSED(yPos
))
165 InvalidateBestSize();
166 return InsertTool(GetToolsCount(), id
, label
, bitmap
, bmpDisabled
,
167 kind
, shortHelp
, longHelp
, clientData
);
170 wxToolBarToolBase
*wxToolBarBase::InsertTool(size_t pos
,
172 const wxString
& label
,
173 const wxBitmap
& bitmap
,
174 const wxBitmap
& bmpDisabled
,
176 const wxString
& shortHelp
,
177 const wxString
& longHelp
,
178 wxObject
*clientData
)
180 wxCHECK_MSG( pos
<= GetToolsCount(), NULL
,
181 wxT("invalid position in wxToolBar::InsertTool()") );
183 return DoInsertNewTool(pos
, CreateTool(id
, label
, bitmap
, bmpDisabled
, kind
,
184 clientData
, shortHelp
, longHelp
));
187 wxToolBarToolBase
*wxToolBarBase::AddTool(wxToolBarToolBase
*tool
)
189 return InsertTool(GetToolsCount(), tool
);
193 wxToolBarBase::InsertTool(size_t pos
, wxToolBarToolBase
*tool
)
195 wxCHECK_MSG( pos
<= GetToolsCount(), NULL
,
196 wxT("invalid position in wxToolBar::InsertTool()") );
198 if ( !tool
|| !DoInsertTool(pos
, tool
) )
203 m_tools
.Insert(pos
, tool
);
210 wxToolBarBase::AddControl(wxControl
*control
, const wxString
& label
)
212 return InsertControl(GetToolsCount(), control
, label
);
216 wxToolBarBase::InsertControl(size_t pos
,
218 const wxString
& label
)
220 wxCHECK_MSG( control
, NULL
,
221 wxT("toolbar: can't insert NULL control") );
223 wxCHECK_MSG( control
->GetParent() == this, NULL
,
224 wxT("control must have toolbar as parent") );
226 return DoInsertNewTool(pos
, CreateTool(control
, label
));
229 wxControl
*wxToolBarBase::FindControl( int id
)
231 for ( wxToolBarToolsList::compatibility_iterator node
= m_tools
.GetFirst();
233 node
= node
->GetNext() )
235 const wxToolBarToolBase
* const tool
= node
->GetData();
236 if ( tool
->IsControl() )
238 wxControl
* const control
= tool
->GetControl();
242 wxFAIL_MSG( wxT("NULL control in toolbar?") );
244 else if ( control
->GetId() == id
)
255 wxToolBarToolBase
*wxToolBarBase::AddSeparator()
257 return InsertSeparator(GetToolsCount());
260 wxToolBarToolBase
*wxToolBarBase::InsertSeparator(size_t pos
)
262 return DoInsertNewTool(pos
, CreateSeparator());
265 wxToolBarToolBase
*wxToolBarBase::AddStretchableSpace()
267 return InsertStretchableSpace(GetToolsCount());
270 wxToolBarToolBase
*wxToolBarBase::InsertStretchableSpace(size_t pos
)
272 wxToolBarToolBase
* const tool
= CreateSeparator();
275 // this is a hack but we know that all the current implementations
276 // don't really use the tool when it's created, they will do it
277 // InsertTool() at earliest and maybe even in Realize() much later
279 // so we can create the tool as a plain separator and mark it as being
280 // a stretchable space later
281 tool
->MakeStretchable();
284 return DoInsertNewTool(pos
, tool
);
287 wxToolBarToolBase
*wxToolBarBase::RemoveTool(int id
)
290 wxToolBarToolsList::compatibility_iterator node
;
291 for ( node
= m_tools
.GetFirst(); node
; node
= node
->GetNext() )
293 if ( node
->GetData()->GetId() == id
)
301 // don't give any error messages - sometimes we might call RemoveTool()
302 // without knowing whether the tool is or not in the toolbar
306 wxToolBarToolBase
*tool
= node
->GetData();
307 wxCHECK_MSG( tool
, NULL
, "NULL tool in the tools list?" );
309 if ( !DoDeleteTool(pos
, tool
) )
319 bool wxToolBarBase::DeleteToolByPos(size_t pos
)
321 wxCHECK_MSG( pos
< GetToolsCount(), false,
322 wxT("invalid position in wxToolBar::DeleteToolByPos()") );
324 wxToolBarToolsList::compatibility_iterator node
= m_tools
.Item(pos
);
326 if ( !DoDeleteTool(pos
, node
->GetData()) )
331 delete node
->GetData();
337 bool wxToolBarBase::DeleteTool(int id
)
340 wxToolBarToolsList::compatibility_iterator node
;
341 for ( node
= m_tools
.GetFirst(); node
; node
= node
->GetNext() )
343 if ( node
->GetData()->GetId() == id
)
349 if ( !node
|| !DoDeleteTool(pos
, node
->GetData()) )
354 delete node
->GetData();
360 wxToolBarToolBase
*wxToolBarBase::FindById(int id
) const
362 wxToolBarToolBase
*tool
= NULL
;
364 for ( wxToolBarToolsList::compatibility_iterator node
= m_tools
.GetFirst();
366 node
= node
->GetNext() )
368 tool
= node
->GetData();
369 if ( tool
->GetId() == id
)
381 void wxToolBarBase::UnToggleRadioGroup(wxToolBarToolBase
*tool
)
383 wxCHECK_RET( tool
, wxT("NULL tool in wxToolBarTool::UnToggleRadioGroup") );
385 if ( !tool
->IsButton() || tool
->GetKind() != wxITEM_RADIO
)
388 wxToolBarToolsList::compatibility_iterator node
= m_tools
.Find(tool
);
389 wxCHECK_RET( node
, wxT("invalid tool in wxToolBarTool::UnToggleRadioGroup") );
391 wxToolBarToolsList::compatibility_iterator nodeNext
= node
->GetNext();
394 wxToolBarToolBase
*toolNext
= nodeNext
->GetData();
396 if ( !toolNext
->IsButton() || toolNext
->GetKind() != wxITEM_RADIO
)
399 if ( toolNext
->Toggle(false) )
401 DoToggleTool(toolNext
, false);
404 nodeNext
= nodeNext
->GetNext();
407 wxToolBarToolsList::compatibility_iterator nodePrev
= node
->GetPrevious();
410 wxToolBarToolBase
*toolNext
= nodePrev
->GetData();
412 if ( !toolNext
->IsButton() || toolNext
->GetKind() != wxITEM_RADIO
)
415 if ( toolNext
->Toggle(false) )
417 DoToggleTool(toolNext
, false);
420 nodePrev
= nodePrev
->GetPrevious();
424 void wxToolBarBase::ClearTools()
426 while ( GetToolsCount() )
432 void wxToolBarBase::AdjustToolBitmapSize()
434 const wxSize
sizeOrig(m_defaultWidth
, m_defaultHeight
);
436 wxSize
sizeActual(sizeOrig
);
438 for ( wxToolBarToolsList::const_iterator i
= m_tools
.begin();
442 const wxBitmap
& bmp
= (*i
)->GetNormalBitmap();
444 sizeActual
.IncTo(bmp
.GetSize());
447 if ( sizeActual
!= sizeOrig
)
448 SetToolBitmapSize(sizeActual
);
451 bool wxToolBarBase::Realize()
453 // check if we have anything to do
454 if ( m_tools
.empty() )
457 // make sure tool size is large enough for all bitmaps to fit in
458 AdjustToolBitmapSize();
463 wxToolBarBase::~wxToolBarBase()
465 WX_CLEAR_LIST(wxToolBarToolsList
, m_tools
);
467 // notify the frame that it doesn't have a tool bar any longer to avoid
469 wxFrame
*frame
= wxDynamicCast(GetParent(), wxFrame
);
470 if ( frame
&& frame
->GetToolBar() == this )
472 frame
->SetToolBar(NULL
);
476 // ----------------------------------------------------------------------------
477 // wxToolBarBase tools state
478 // ----------------------------------------------------------------------------
480 void wxToolBarBase::EnableTool(int id
, bool enable
)
482 wxToolBarToolBase
*tool
= FindById(id
);
485 if ( tool
->Enable(enable
) )
487 DoEnableTool(tool
, enable
);
492 void wxToolBarBase::ToggleTool(int id
, bool toggle
)
494 wxToolBarToolBase
*tool
= FindById(id
);
495 if ( tool
&& tool
->CanBeToggled() )
497 if ( tool
->Toggle(toggle
) )
499 UnToggleRadioGroup(tool
);
500 DoToggleTool(tool
, toggle
);
505 void wxToolBarBase::SetToggle(int id
, bool toggle
)
507 wxToolBarToolBase
*tool
= FindById(id
);
510 if ( tool
->SetToggle(toggle
) )
512 DoSetToggle(tool
, toggle
);
517 void wxToolBarBase::SetToolShortHelp(int id
, const wxString
& help
)
519 wxToolBarToolBase
*tool
= FindById(id
);
522 (void)tool
->SetShortHelp(help
);
526 void wxToolBarBase::SetToolLongHelp(int id
, const wxString
& help
)
528 wxToolBarToolBase
*tool
= FindById(id
);
531 (void)tool
->SetLongHelp(help
);
535 wxObject
*wxToolBarBase::GetToolClientData(int id
) const
537 wxToolBarToolBase
*tool
= FindById(id
);
539 return tool
? tool
->GetClientData() : NULL
;
542 void wxToolBarBase::SetToolClientData(int id
, wxObject
*clientData
)
544 wxToolBarToolBase
*tool
= FindById(id
);
546 wxCHECK_RET( tool
, wxT("no such tool in wxToolBar::SetToolClientData") );
548 tool
->SetClientData(clientData
);
551 int wxToolBarBase::GetToolPos(int id
) const
554 wxToolBarToolsList::compatibility_iterator node
;
556 for ( node
= m_tools
.GetFirst(); node
; node
= node
->GetNext() )
558 if ( node
->GetData()->GetId() == id
)
567 bool wxToolBarBase::GetToolState(int id
) const
569 wxToolBarToolBase
*tool
= FindById(id
);
570 wxCHECK_MSG( tool
, false, wxT("no such tool") );
572 return tool
->IsToggled();
575 bool wxToolBarBase::GetToolEnabled(int id
) const
577 wxToolBarToolBase
*tool
= FindById(id
);
578 wxCHECK_MSG( tool
, false, wxT("no such tool") );
580 return tool
->IsEnabled();
583 wxString
wxToolBarBase::GetToolShortHelp(int id
) const
585 wxToolBarToolBase
*tool
= FindById(id
);
586 wxCHECK_MSG( tool
, wxEmptyString
, wxT("no such tool") );
588 return tool
->GetShortHelp();
591 wxString
wxToolBarBase::GetToolLongHelp(int id
) const
593 wxToolBarToolBase
*tool
= FindById(id
);
594 wxCHECK_MSG( tool
, wxEmptyString
, wxT("no such tool") );
596 return tool
->GetLongHelp();
599 // ----------------------------------------------------------------------------
600 // wxToolBarBase geometry
601 // ----------------------------------------------------------------------------
603 void wxToolBarBase::SetMargins(int x
, int y
)
609 void wxToolBarBase::SetRows(int WXUNUSED(nRows
))
614 bool wxToolBarBase::IsVertical() const
616 return HasFlag(wxTB_LEFT
| wxTB_RIGHT
);
620 // ----------------------------------------------------------------------------
622 // ----------------------------------------------------------------------------
624 // Only allow toggle if returns true
625 bool wxToolBarBase::OnLeftClick(int id
, bool toggleDown
)
627 wxCommandEvent
event(wxEVT_COMMAND_TOOL_CLICKED
, id
);
628 event
.SetEventObject(this);
630 // we use SetInt() to make wxCommandEvent::IsChecked() return toggleDown
631 event
.SetInt((int)toggleDown
);
633 // and SetExtraLong() for backwards compatibility
634 event
.SetExtraLong((long)toggleDown
);
636 // Send events to this toolbar instead (and thence up the window hierarchy)
637 HandleWindowEvent(event
);
642 // Call when right button down.
643 void wxToolBarBase::OnRightClick(int id
,
647 wxCommandEvent
event(wxEVT_COMMAND_TOOL_RCLICKED
, id
);
648 event
.SetEventObject(this);
651 GetEventHandler()->ProcessEvent(event
);
654 // Called when the mouse cursor enters a tool bitmap (no button pressed).
655 // Argument is wxID_ANY if mouse is exiting the toolbar.
656 // Note that for this event, the id of the window is used,
657 // and the integer parameter of wxCommandEvent is used to retrieve
659 void wxToolBarBase::OnMouseEnter(int id
)
661 wxCommandEvent
event(wxEVT_COMMAND_TOOL_ENTER
, GetId());
662 event
.SetEventObject(this);
665 wxFrame
*frame
= wxDynamicCast(GetParent(), wxFrame
);
669 if ( id
!= wxID_ANY
)
671 const wxToolBarToolBase
* const tool
= FindById(id
);
673 help
= tool
->GetLongHelp();
676 // call DoGiveHelp() even if help string is empty to avoid showing the
677 // help for the previously selected tool when another one is selected
678 frame
->DoGiveHelp(help
, id
!= wxID_ANY
);
681 (void)GetEventHandler()->ProcessEvent(event
);
684 // ----------------------------------------------------------------------------
686 // ----------------------------------------------------------------------------
688 // Do the toolbar button updates (check for EVT_UPDATE_UI handlers)
689 void wxToolBarBase::UpdateWindowUI(long flags
)
691 wxWindowBase::UpdateWindowUI(flags
);
693 // don't waste time updating state of tools in a hidden toolbar
697 // There is no sense in updating the toolbar UI
698 // if the parent window is about to get destroyed
699 wxWindow
*tlw
= wxGetTopLevelParent( this );
700 if (tlw
&& wxPendingDelete
.Member( tlw
))
703 wxEvtHandler
* evtHandler
= GetEventHandler() ;
705 for ( wxToolBarToolsList::compatibility_iterator node
= m_tools
.GetFirst();
707 node
= node
->GetNext() )
709 wxToolBarToolBase
* const tool
= node
->GetData();
710 if ( tool
->IsSeparator() )
713 int id
= tool
->GetId();
715 wxUpdateUIEvent
event(id
);
716 event
.SetEventObject(this);
718 if ( evtHandler
->ProcessEvent(event
) )
720 if ( event
.GetSetEnabled() )
721 EnableTool(id
, event
.GetEnabled());
722 if ( event
.GetSetChecked() )
723 ToggleTool(id
, event
.GetChecked());
725 if ( event
.GetSetText() )
732 bool wxToolBarBase::SetDropdownMenu(int toolid
, wxMenu
* menu
)
734 wxToolBarToolBase
* const tool
= FindById(toolid
);
735 wxCHECK_MSG( tool
, false, wxT("invalid tool id") );
737 wxCHECK_MSG( tool
->GetKind() == wxITEM_DROPDOWN
, false,
738 wxT("menu can be only associated with drop down tools") );
740 tool
->SetDropdownMenu(menu
);
745 #if WXWIN_COMPATIBILITY_2_8
747 bool wxCreateGreyedImage(const wxImage
& in
, wxImage
& out
)
750 out
= in
.ConvertToGreyscale();
753 #endif // wxUSE_IMAGE
757 #endif // WXWIN_COMPATIBILITY_2_8
759 #endif // wxUSE_TOOLBAR