1 /////////////////////////////////////////////////////////////////////////////
2 // Name: common/tbarbase.cpp
3 // Purpose: wxToolBarBase implementation
4 // Author: Julian Smart
5 // Modified by: VZ at 11.12.99 (wxScrollableToolBar splitted off)
8 // Copyright: (c) Julian Smart and Markus Holzem
9 // Licence: wxWindows license
10 /////////////////////////////////////////////////////////////////////////////
12 // ============================================================================
14 // ============================================================================
16 // ----------------------------------------------------------------------------
18 // ----------------------------------------------------------------------------
21 #pragma implementation "tbarbase.h"
24 // For compilers that support precompilation, includes "wx.h".
25 #include "wx/wxprec.h"
34 #include "wx/control.h"
39 #include "wx/settings.h"
41 #include "wx/tbarbase.h"
43 // ----------------------------------------------------------------------------
45 // ----------------------------------------------------------------------------
47 IMPLEMENT_CLASS(wxToolBarBase
, wxControl
)
49 BEGIN_EVENT_TABLE(wxToolBarBase
, wxControl
)
50 EVT_IDLE(wxToolBarBase::OnIdle
)
53 #include "wx/listimpl.cpp"
55 WX_DEFINE_LIST(wxToolBarToolsList
);
57 // ============================================================================
59 // ============================================================================
61 // ----------------------------------------------------------------------------
63 // ----------------------------------------------------------------------------
65 bool wxToolBarToolBase::Enable(bool enable
)
67 if ( m_enabled
== enable
)
75 bool wxToolBarToolBase::Toggle(bool toggle
)
77 wxASSERT_MSG( CanBeToggled(), _T("can't toggle this tool") );
79 if ( m_toggled
== toggle
)
87 bool wxToolBarToolBase::SetToggle(bool toggle
)
89 wxItemKind kind
= toggle
? wxITEM_CHECK
: wxITEM_NORMAL
;
98 bool wxToolBarToolBase::SetShortHelp(const wxString
& help
)
100 if ( m_shortHelpString
== help
)
103 m_shortHelpString
= help
;
108 bool wxToolBarToolBase::SetLongHelp(const wxString
& help
)
110 if ( m_longHelpString
== help
)
113 m_longHelpString
= help
;
118 wxToolBarToolBase::~wxToolBarToolBase()
122 // ----------------------------------------------------------------------------
123 // wxToolBarBase adding/deleting items
124 // ----------------------------------------------------------------------------
126 wxToolBarBase::wxToolBarBase()
128 // the list owns the pointers
129 m_tools
.DeleteContents(TRUE
);
131 m_xMargin
= m_yMargin
= 0;
133 m_maxRows
= m_maxCols
= 0;
136 wxToolBarToolBase
*wxToolBarBase::DoAddTool(int id
,
137 const wxString
& label
,
138 const wxBitmap
& bitmap
,
139 const wxBitmap
& bmpDisabled
,
141 const wxString
& shortHelp
,
142 const wxString
& longHelp
,
143 wxObject
*clientData
,
144 wxCoord
WXUNUSED(xPos
),
145 wxCoord
WXUNUSED(yPos
))
147 return InsertTool(GetToolsCount(), id
, label
, bitmap
, bmpDisabled
,
148 kind
, shortHelp
, longHelp
, clientData
);
151 wxToolBarToolBase
*wxToolBarBase::InsertTool(size_t pos
,
153 const wxString
& label
,
154 const wxBitmap
& bitmap
,
155 const wxBitmap
& bmpDisabled
,
157 const wxString
& shortHelp
,
158 const wxString
& longHelp
,
159 wxObject
*clientData
)
161 wxCHECK_MSG( pos
<= GetToolsCount(), (wxToolBarToolBase
*)NULL
,
162 _T("invalid position in wxToolBar::InsertTool()") );
164 wxToolBarToolBase
*tool
= CreateTool(id
, label
, bitmap
, bmpDisabled
, kind
,
165 clientData
, shortHelp
, longHelp
);
167 if ( !InsertTool(pos
, tool
) )
177 wxToolBarToolBase
*wxToolBarBase::AddTool(wxToolBarToolBase
*tool
)
179 return InsertTool(GetToolsCount(), tool
);
183 wxToolBarBase::InsertTool(size_t pos
, wxToolBarToolBase
*tool
)
185 wxCHECK_MSG( pos
<= GetToolsCount(), (wxToolBarToolBase
*)NULL
,
186 _T("invalid position in wxToolBar::InsertTool()") );
188 if ( !tool
|| !DoInsertTool(pos
, tool
) )
193 m_tools
.Insert(pos
, tool
);
198 wxToolBarToolBase
*wxToolBarBase::AddControl(wxControl
*control
)
200 return InsertControl(GetToolsCount(), control
);
203 wxToolBarToolBase
*wxToolBarBase::InsertControl(size_t pos
, wxControl
*control
)
205 wxCHECK_MSG( control
, (wxToolBarToolBase
*)NULL
,
206 _T("toolbar: can't insert NULL control") );
208 wxCHECK_MSG( control
->GetParent() == this, (wxToolBarToolBase
*)NULL
,
209 _T("control must have toolbar as parent") );
211 wxCHECK_MSG( pos
<= GetToolsCount(), (wxToolBarToolBase
*)NULL
,
212 _T("invalid position in wxToolBar::InsertControl()") );
214 wxToolBarToolBase
*tool
= CreateTool(control
);
216 if ( !InsertTool(pos
, tool
) )
226 wxControl
*wxToolBarBase::FindControl( int id
)
228 for ( wxToolBarToolsList::Node
* node
= m_tools
.GetFirst();
230 node
= node
->GetNext() )
232 wxControl
*control
= node
->GetData()->GetControl();
236 if (control
->GetId() == id
)
244 wxToolBarToolBase
*wxToolBarBase::AddSeparator()
246 return InsertSeparator(GetToolsCount());
249 wxToolBarToolBase
*wxToolBarBase::InsertSeparator(size_t pos
)
251 wxCHECK_MSG( pos
<= GetToolsCount(), (wxToolBarToolBase
*)NULL
,
252 _T("invalid position in wxToolBar::InsertSeparator()") );
254 wxToolBarToolBase
*tool
= CreateTool(wxID_SEPARATOR
,
256 wxNullBitmap
, wxNullBitmap
,
257 wxITEM_SEPARATOR
, (wxObject
*)NULL
,
258 wxEmptyString
, wxEmptyString
);
260 if ( !tool
|| !DoInsertTool(pos
, tool
) )
267 m_tools
.Insert(pos
, tool
);
272 wxToolBarToolBase
*wxToolBarBase::RemoveTool(int id
)
275 wxToolBarToolsList::Node
*node
;
276 for ( node
= m_tools
.GetFirst(); node
; node
= node
->GetNext() )
278 if ( node
->GetData()->GetId() == id
)
286 // don't give any error messages - sometimes we might call RemoveTool()
287 // without knowing whether the tool is or not in the toolbar
288 return (wxToolBarToolBase
*)NULL
;
291 wxToolBarToolBase
*tool
= node
->GetData();
292 if ( !DoDeleteTool(pos
, tool
) )
294 return (wxToolBarToolBase
*)NULL
;
297 // the node would delete the data, so set it to NULL to avoid this
300 m_tools
.DeleteNode(node
);
305 bool wxToolBarBase::DeleteToolByPos(size_t pos
)
307 wxCHECK_MSG( pos
< GetToolsCount(), FALSE
,
308 _T("invalid position in wxToolBar::DeleteToolByPos()") );
310 wxToolBarToolsList::Node
*node
= m_tools
.Item(pos
);
312 if ( !DoDeleteTool(pos
, node
->GetData()) )
317 m_tools
.DeleteNode(node
);
322 bool wxToolBarBase::DeleteTool(int id
)
325 wxToolBarToolsList::Node
*node
;
326 for ( node
= m_tools
.GetFirst(); node
; node
= node
->GetNext() )
328 if ( node
->GetData()->GetId() == id
)
334 if ( !node
|| !DoDeleteTool(pos
, node
->GetData()) )
339 m_tools
.DeleteNode(node
);
344 wxToolBarToolBase
*wxToolBarBase::FindById(int id
) const
346 wxToolBarToolBase
*tool
= (wxToolBarToolBase
*)NULL
;
348 for ( wxToolBarToolsList::Node
*node
= m_tools
.GetFirst();
350 node
= node
->GetNext() )
352 tool
= node
->GetData();
353 if ( tool
->GetId() == id
)
365 void wxToolBarBase::ClearTools()
370 bool wxToolBarBase::Realize()
375 wxToolBarBase::~wxToolBarBase()
379 // ----------------------------------------------------------------------------
380 // wxToolBarBase tools state
381 // ----------------------------------------------------------------------------
383 void wxToolBarBase::EnableTool(int id
, bool enable
)
385 wxToolBarToolBase
*tool
= FindById(id
);
388 if ( tool
->Enable(enable
) )
390 DoEnableTool(tool
, enable
);
395 void wxToolBarBase::ToggleTool(int id
, bool toggle
)
397 wxToolBarToolBase
*tool
= FindById(id
);
398 if ( tool
&& tool
->CanBeToggled() )
400 if ( tool
->Toggle(toggle
) )
402 DoToggleTool(tool
, toggle
);
407 void wxToolBarBase::SetToggle(int id
, bool toggle
)
409 wxToolBarToolBase
*tool
= FindById(id
);
412 if ( tool
->SetToggle(toggle
) )
414 DoSetToggle(tool
, toggle
);
419 void wxToolBarBase::SetToolShortHelp(int id
, const wxString
& help
)
421 wxToolBarToolBase
*tool
= FindById(id
);
424 (void)tool
->SetShortHelp(help
);
428 void wxToolBarBase::SetToolLongHelp(int id
, const wxString
& help
)
430 wxToolBarToolBase
*tool
= FindById(id
);
433 (void)tool
->SetLongHelp(help
);
437 wxObject
*wxToolBarBase::GetToolClientData(int id
) const
439 wxToolBarToolBase
*tool
= FindById(id
);
441 return tool
? tool
->GetClientData() : (wxObject
*)NULL
;
444 void wxToolBarBase::SetToolClientData(int id
, wxObject
*clientData
)
446 wxToolBarToolBase
*tool
= FindById(id
);
448 wxCHECK_RET( tool
, _T("no such tool in wxToolBar::SetToolClientData") );
450 tool
->SetClientData(clientData
);
453 bool wxToolBarBase::GetToolState(int id
) const
455 wxToolBarToolBase
*tool
= FindById(id
);
456 wxCHECK_MSG( tool
, FALSE
, _T("no such tool") );
458 return tool
->IsToggled();
461 bool wxToolBarBase::GetToolEnabled(int id
) const
463 wxToolBarToolBase
*tool
= FindById(id
);
464 wxCHECK_MSG( tool
, FALSE
, _T("no such tool") );
466 return tool
->IsEnabled();
469 wxString
wxToolBarBase::GetToolShortHelp(int id
) const
471 wxToolBarToolBase
*tool
= FindById(id
);
472 wxCHECK_MSG( tool
, _T(""), _T("no such tool") );
474 return tool
->GetShortHelp();
477 wxString
wxToolBarBase::GetToolLongHelp(int id
) const
479 wxToolBarToolBase
*tool
= FindById(id
);
480 wxCHECK_MSG( tool
, _T(""), _T("no such tool") );
482 return tool
->GetLongHelp();
485 // ----------------------------------------------------------------------------
486 // wxToolBarBase geometry
487 // ----------------------------------------------------------------------------
489 void wxToolBarBase::SetMargins(int x
, int y
)
495 void wxToolBarBase::SetRows(int WXUNUSED(nRows
))
500 // ----------------------------------------------------------------------------
502 // ----------------------------------------------------------------------------
504 // Only allow toggle if returns TRUE
505 bool wxToolBarBase::OnLeftClick(int id
, bool toggleDown
)
507 wxCommandEvent
event(wxEVT_COMMAND_TOOL_CLICKED
, id
);
508 event
.SetEventObject(this);
510 // we use SetInt() to make wxCommandEvent::IsChecked() return toggleDown
511 event
.SetInt((int)toggleDown
);
513 // and SetExtraLong() for backwards compatibility
514 event
.SetExtraLong((long)toggleDown
);
516 // Send events to this toolbar instead (and thence up the window hierarchy)
517 GetEventHandler()->ProcessEvent(event
);
522 // Call when right button down.
523 void wxToolBarBase::OnRightClick(int id
,
527 wxCommandEvent
event(wxEVT_COMMAND_TOOL_RCLICKED
, id
);
528 event
.SetEventObject(this);
531 GetEventHandler()->ProcessEvent(event
);
534 // Called when the mouse cursor enters a tool bitmap (no button pressed).
535 // Argument is -1 if mouse is exiting the toolbar.
536 // Note that for this event, the id of the window is used,
537 // and the integer parameter of wxCommandEvent is used to retrieve
539 void wxToolBarBase::OnMouseEnter(int id
)
541 wxCommandEvent
event(wxEVT_COMMAND_TOOL_ENTER
, GetId());
542 event
.SetEventObject(this);
545 wxFrame
*frame
= wxDynamicCast(GetParent(), wxFrame
);
548 wxToolBarToolBase
* tool
= id
== -1 ? (wxToolBarToolBase
*)0 : FindById(id
);
549 wxString help
= tool
? tool
->GetLongHelp() : wxString();
550 frame
->DoGiveHelp( help
, id
!= -1 );
553 (void)GetEventHandler()->ProcessEvent(event
);
556 // ----------------------------------------------------------------------------
558 // ----------------------------------------------------------------------------
560 void wxToolBarBase::OnIdle(wxIdleEvent
& event
)
567 // Do the toolbar button updates (check for EVT_UPDATE_UI handlers)
568 void wxToolBarBase::DoToolbarUpdates()
570 wxWindow
* parent
= this;
571 while (parent
->GetParent())
572 parent
= parent
->GetParent();
574 // This kind of #ifdef is a good way to annoy people. It breaks
575 // apps, but only on one platform and due to a hack in officially
576 // platform independent code. It took me hours to fix this. RR.
579 // wxWindow* focusWin = wxFindFocusDescendant(parent);
581 wxWindow
* focusWin
= (wxWindow
*) NULL
;
584 wxEvtHandler
* evtHandler
= focusWin
? focusWin
->GetEventHandler() : GetEventHandler() ;
586 for ( wxToolBarToolsList::Node
* node
= m_tools
.GetFirst();
588 node
= node
->GetNext() )
590 int id
= node
->GetData()->GetId();
592 wxUpdateUIEvent
event(id
);
593 event
.SetEventObject(this);
595 if ( evtHandler
->ProcessEvent(event
) )
597 if ( event
.GetSetEnabled() )
598 EnableTool(id
, event
.GetEnabled());
599 if ( event
.GetSetChecked() )
600 ToggleTool(id
, event
.GetChecked());
602 if ( event
.GetSetText() )
609 // Helper function, used by wxCreateGreyedImage
611 static void wxGreyOutImage( const wxImage
& src
,
613 const wxColour
& darkCol
,
614 const wxColour
& lightCol
,
615 const wxColour
& bgCol
)
617 // Second attempt, just making things monochrome
618 int width
= src
.GetWidth();
619 int height
= src
.GetHeight();
621 int redCur
, greenCur
, blueCur
;
622 for ( int x
= 0; x
< width
; x
++ )
624 for ( int y
= 1; y
< height
; y
++ )
626 redCur
= src
.GetRed(x
, y
);
627 greenCur
= src
.GetGreen(x
, y
);
628 blueCur
= src
.GetBlue(x
, y
);
630 // Change light things to the background colour
631 if ( redCur
>= (lightCol
.Red() - 50) && greenCur
>= (lightCol
.Green() - 50) && blueCur
>= (lightCol
.Blue() - 50) )
633 dest
.SetRGB(x
,y
, bgCol
.Red(), bgCol
.Green(), bgCol
.Blue());
635 else if ( redCur
== bgCol
.Red() && greenCur
== bgCol
.Green() && blueCur
== bgCol
.Blue() )
637 // Leave the background colour as-is
638 // dest.SetRGB(x,y, bgCol.Red(), bgCol.Green(), bgCol.Blue());
640 else // if ( redCur <= darkCol.Red() && greenCur <= darkCol.Green() && blueCur <= darkCol.Blue() )
642 // Change dark things to really dark
643 dest
.SetRGB(x
,y
, darkCol
.Red(), darkCol
.Green(), darkCol
.Blue());
650 * Make a greyed-out image suitable for disabled buttons.
651 * This code is adapted from wxNewBitmapButton in FL.
654 bool wxCreateGreyedImage(const wxImage
& in
, wxImage
& out
)
658 // assuming the pixels along the edges are of the background color
659 wxColour
bgCol(in
.GetRed(0, 0), in
.GetGreen(0, 0), in
.GetBlue(0, 0));
661 wxColour darkCol
= wxSystemSettings::GetColour(wxSYS_COLOUR_3DSHADOW
) ;
662 wxColour lightCol
= wxSystemSettings::GetColour(wxSYS_COLOUR_3DHIGHLIGHT
) ;
664 wxGreyOutImage(in
, out
, darkCol
, lightCol
, bgCol
);
669 #endif // wxUSE_TOOLBAR