]> git.saurik.com Git - wxWidgets.git/blame_incremental - src/aui/auibar.cpp
Do not use GTKIsUsingGlobalMenu when compiled against gtk1.x
[wxWidgets.git] / src / aui / auibar.cpp
... / ...
CommitLineData
1///////////////////////////////////////////////////////////////////////////////
2
3// Name: src/aui/auibar.cpp
4// Purpose: wxaui: wx advanced user interface - docking window manager
5// Author: Benjamin I. Williams
6// Modified by:
7// Created: 2005-05-17
8// RCS-ID: $Id$
9// Copyright: (C) Copyright 2005-2006, Kirix Corporation, All Rights Reserved
10// Licence: wxWindows Library Licence, Version 3.1
11///////////////////////////////////////////////////////////////////////////////
12
13// ============================================================================
14// declarations
15// ============================================================================
16
17// ----------------------------------------------------------------------------
18// headers
19// ----------------------------------------------------------------------------
20
21#include "wx/wxprec.h"
22
23#ifdef __BORLANDC__
24 #pragma hdrstop
25#endif
26
27#if wxUSE_AUI
28
29#include "wx/statline.h"
30#include "wx/dcbuffer.h"
31#include "wx/sizer.h"
32#include "wx/image.h"
33#include "wx/settings.h"
34#include "wx/menu.h"
35
36#include "wx/aui/auibar.h"
37#include "wx/aui/framemanager.h"
38
39#ifdef __WXMAC__
40#include "wx/osx/private.h"
41#endif
42
43#include "wx/arrimpl.cpp"
44WX_DEFINE_OBJARRAY(wxAuiToolBarItemArray)
45
46
47wxDEFINE_EVENT( wxEVT_COMMAND_AUITOOLBAR_TOOL_DROPDOWN, wxAuiToolBarEvent );
48wxDEFINE_EVENT( wxEVT_COMMAND_AUITOOLBAR_OVERFLOW_CLICK, wxAuiToolBarEvent );
49wxDEFINE_EVENT( wxEVT_COMMAND_AUITOOLBAR_RIGHT_CLICK, wxAuiToolBarEvent );
50wxDEFINE_EVENT( wxEVT_COMMAND_AUITOOLBAR_MIDDLE_CLICK, wxAuiToolBarEvent );
51wxDEFINE_EVENT( wxEVT_COMMAND_AUITOOLBAR_BEGIN_DRAG, wxAuiToolBarEvent );
52
53
54IMPLEMENT_CLASS(wxAuiToolBar, wxControl)
55IMPLEMENT_DYNAMIC_CLASS(wxAuiToolBarEvent, wxEvent)
56
57
58// missing wxITEM_* items
59enum
60{
61 wxITEM_CONTROL = wxITEM_MAX,
62 wxITEM_LABEL,
63 wxITEM_SPACER
64};
65
66const int BUTTON_DROPDOWN_WIDTH = 10;
67
68
69wxBitmap wxAuiBitmapFromBits(const unsigned char bits[], int w, int h,
70 const wxColour& color);
71
72static wxColor GetBaseColor()
73{
74
75#if defined( __WXMAC__ ) && wxOSX_USE_COCOA_OR_CARBON
76 wxColor baseColour = wxColour( wxMacCreateCGColorFromHITheme(kThemeBrushToolbarBackground));
77#else
78 wxColor baseColour = wxSystemSettings::GetColour(wxSYS_COLOUR_3DFACE);
79#endif
80
81 // the baseColour is too pale to use as our base colour,
82 // so darken it a bit --
83 if ((255-baseColour.Red()) +
84 (255-baseColour.Green()) +
85 (255-baseColour.Blue()) < 60)
86 {
87 baseColour = baseColour.ChangeLightness(92);
88 }
89
90 return baseColour;
91}
92
93
94
95class ToolbarCommandCapture : public wxEvtHandler
96{
97public:
98
99 ToolbarCommandCapture() { m_lastId = 0; }
100 int GetCommandId() const { return m_lastId; }
101
102 bool ProcessEvent(wxEvent& evt)
103 {
104 if (evt.GetEventType() == wxEVT_COMMAND_MENU_SELECTED)
105 {
106 m_lastId = evt.GetId();
107 return true;
108 }
109
110 if (GetNextHandler())
111 return GetNextHandler()->ProcessEvent(evt);
112
113 return false;
114 }
115
116private:
117 int m_lastId;
118};
119
120
121
122static const unsigned char
123 DISABLED_TEXT_GREY_HUE = wxColour::AlphaBlend(0, 255, 0.4);
124const wxColour DISABLED_TEXT_COLOR(DISABLED_TEXT_GREY_HUE,
125 DISABLED_TEXT_GREY_HUE,
126 DISABLED_TEXT_GREY_HUE);
127
128wxAuiDefaultToolBarArt::wxAuiDefaultToolBarArt()
129{
130 m_baseColour = GetBaseColor();
131
132 m_flags = 0;
133 m_textOrientation = wxAUI_TBTOOL_TEXT_BOTTOM;
134 m_highlightColour = wxSystemSettings::GetColour(wxSYS_COLOUR_HIGHLIGHT);
135
136 m_separatorSize = 7;
137 m_gripperSize = 7;
138 m_overflowSize = 16;
139
140 wxColor darker1Colour = m_baseColour.ChangeLightness(85);
141 wxColor darker2Colour = m_baseColour.ChangeLightness(75);
142 wxColor darker3Colour = m_baseColour.ChangeLightness(60);
143 wxColor darker4Colour = m_baseColour.ChangeLightness(50);
144 wxColor darker5Colour = m_baseColour.ChangeLightness(40);
145
146 m_gripperPen1 = wxPen(darker5Colour);
147 m_gripperPen2 = wxPen(darker3Colour);
148 m_gripperPen3 = *wxWHITE_PEN;
149
150 static const unsigned char buttonDropdownBits[] = { 0xe0, 0xf1, 0xfb };
151 static const unsigned char overflowBits[] = { 0x80, 0xff, 0x80, 0xc1, 0xe3, 0xf7 };
152
153 m_buttonDropDownBmp = wxAuiBitmapFromBits(buttonDropdownBits, 5, 3,
154 *wxBLACK);
155 m_disabledButtonDropDownBmp = wxAuiBitmapFromBits(
156 buttonDropdownBits, 5, 3,
157 wxColor(128,128,128));
158 m_overflowBmp = wxAuiBitmapFromBits(overflowBits, 7, 6, *wxBLACK);
159 m_disabledOverflowBmp = wxAuiBitmapFromBits(overflowBits, 7, 6, wxColor(128,128,128));
160
161 m_font = *wxNORMAL_FONT;
162}
163
164wxAuiDefaultToolBarArt::~wxAuiDefaultToolBarArt()
165{
166 m_font = *wxNORMAL_FONT;
167}
168
169
170wxAuiToolBarArt* wxAuiDefaultToolBarArt::Clone()
171{
172 return static_cast<wxAuiToolBarArt*>(new wxAuiDefaultToolBarArt);
173}
174
175void wxAuiDefaultToolBarArt::SetFlags(unsigned int flags)
176{
177 m_flags = flags;
178}
179
180void wxAuiDefaultToolBarArt::SetFont(const wxFont& font)
181{
182 m_font = font;
183}
184
185void wxAuiDefaultToolBarArt::SetTextOrientation(int orientation)
186{
187 m_textOrientation = orientation;
188}
189
190unsigned int wxAuiDefaultToolBarArt::GetFlags()
191{
192 return m_flags;
193}
194
195wxFont wxAuiDefaultToolBarArt::GetFont()
196{
197 return m_font;
198}
199
200int wxAuiDefaultToolBarArt::GetTextOrientation()
201{
202 return m_textOrientation;
203}
204
205void wxAuiDefaultToolBarArt::DrawBackground(
206 wxDC& dc,
207 wxWindow* WXUNUSED(wnd),
208 const wxRect& _rect)
209{
210 wxRect rect = _rect;
211 rect.height++;
212 wxColour startColour = m_baseColour.ChangeLightness(150);
213 wxColour endColour = m_baseColour.ChangeLightness(90);
214 dc.GradientFillLinear(rect, startColour, endColour, wxSOUTH);
215}
216
217void wxAuiDefaultToolBarArt::DrawLabel(
218 wxDC& dc,
219 wxWindow* WXUNUSED(wnd),
220 const wxAuiToolBarItem& item,
221 const wxRect& rect)
222{
223 dc.SetFont(m_font);
224 dc.SetTextForeground(*wxBLACK);
225
226 // we only care about the text height here since the text
227 // will get cropped based on the width of the item
228 int textWidth = 0, textHeight = 0;
229 dc.GetTextExtent(wxT("ABCDHgj"), &textWidth, &textHeight);
230
231 // set the clipping region
232 wxRect clipRect = rect;
233 clipRect.width -= 1;
234 dc.SetClippingRegion(clipRect);
235
236 int textX, textY;
237 textX = rect.x + 1;
238 textY = rect.y + (rect.height-textHeight)/2;
239 dc.DrawText(item.GetLabel(), textX, textY);
240 dc.DestroyClippingRegion();
241}
242
243
244void wxAuiDefaultToolBarArt::DrawButton(
245 wxDC& dc,
246 wxWindow* WXUNUSED(wnd),
247 const wxAuiToolBarItem& item,
248 const wxRect& rect)
249{
250 int textWidth = 0, textHeight = 0;
251
252 if (m_flags & wxAUI_TB_TEXT)
253 {
254 dc.SetFont(m_font);
255
256 int tx, ty;
257
258 dc.GetTextExtent(wxT("ABCDHgj"), &tx, &textHeight);
259 textWidth = 0;
260 dc.GetTextExtent(item.GetLabel(), &textWidth, &ty);
261 }
262
263 int bmpX = 0, bmpY = 0;
264 int textX = 0, textY = 0;
265
266 if (m_textOrientation == wxAUI_TBTOOL_TEXT_BOTTOM)
267 {
268 bmpX = rect.x +
269 (rect.width/2) -
270 (item.GetBitmap().GetWidth()/2);
271
272 bmpY = rect.y +
273 ((rect.height-textHeight)/2) -
274 (item.GetBitmap().GetHeight()/2);
275
276 textX = rect.x + (rect.width/2) - (textWidth/2) + 1;
277 textY = rect.y + rect.height - textHeight - 1;
278 }
279 else if (m_textOrientation == wxAUI_TBTOOL_TEXT_RIGHT)
280 {
281 bmpX = rect.x + 3;
282
283 bmpY = rect.y +
284 (rect.height/2) -
285 (item.GetBitmap().GetHeight()/2);
286
287 textX = bmpX + 3 + item.GetBitmap().GetWidth();
288 textY = rect.y +
289 (rect.height/2) -
290 (textHeight/2);
291 }
292
293
294 if (!(item.GetState() & wxAUI_BUTTON_STATE_DISABLED))
295 {
296 if (item.GetState() & wxAUI_BUTTON_STATE_PRESSED)
297 {
298 dc.SetPen(wxPen(m_highlightColour));
299 dc.SetBrush(wxBrush(m_highlightColour.ChangeLightness(150)));
300 dc.DrawRectangle(rect);
301 }
302 else if ((item.GetState() & wxAUI_BUTTON_STATE_HOVER) || item.IsSticky())
303 {
304 dc.SetPen(wxPen(m_highlightColour));
305 dc.SetBrush(wxBrush(m_highlightColour.ChangeLightness(170)));
306
307 // draw an even lighter background for checked item hovers (since
308 // the hover background is the same color as the check background)
309 if (item.GetState() & wxAUI_BUTTON_STATE_CHECKED)
310 dc.SetBrush(wxBrush(m_highlightColour.ChangeLightness(180)));
311
312 dc.DrawRectangle(rect);
313 }
314 else if (item.GetState() & wxAUI_BUTTON_STATE_CHECKED)
315 {
316 // it's important to put this code in an else statment after the
317 // hover, otherwise hovers won't draw properly for checked items
318 dc.SetPen(wxPen(m_highlightColour));
319 dc.SetBrush(wxBrush(m_highlightColour.ChangeLightness(170)));
320 dc.DrawRectangle(rect);
321 }
322 }
323
324 wxBitmap bmp;
325 if (item.GetState() & wxAUI_BUTTON_STATE_DISABLED)
326 bmp = item.GetDisabledBitmap();
327 else
328 bmp = item.GetBitmap();
329
330 if ( bmp.IsOk() )
331 dc.DrawBitmap(bmp, bmpX, bmpY, true);
332
333 // set the item's text color based on if it is disabled
334 dc.SetTextForeground(*wxBLACK);
335 if (item.GetState() & wxAUI_BUTTON_STATE_DISABLED)
336 dc.SetTextForeground(DISABLED_TEXT_COLOR);
337
338 if ( (m_flags & wxAUI_TB_TEXT) && !item.GetLabel().empty() )
339 {
340 dc.DrawText(item.GetLabel(), textX, textY);
341 }
342}
343
344
345void wxAuiDefaultToolBarArt::DrawDropDownButton(
346 wxDC& dc,
347 wxWindow* WXUNUSED(wnd),
348 const wxAuiToolBarItem& item,
349 const wxRect& rect)
350{
351 int textWidth = 0, textHeight = 0, textX = 0, textY = 0;
352 int bmpX = 0, bmpY = 0, dropBmpX = 0, dropBmpY = 0;
353
354 wxRect buttonRect = wxRect(rect.x,
355 rect.y,
356 rect.width-BUTTON_DROPDOWN_WIDTH,
357 rect.height);
358 wxRect dropDownRect = wxRect(rect.x+rect.width-BUTTON_DROPDOWN_WIDTH-1,
359 rect.y,
360 BUTTON_DROPDOWN_WIDTH+1,
361 rect.height);
362
363 if (m_flags & wxAUI_TB_TEXT)
364 {
365 dc.SetFont(m_font);
366
367 int tx, ty;
368 if (m_flags & wxAUI_TB_TEXT)
369 {
370 dc.GetTextExtent(wxT("ABCDHgj"), &tx, &textHeight);
371 textWidth = 0;
372 }
373
374 dc.GetTextExtent(item.GetLabel(), &textWidth, &ty);
375 }
376
377
378
379 dropBmpX = dropDownRect.x +
380 (dropDownRect.width/2) -
381 (m_buttonDropDownBmp.GetWidth()/2);
382 dropBmpY = dropDownRect.y +
383 (dropDownRect.height/2) -
384 (m_buttonDropDownBmp.GetHeight()/2);
385
386
387 if (m_textOrientation == wxAUI_TBTOOL_TEXT_BOTTOM)
388 {
389 bmpX = buttonRect.x +
390 (buttonRect.width/2) -
391 (item.GetBitmap().GetWidth()/2);
392 bmpY = buttonRect.y +
393 ((buttonRect.height-textHeight)/2) -
394 (item.GetBitmap().GetHeight()/2);
395
396 textX = rect.x + (rect.width/2) - (textWidth/2) + 1;
397 textY = rect.y + rect.height - textHeight - 1;
398 }
399 else if (m_textOrientation == wxAUI_TBTOOL_TEXT_RIGHT)
400 {
401 bmpX = rect.x + 3;
402
403 bmpY = rect.y +
404 (rect.height/2) -
405 (item.GetBitmap().GetHeight()/2);
406
407 textX = bmpX + 3 + item.GetBitmap().GetWidth();
408 textY = rect.y +
409 (rect.height/2) -
410 (textHeight/2);
411 }
412
413
414 if (item.GetState() & wxAUI_BUTTON_STATE_PRESSED)
415 {
416 dc.SetPen(wxPen(m_highlightColour));
417 dc.SetBrush(wxBrush(m_highlightColour.ChangeLightness(140)));
418 dc.DrawRectangle(buttonRect);
419
420 dc.SetBrush(wxBrush(m_highlightColour.ChangeLightness(170)));
421 dc.DrawRectangle(dropDownRect);
422 }
423 else if (item.GetState() & wxAUI_BUTTON_STATE_HOVER ||
424 item.IsSticky())
425 {
426 dc.SetPen(wxPen(m_highlightColour));
427 dc.SetBrush(wxBrush(m_highlightColour.ChangeLightness(170)));
428 dc.DrawRectangle(buttonRect);
429 dc.DrawRectangle(dropDownRect);
430 }
431 else if (item.GetState() & wxAUI_BUTTON_STATE_CHECKED)
432 {
433 // Notice that this branch must come after the hover one to ensure the
434 // correct appearance when the mouse hovers over a checked item.m_
435 dc.SetPen(wxPen(m_highlightColour));
436 dc.SetBrush(wxBrush(m_highlightColour.ChangeLightness(170)));
437 dc.DrawRectangle(buttonRect);
438 dc.DrawRectangle(dropDownRect);
439 }
440
441 wxBitmap bmp;
442 wxBitmap dropbmp;
443 if (item.GetState() & wxAUI_BUTTON_STATE_DISABLED)
444 {
445 bmp = item.GetDisabledBitmap();
446 dropbmp = m_disabledButtonDropDownBmp;
447 }
448 else
449 {
450 bmp = item.GetBitmap();
451 dropbmp = m_buttonDropDownBmp;
452 }
453
454 if (!bmp.IsOk())
455 return;
456
457 dc.DrawBitmap(bmp, bmpX, bmpY, true);
458 dc.DrawBitmap(dropbmp, dropBmpX, dropBmpY, true);
459
460 // set the item's text color based on if it is disabled
461 dc.SetTextForeground(*wxBLACK);
462 if (item.GetState() & wxAUI_BUTTON_STATE_DISABLED)
463 dc.SetTextForeground(DISABLED_TEXT_COLOR);
464
465 if ( (m_flags & wxAUI_TB_TEXT) && !item.GetLabel().empty() )
466 {
467 dc.DrawText(item.GetLabel(), textX, textY);
468 }
469}
470
471void wxAuiDefaultToolBarArt::DrawControlLabel(
472 wxDC& dc,
473 wxWindow* WXUNUSED(wnd),
474 const wxAuiToolBarItem& item,
475 const wxRect& rect)
476{
477 if (!(m_flags & wxAUI_TB_TEXT))
478 return;
479
480 if (m_textOrientation != wxAUI_TBTOOL_TEXT_BOTTOM)
481 return;
482
483 int textX = 0, textY = 0;
484 int textWidth = 0, textHeight = 0;
485
486 dc.SetFont(m_font);
487
488 int tx, ty;
489 if (m_flags & wxAUI_TB_TEXT)
490 {
491 dc.GetTextExtent(wxT("ABCDHgj"), &tx, &textHeight);
492 textWidth = 0;
493 }
494
495 dc.GetTextExtent(item.GetLabel(), &textWidth, &ty);
496
497 // don't draw the label if it is wider than the item width
498 if (textWidth > rect.width)
499 return;
500
501 // set the label's text color
502 dc.SetTextForeground(*wxBLACK);
503
504 textX = rect.x + (rect.width/2) - (textWidth/2) + 1;
505 textY = rect.y + rect.height - textHeight - 1;
506
507 if ( (m_flags & wxAUI_TB_TEXT) && !item.GetLabel().empty() )
508 {
509 dc.DrawText(item.GetLabel(), textX, textY);
510 }
511}
512
513wxSize wxAuiDefaultToolBarArt::GetLabelSize(
514 wxDC& dc,
515 wxWindow* WXUNUSED(wnd),
516 const wxAuiToolBarItem& item)
517{
518 dc.SetFont(m_font);
519
520 // get label's height
521 int width = 0, height = 0;
522 dc.GetTextExtent(wxT("ABCDHgj"), &width, &height);
523
524 // get item's width
525 width = item.GetMinSize().GetWidth();
526
527 if (width == -1)
528 {
529 // no width specified, measure the text ourselves
530 width = dc.GetTextExtent(item.GetLabel()).GetX();
531 }
532
533 return wxSize(width, height);
534}
535
536wxSize wxAuiDefaultToolBarArt::GetToolSize(
537 wxDC& dc,
538 wxWindow* WXUNUSED(wnd),
539 const wxAuiToolBarItem& item)
540{
541 if (!item.GetBitmap().IsOk() && !(m_flags & wxAUI_TB_TEXT))
542 return wxSize(16,16);
543
544 int width = item.GetBitmap().GetWidth();
545 int height = item.GetBitmap().GetHeight();
546
547 if (m_flags & wxAUI_TB_TEXT)
548 {
549 dc.SetFont(m_font);
550 int tx, ty;
551
552 if (m_textOrientation == wxAUI_TBTOOL_TEXT_BOTTOM)
553 {
554 dc.GetTextExtent(wxT("ABCDHgj"), &tx, &ty);
555 height += ty;
556
557 if ( !item.GetLabel().empty() )
558 {
559 dc.GetTextExtent(item.GetLabel(), &tx, &ty);
560 width = wxMax(width, tx+6);
561 }
562 }
563 else if ( m_textOrientation == wxAUI_TBTOOL_TEXT_RIGHT &&
564 !item.GetLabel().empty() )
565 {
566 width += 3; // space between left border and bitmap
567 width += 3; // space between bitmap and text
568
569 if ( !item.GetLabel().empty() )
570 {
571 dc.GetTextExtent(item.GetLabel(), &tx, &ty);
572 width += tx;
573 height = wxMax(height, ty);
574 }
575 }
576 }
577
578 // if the tool has a dropdown button, add it to the width
579 if (item.HasDropDown())
580 width += (BUTTON_DROPDOWN_WIDTH+4);
581
582 return wxSize(width, height);
583}
584
585void wxAuiDefaultToolBarArt::DrawSeparator(
586 wxDC& dc,
587 wxWindow* WXUNUSED(wnd),
588 const wxRect& _rect)
589{
590 bool horizontal = true;
591 if (m_flags & wxAUI_TB_VERTICAL)
592 horizontal = false;
593
594 wxRect rect = _rect;
595
596 if (horizontal)
597 {
598 rect.x += (rect.width/2);
599 rect.width = 1;
600 int new_height = (rect.height*3)/4;
601 rect.y += (rect.height/2) - (new_height/2);
602 rect.height = new_height;
603 }
604 else
605 {
606 rect.y += (rect.height/2);
607 rect.height = 1;
608 int new_width = (rect.width*3)/4;
609 rect.x += (rect.width/2) - (new_width/2);
610 rect.width = new_width;
611 }
612
613 wxColour startColour = m_baseColour.ChangeLightness(80);
614 wxColour endColour = m_baseColour.ChangeLightness(80);
615 dc.GradientFillLinear(rect, startColour, endColour, horizontal ? wxSOUTH : wxEAST);
616}
617
618void wxAuiDefaultToolBarArt::DrawGripper(wxDC& dc,
619 wxWindow* WXUNUSED(wnd),
620 const wxRect& rect)
621{
622 int i = 0;
623 while (1)
624 {
625 int x, y;
626
627 if (m_flags & wxAUI_TB_VERTICAL)
628 {
629 x = rect.x + (i*4) + 5;
630 y = rect.y + 3;
631 if (x > rect.GetWidth()-5)
632 break;
633 }
634 else
635 {
636 x = rect.x + 3;
637 y = rect.y + (i*4) + 5;
638 if (y > rect.GetHeight()-5)
639 break;
640 }
641
642 dc.SetPen(m_gripperPen1);
643 dc.DrawPoint(x, y);
644 dc.SetPen(m_gripperPen2);
645 dc.DrawPoint(x, y+1);
646 dc.DrawPoint(x+1, y);
647 dc.SetPen(m_gripperPen3);
648 dc.DrawPoint(x+2, y+1);
649 dc.DrawPoint(x+2, y+2);
650 dc.DrawPoint(x+1, y+2);
651
652 i++;
653 }
654
655}
656
657void wxAuiDefaultToolBarArt::DrawOverflowButton(wxDC& dc,
658 wxWindow* /*wnd*/,
659 const wxRect& rect,
660 int state)
661{
662 if (state & wxAUI_BUTTON_STATE_HOVER ||
663 state & wxAUI_BUTTON_STATE_PRESSED)
664 {
665 wxColor light_gray_bg = m_highlightColour.ChangeLightness(170);
666
667 if (m_flags & wxAUI_TB_VERTICAL)
668 {
669 dc.SetPen(wxPen(m_highlightColour));
670 dc.DrawLine(rect.x, rect.y, rect.x+rect.width, rect.y);
671 dc.SetPen(wxPen(light_gray_bg));
672 dc.SetBrush(wxBrush(light_gray_bg));
673 dc.DrawRectangle(rect.x, rect.y+1, rect.width, rect.height);
674 }
675 else
676 {
677 dc.SetPen(wxPen(m_highlightColour));
678 dc.DrawLine(rect.x, rect.y, rect.x, rect.y+rect.height);
679 dc.SetPen(wxPen(light_gray_bg));
680 dc.SetBrush(wxBrush(light_gray_bg));
681 dc.DrawRectangle(rect.x+1, rect.y, rect.width, rect.height);
682 }
683 }
684
685 int x = rect.x+1+(rect.width-m_overflowBmp.GetWidth())/2;
686 int y = rect.y+1+(rect.height-m_overflowBmp.GetHeight())/2;
687 dc.DrawBitmap(m_overflowBmp, x, y, true);
688}
689
690int wxAuiDefaultToolBarArt::GetElementSize(int element_id)
691{
692 switch (element_id)
693 {
694 case wxAUI_TBART_SEPARATOR_SIZE: return m_separatorSize;
695 case wxAUI_TBART_GRIPPER_SIZE: return m_gripperSize;
696 case wxAUI_TBART_OVERFLOW_SIZE: return m_overflowSize;
697 default: return 0;
698 }
699}
700
701void wxAuiDefaultToolBarArt::SetElementSize(int element_id, int size)
702{
703 switch (element_id)
704 {
705 case wxAUI_TBART_SEPARATOR_SIZE: m_separatorSize = size; break;
706 case wxAUI_TBART_GRIPPER_SIZE: m_gripperSize = size; break;
707 case wxAUI_TBART_OVERFLOW_SIZE: m_overflowSize = size; break;
708 }
709}
710
711int wxAuiDefaultToolBarArt::ShowDropDown(wxWindow* wnd,
712 const wxAuiToolBarItemArray& items)
713{
714 wxMenu menuPopup;
715
716 size_t items_added = 0;
717
718 size_t i, count = items.GetCount();
719 for (i = 0; i < count; ++i)
720 {
721 wxAuiToolBarItem& item = items.Item(i);
722
723 if (item.GetKind() == wxITEM_NORMAL)
724 {
725 wxString text = item.GetShortHelp();
726 if (text.empty())
727 text = item.GetLabel();
728
729 if (text.empty())
730 text = wxT(" ");
731
732 wxMenuItem* m = new wxMenuItem(&menuPopup, item.GetId(), text, item.GetShortHelp());
733
734 m->SetBitmap(item.GetBitmap());
735 menuPopup.Append(m);
736 items_added++;
737 }
738 else if (item.GetKind() == wxITEM_SEPARATOR)
739 {
740 if (items_added > 0)
741 menuPopup.AppendSeparator();
742 }
743 }
744
745 // find out where to put the popup menu of window items
746 wxPoint pt = ::wxGetMousePosition();
747 pt = wnd->ScreenToClient(pt);
748
749 // find out the screen coordinate at the bottom of the tab ctrl
750 wxRect cli_rect = wnd->GetClientRect();
751 pt.y = cli_rect.y + cli_rect.height;
752
753 ToolbarCommandCapture* cc = new ToolbarCommandCapture;
754 wnd->PushEventHandler(cc);
755 wnd->PopupMenu(&menuPopup, pt);
756 int command = cc->GetCommandId();
757 wnd->PopEventHandler(true);
758
759 return command;
760}
761
762
763
764
765static wxOrientation GetOrientation(long& style)
766{
767 switch (style & wxAUI_ORIENTATION_MASK)
768 {
769 case wxAUI_TB_HORIZONTAL:
770 return wxHORIZONTAL;
771 case wxAUI_TB_VERTICAL:
772 return wxVERTICAL;
773 default:
774 wxFAIL_MSG("toolbar cannot be locked in both horizontal and vertical orientations (maybe no lock was intended?)");
775 // fall through
776 case 0:
777 return wxBOTH;
778 }
779}
780
781BEGIN_EVENT_TABLE(wxAuiToolBar, wxControl)
782 EVT_SIZE(wxAuiToolBar::OnSize)
783 EVT_IDLE(wxAuiToolBar::OnIdle)
784 EVT_ERASE_BACKGROUND(wxAuiToolBar::OnEraseBackground)
785 EVT_PAINT(wxAuiToolBar::OnPaint)
786 EVT_LEFT_DOWN(wxAuiToolBar::OnLeftDown)
787 EVT_LEFT_DCLICK(wxAuiToolBar::OnLeftDown)
788 EVT_LEFT_UP(wxAuiToolBar::OnLeftUp)
789 EVT_RIGHT_DOWN(wxAuiToolBar::OnRightDown)
790 EVT_RIGHT_DCLICK(wxAuiToolBar::OnRightDown)
791 EVT_RIGHT_UP(wxAuiToolBar::OnRightUp)
792 EVT_MIDDLE_DOWN(wxAuiToolBar::OnMiddleDown)
793 EVT_MIDDLE_DCLICK(wxAuiToolBar::OnMiddleDown)
794 EVT_MIDDLE_UP(wxAuiToolBar::OnMiddleUp)
795 EVT_MOTION(wxAuiToolBar::OnMotion)
796 EVT_LEAVE_WINDOW(wxAuiToolBar::OnLeaveWindow)
797 EVT_MOUSE_CAPTURE_LOST(wxAuiToolBar::OnCaptureLost)
798 EVT_SET_CURSOR(wxAuiToolBar::OnSetCursor)
799END_EVENT_TABLE()
800
801void wxAuiToolBar::Init()
802{
803 m_sizer = new wxBoxSizer(wxHORIZONTAL);
804 m_buttonWidth = -1;
805 m_buttonHeight = -1;
806 m_sizerElementCount = 0;
807 m_actionPos = wxDefaultPosition;
808 m_actionItem = NULL;
809 m_tipItem = NULL;
810 m_art = new wxAuiDefaultToolBarArt;
811 m_toolPacking = 2;
812 m_toolBorderPadding = 3;
813 m_toolTextOrientation = wxAUI_TBTOOL_TEXT_BOTTOM;
814 m_gripperSizerItem = NULL;
815 m_overflowSizerItem = NULL;
816 m_dragging = false;
817 m_gripperVisible = false;
818 m_overflowVisible = false;
819 m_overflowState = 0;
820 m_orientation = wxHORIZONTAL;
821}
822
823bool wxAuiToolBar::Create(wxWindow* parent,
824 wxWindowID id,
825 const wxPoint& pos,
826 const wxSize& size,
827 long style)
828{
829 style = style|wxBORDER_NONE;
830
831 if (!wxControl::Create(parent, id, pos, size, style))
832 return false;
833
834 m_windowStyle = style;
835
836 m_gripperVisible = (style & wxAUI_TB_GRIPPER) ? true : false;
837 m_overflowVisible = (style & wxAUI_TB_OVERFLOW) ? true : false;
838
839 m_orientation = GetOrientation(style);
840 if (m_orientation == wxBOTH)
841 {
842 m_orientation = wxHORIZONTAL;
843 }
844
845 SetMargins(5, 5, 2, 2);
846 SetFont(*wxNORMAL_FONT);
847 SetArtFlags();
848 SetExtraStyle(wxWS_EX_PROCESS_IDLE);
849 if (style & wxAUI_TB_HORZ_LAYOUT)
850 SetToolTextOrientation(wxAUI_TBTOOL_TEXT_RIGHT);
851 SetBackgroundStyle(wxBG_STYLE_CUSTOM);
852
853 return true;
854}
855
856wxAuiToolBar::~wxAuiToolBar()
857{
858 delete m_art;
859 delete m_sizer;
860}
861
862void wxAuiToolBar::SetWindowStyleFlag(long style)
863{
864 GetOrientation(style); // assert if style is invalid
865 wxCHECK_RET(IsPaneValid(style),
866 "window settings and pane settings are incompatible");
867
868 wxControl::SetWindowStyleFlag(style);
869
870 m_windowStyle = style;
871
872 if (m_art)
873 {
874 SetArtFlags();
875 }
876
877 if (m_windowStyle & wxAUI_TB_GRIPPER)
878 m_gripperVisible = true;
879 else
880 m_gripperVisible = false;
881
882
883 if (m_windowStyle & wxAUI_TB_OVERFLOW)
884 m_overflowVisible = true;
885 else
886 m_overflowVisible = false;
887
888 if (style & wxAUI_TB_HORZ_LAYOUT)
889 SetToolTextOrientation(wxAUI_TBTOOL_TEXT_RIGHT);
890 else
891 SetToolTextOrientation(wxAUI_TBTOOL_TEXT_BOTTOM);
892}
893
894void wxAuiToolBar::SetArtProvider(wxAuiToolBarArt* art)
895{
896 delete m_art;
897
898 m_art = art;
899
900 if (m_art)
901 {
902 SetArtFlags();
903 m_art->SetTextOrientation(m_toolTextOrientation);
904 }
905}
906
907wxAuiToolBarArt* wxAuiToolBar::GetArtProvider() const
908{
909 return m_art;
910}
911
912
913
914
915wxAuiToolBarItem* wxAuiToolBar::AddTool(int tool_id,
916 const wxString& label,
917 const wxBitmap& bitmap,
918 const wxString& shortHelp_string,
919 wxItemKind kind)
920{
921 return AddTool(tool_id,
922 label,
923 bitmap,
924 wxNullBitmap,
925 kind,
926 shortHelp_string,
927 wxEmptyString,
928 NULL);
929}
930
931
932wxAuiToolBarItem* wxAuiToolBar::AddTool(int tool_id,
933 const wxString& label,
934 const wxBitmap& bitmap,
935 const wxBitmap& disabledBitmap,
936 wxItemKind kind,
937 const wxString& shortHelpString,
938 const wxString& longHelpString,
939 wxObject* WXUNUSED(client_data))
940{
941 wxAuiToolBarItem item;
942 item.m_window = NULL;
943 item.m_label = label;
944 item.m_bitmap = bitmap;
945 item.m_disabledBitmap = disabledBitmap;
946 item.m_shortHelp = shortHelpString;
947 item.m_longHelp = longHelpString;
948 item.m_active = true;
949 item.m_dropDown = false;
950 item.m_spacerPixels = 0;
951 item.m_toolId = tool_id;
952 item.m_state = 0;
953 item.m_proportion = 0;
954 item.m_kind = kind;
955 item.m_sizerItem = NULL;
956 item.m_minSize = wxDefaultSize;
957 item.m_userData = 0;
958 item.m_sticky = false;
959
960 if (item.m_toolId == wxID_ANY)
961 item.m_toolId = wxNewId();
962
963 if (!item.m_disabledBitmap.IsOk())
964 {
965 // no disabled bitmap specified, we need to make one
966 if (item.m_bitmap.IsOk())
967 {
968 item.m_disabledBitmap = item.m_bitmap.ConvertToDisabled();
969 }
970 }
971 m_items.Add(item);
972 return &m_items.Last();
973}
974
975wxAuiToolBarItem* wxAuiToolBar::AddControl(wxControl* control,
976 const wxString& label)
977{
978 wxAuiToolBarItem item;
979 item.m_window = (wxWindow*)control;
980 item.m_label = label;
981 item.m_bitmap = wxNullBitmap;
982 item.m_disabledBitmap = wxNullBitmap;
983 item.m_active = true;
984 item.m_dropDown = false;
985 item.m_spacerPixels = 0;
986 item.m_toolId = control->GetId();
987 item.m_state = 0;
988 item.m_proportion = 0;
989 item.m_kind = wxITEM_CONTROL;
990 item.m_sizerItem = NULL;
991 item.m_minSize = control->GetEffectiveMinSize();
992 item.m_userData = 0;
993 item.m_sticky = false;
994
995 m_items.Add(item);
996 return &m_items.Last();
997}
998
999wxAuiToolBarItem* wxAuiToolBar::AddLabel(int tool_id,
1000 const wxString& label,
1001 const int width)
1002{
1003 wxSize min_size = wxDefaultSize;
1004 if (width != -1)
1005 min_size.x = width;
1006
1007 wxAuiToolBarItem item;
1008 item.m_window = NULL;
1009 item.m_label = label;
1010 item.m_bitmap = wxNullBitmap;
1011 item.m_disabledBitmap = wxNullBitmap;
1012 item.m_active = true;
1013 item.m_dropDown = false;
1014 item.m_spacerPixels = 0;
1015 item.m_toolId = tool_id;
1016 item.m_state = 0;
1017 item.m_proportion = 0;
1018 item.m_kind = wxITEM_LABEL;
1019 item.m_sizerItem = NULL;
1020 item.m_minSize = min_size;
1021 item.m_userData = 0;
1022 item.m_sticky = false;
1023
1024 if (item.m_toolId == wxID_ANY)
1025 item.m_toolId = wxNewId();
1026
1027 m_items.Add(item);
1028 return &m_items.Last();
1029}
1030
1031wxAuiToolBarItem* wxAuiToolBar::AddSeparator()
1032{
1033 wxAuiToolBarItem item;
1034 item.m_window = NULL;
1035 item.m_label = wxEmptyString;
1036 item.m_bitmap = wxNullBitmap;
1037 item.m_disabledBitmap = wxNullBitmap;
1038 item.m_active = true;
1039 item.m_dropDown = false;
1040 item.m_toolId = -1;
1041 item.m_state = 0;
1042 item.m_proportion = 0;
1043 item.m_kind = wxITEM_SEPARATOR;
1044 item.m_sizerItem = NULL;
1045 item.m_minSize = wxDefaultSize;
1046 item.m_userData = 0;
1047 item.m_sticky = false;
1048
1049 m_items.Add(item);
1050 return &m_items.Last();
1051}
1052
1053wxAuiToolBarItem* wxAuiToolBar::AddSpacer(int pixels)
1054{
1055 wxAuiToolBarItem item;
1056 item.m_window = NULL;
1057 item.m_label = wxEmptyString;
1058 item.m_bitmap = wxNullBitmap;
1059 item.m_disabledBitmap = wxNullBitmap;
1060 item.m_active = true;
1061 item.m_dropDown = false;
1062 item.m_spacerPixels = pixels;
1063 item.m_toolId = -1;
1064 item.m_state = 0;
1065 item.m_proportion = 0;
1066 item.m_kind = wxITEM_SPACER;
1067 item.m_sizerItem = NULL;
1068 item.m_minSize = wxDefaultSize;
1069 item.m_userData = 0;
1070 item.m_sticky = false;
1071
1072 m_items.Add(item);
1073 return &m_items.Last();
1074}
1075
1076wxAuiToolBarItem* wxAuiToolBar::AddStretchSpacer(int proportion)
1077{
1078 wxAuiToolBarItem item;
1079 item.m_window = NULL;
1080 item.m_label = wxEmptyString;
1081 item.m_bitmap = wxNullBitmap;
1082 item.m_disabledBitmap = wxNullBitmap;
1083 item.m_active = true;
1084 item.m_dropDown = false;
1085 item.m_spacerPixels = 0;
1086 item.m_toolId = -1;
1087 item.m_state = 0;
1088 item.m_proportion = proportion;
1089 item.m_kind = wxITEM_SPACER;
1090 item.m_sizerItem = NULL;
1091 item.m_minSize = wxDefaultSize;
1092 item.m_userData = 0;
1093 item.m_sticky = false;
1094
1095 m_items.Add(item);
1096 return &m_items.Last();
1097}
1098
1099void wxAuiToolBar::Clear()
1100{
1101 m_items.Clear();
1102 m_sizerElementCount = 0;
1103}
1104
1105bool wxAuiToolBar::DeleteTool(int tool_id)
1106{
1107 int idx = GetToolIndex(tool_id);
1108 if (idx >= 0 && idx < (int)m_items.GetCount())
1109 {
1110 m_items.RemoveAt(idx);
1111 Realize();
1112 return true;
1113 }
1114
1115 return false;
1116}
1117
1118bool wxAuiToolBar::DeleteByIndex(int idx)
1119{
1120 if (idx >= 0 && idx < (int)m_items.GetCount())
1121 {
1122 m_items.RemoveAt(idx);
1123 Realize();
1124 return true;
1125 }
1126
1127 return false;
1128}
1129
1130
1131wxControl* wxAuiToolBar::FindControl(int id)
1132{
1133 wxWindow* wnd = FindWindow(id);
1134 return (wxControl*)wnd;
1135}
1136
1137wxAuiToolBarItem* wxAuiToolBar::FindTool(int tool_id) const
1138{
1139 size_t i, count;
1140 for (i = 0, count = m_items.GetCount(); i < count; ++i)
1141 {
1142 wxAuiToolBarItem& item = m_items.Item(i);
1143 if (item.m_toolId == tool_id)
1144 return &item;
1145 }
1146
1147 return NULL;
1148}
1149
1150wxAuiToolBarItem* wxAuiToolBar::FindToolByPosition(wxCoord x, wxCoord y) const
1151{
1152 size_t i, count;
1153 for (i = 0, count = m_items.GetCount(); i < count; ++i)
1154 {
1155 wxAuiToolBarItem& item = m_items.Item(i);
1156
1157 if (!item.m_sizerItem)
1158 continue;
1159
1160 wxRect rect = item.m_sizerItem->GetRect();
1161 if (rect.Contains(x,y))
1162 {
1163 // if the item doesn't fit on the toolbar, return NULL
1164 if (!GetToolFitsByIndex(i))
1165 return NULL;
1166
1167 return &item;
1168 }
1169 }
1170
1171 return NULL;
1172}
1173
1174wxAuiToolBarItem* wxAuiToolBar::FindToolByPositionWithPacking(wxCoord x, wxCoord y) const
1175{
1176 size_t i, count;
1177 for (i = 0, count = m_items.GetCount(); i < count; ++i)
1178 {
1179 wxAuiToolBarItem& item = m_items.Item(i);
1180
1181 if (!item.m_sizerItem)
1182 continue;
1183
1184 wxRect rect = item.m_sizerItem->GetRect();
1185
1186 // apply tool packing
1187 if (i+1 < count)
1188 rect.width += m_toolPacking;
1189
1190 if (rect.Contains(x,y))
1191 {
1192 // if the item doesn't fit on the toolbar, return NULL
1193 if (!GetToolFitsByIndex(i))
1194 return NULL;
1195
1196 return &item;
1197 }
1198 }
1199
1200 return NULL;
1201}
1202
1203wxAuiToolBarItem* wxAuiToolBar::FindToolByIndex(int idx) const
1204{
1205 if (idx < 0)
1206 return NULL;
1207
1208 if (idx >= (int)m_items.size())
1209 return NULL;
1210
1211 return &(m_items[idx]);
1212}
1213
1214void wxAuiToolBar::SetToolBitmapSize(const wxSize& WXUNUSED(size))
1215{
1216 // TODO: wxToolBar compatibility
1217}
1218
1219wxSize wxAuiToolBar::GetToolBitmapSize() const
1220{
1221 // TODO: wxToolBar compatibility
1222 return wxSize(16,15);
1223}
1224
1225void wxAuiToolBar::SetToolProportion(int tool_id, int proportion)
1226{
1227 wxAuiToolBarItem* item = FindTool(tool_id);
1228 if (!item)
1229 return;
1230
1231 item->m_proportion = proportion;
1232}
1233
1234int wxAuiToolBar::GetToolProportion(int tool_id) const
1235{
1236 wxAuiToolBarItem* item = FindTool(tool_id);
1237 if (!item)
1238 return 0;
1239
1240 return item->m_proportion;
1241}
1242
1243void wxAuiToolBar::SetToolSeparation(int separation)
1244{
1245 if (m_art)
1246 m_art->SetElementSize(wxAUI_TBART_SEPARATOR_SIZE, separation);
1247}
1248
1249int wxAuiToolBar::GetToolSeparation() const
1250{
1251 if (m_art)
1252 return m_art->GetElementSize(wxAUI_TBART_SEPARATOR_SIZE);
1253 else
1254 return 5;
1255}
1256
1257
1258void wxAuiToolBar::SetToolDropDown(int tool_id, bool dropdown)
1259{
1260 wxAuiToolBarItem* item = FindTool(tool_id);
1261 if (!item)
1262 return;
1263
1264 item->SetHasDropDown(dropdown);
1265}
1266
1267bool wxAuiToolBar::GetToolDropDown(int tool_id) const
1268{
1269 wxAuiToolBarItem* item = FindTool(tool_id);
1270 if (!item)
1271 return false;
1272
1273 return item->HasDropDown();
1274}
1275
1276void wxAuiToolBar::SetToolSticky(int tool_id, bool sticky)
1277{
1278 // ignore separators
1279 if (tool_id == -1)
1280 return;
1281
1282 wxAuiToolBarItem* item = FindTool(tool_id);
1283 if (!item)
1284 return;
1285
1286 if (item->m_sticky == sticky)
1287 return;
1288
1289 item->m_sticky = sticky;
1290
1291 Refresh(false);
1292 Update();
1293}
1294
1295bool wxAuiToolBar::GetToolSticky(int tool_id) const
1296{
1297 wxAuiToolBarItem* item = FindTool(tool_id);
1298 if (!item)
1299 return 0;
1300
1301 return item->m_sticky;
1302}
1303
1304
1305
1306
1307void wxAuiToolBar::SetToolBorderPadding(int padding)
1308{
1309 m_toolBorderPadding = padding;
1310}
1311
1312int wxAuiToolBar::GetToolBorderPadding() const
1313{
1314 return m_toolBorderPadding;
1315}
1316
1317void wxAuiToolBar::SetToolTextOrientation(int orientation)
1318{
1319 m_toolTextOrientation = orientation;
1320
1321 if (m_art)
1322 {
1323 m_art->SetTextOrientation(orientation);
1324 }
1325}
1326
1327int wxAuiToolBar::GetToolTextOrientation() const
1328{
1329 return m_toolTextOrientation;
1330}
1331
1332void wxAuiToolBar::SetToolPacking(int packing)
1333{
1334 m_toolPacking = packing;
1335}
1336
1337int wxAuiToolBar::GetToolPacking() const
1338{
1339 return m_toolPacking;
1340}
1341
1342
1343void wxAuiToolBar::SetOrientation(int orientation)
1344{
1345 wxCHECK_RET(orientation == wxHORIZONTAL ||
1346 orientation == wxVERTICAL,
1347 "invalid orientation value");
1348 if (orientation != m_orientation)
1349 {
1350 m_orientation = wxOrientation(orientation);
1351 SetArtFlags();
1352 }
1353}
1354
1355void wxAuiToolBar::SetMargins(int left, int right, int top, int bottom)
1356{
1357 if (left != -1)
1358 m_leftPadding = left;
1359 if (right != -1)
1360 m_rightPadding = right;
1361 if (top != -1)
1362 m_topPadding = top;
1363 if (bottom != -1)
1364 m_bottomPadding = bottom;
1365}
1366
1367bool wxAuiToolBar::GetGripperVisible() const
1368{
1369 return m_gripperVisible;
1370}
1371
1372void wxAuiToolBar::SetGripperVisible(bool visible)
1373{
1374 m_gripperVisible = visible;
1375 if (visible)
1376 m_windowStyle |= wxAUI_TB_GRIPPER;
1377 else
1378 m_windowStyle &= ~wxAUI_TB_GRIPPER;
1379 Realize();
1380 Refresh(false);
1381}
1382
1383
1384bool wxAuiToolBar::GetOverflowVisible() const
1385{
1386 return m_overflowVisible;
1387}
1388
1389void wxAuiToolBar::SetOverflowVisible(bool visible)
1390{
1391 m_overflowVisible = visible;
1392 if (visible)
1393 m_windowStyle |= wxAUI_TB_OVERFLOW;
1394 else
1395 m_windowStyle &= ~wxAUI_TB_OVERFLOW;
1396 Refresh(false);
1397}
1398
1399bool wxAuiToolBar::SetFont(const wxFont& font)
1400{
1401 bool res = wxWindow::SetFont(font);
1402
1403 if (m_art)
1404 {
1405 m_art->SetFont(font);
1406 }
1407
1408 return res;
1409}
1410
1411
1412void wxAuiToolBar::SetHoverItem(wxAuiToolBarItem* pitem)
1413{
1414 if (pitem && (pitem->m_state & wxAUI_BUTTON_STATE_DISABLED))
1415 pitem = NULL;
1416
1417 wxAuiToolBarItem* former_hover = NULL;
1418
1419 size_t i, count;
1420 for (i = 0, count = m_items.GetCount(); i < count; ++i)
1421 {
1422 wxAuiToolBarItem& item = m_items.Item(i);
1423 if (item.m_state & wxAUI_BUTTON_STATE_HOVER)
1424 former_hover = &item;
1425 item.m_state &= ~wxAUI_BUTTON_STATE_HOVER;
1426 }
1427
1428 if (pitem)
1429 {
1430 pitem->m_state |= wxAUI_BUTTON_STATE_HOVER;
1431 }
1432
1433 if (former_hover != pitem)
1434 {
1435 Refresh(false);
1436 Update();
1437 }
1438}
1439
1440void wxAuiToolBar::SetPressedItem(wxAuiToolBarItem* pitem)
1441{
1442 wxAuiToolBarItem* former_item = NULL;
1443
1444 size_t i, count;
1445 for (i = 0, count = m_items.GetCount(); i < count; ++i)
1446 {
1447 wxAuiToolBarItem& item = m_items.Item(i);
1448 if (item.m_state & wxAUI_BUTTON_STATE_PRESSED)
1449 former_item = &item;
1450 item.m_state &= ~wxAUI_BUTTON_STATE_PRESSED;
1451 }
1452
1453 if (pitem)
1454 {
1455 pitem->m_state &= ~wxAUI_BUTTON_STATE_HOVER;
1456 pitem->m_state |= wxAUI_BUTTON_STATE_PRESSED;
1457 }
1458
1459 if (former_item != pitem)
1460 {
1461 Refresh(false);
1462 Update();
1463 }
1464}
1465
1466void wxAuiToolBar::RefreshOverflowState()
1467{
1468 if (!m_overflowSizerItem)
1469 {
1470 m_overflowState = 0;
1471 return;
1472 }
1473
1474 int overflow_state = 0;
1475
1476 wxRect overflow_rect = GetOverflowRect();
1477
1478
1479 // find out the mouse's current position
1480 wxPoint pt = ::wxGetMousePosition();
1481 pt = this->ScreenToClient(pt);
1482
1483 // find out if the mouse cursor is inside the dropdown rectangle
1484 if (overflow_rect.Contains(pt.x, pt.y))
1485 {
1486 if (::wxGetMouseState().LeftIsDown())
1487 overflow_state = wxAUI_BUTTON_STATE_PRESSED;
1488 else
1489 overflow_state = wxAUI_BUTTON_STATE_HOVER;
1490 }
1491
1492 if (overflow_state != m_overflowState)
1493 {
1494 m_overflowState = overflow_state;
1495 Refresh(false);
1496 Update();
1497 }
1498
1499 m_overflowState = overflow_state;
1500}
1501
1502void wxAuiToolBar::ToggleTool(int tool_id, bool state)
1503{
1504 wxAuiToolBarItem* tool = FindTool(tool_id);
1505
1506 if (tool && (tool->m_kind == wxITEM_CHECK || tool->m_kind == wxITEM_RADIO))
1507 {
1508 if (tool->m_kind == wxITEM_RADIO)
1509 {
1510 int i, idx, count;
1511 idx = GetToolIndex(tool_id);
1512 count = (int)m_items.GetCount();
1513
1514 if (idx >= 0 && idx < count)
1515 {
1516 for (i = idx + 1; i < count; ++i)
1517 {
1518 if (m_items[i].m_kind != wxITEM_RADIO)
1519 break;
1520 m_items[i].m_state &= ~wxAUI_BUTTON_STATE_CHECKED;
1521 }
1522 for (i = idx - 1; i >= 0; i--)
1523 {
1524 if (m_items[i].m_kind != wxITEM_RADIO)
1525 break;
1526 m_items[i].m_state &= ~wxAUI_BUTTON_STATE_CHECKED;
1527 }
1528 }
1529
1530 tool->m_state |= wxAUI_BUTTON_STATE_CHECKED;
1531 }
1532 else if (tool->m_kind == wxITEM_CHECK)
1533 {
1534 if (state == true)
1535 tool->m_state |= wxAUI_BUTTON_STATE_CHECKED;
1536 else
1537 tool->m_state &= ~wxAUI_BUTTON_STATE_CHECKED;
1538 }
1539 }
1540}
1541
1542bool wxAuiToolBar::GetToolToggled(int tool_id) const
1543{
1544 wxAuiToolBarItem* tool = FindTool(tool_id);
1545
1546 if (tool)
1547 {
1548 if ( (tool->m_kind != wxITEM_CHECK) && (tool->m_kind != wxITEM_RADIO) )
1549 return false;
1550
1551 return (tool->m_state & wxAUI_BUTTON_STATE_CHECKED) ? true : false;
1552 }
1553
1554 return false;
1555}
1556
1557void wxAuiToolBar::EnableTool(int tool_id, bool state)
1558{
1559 wxAuiToolBarItem* tool = FindTool(tool_id);
1560
1561 if (tool)
1562 {
1563 if (state == true)
1564 tool->m_state &= ~wxAUI_BUTTON_STATE_DISABLED;
1565 else
1566 tool->m_state |= wxAUI_BUTTON_STATE_DISABLED;
1567 }
1568}
1569
1570bool wxAuiToolBar::GetToolEnabled(int tool_id) const
1571{
1572 wxAuiToolBarItem* tool = FindTool(tool_id);
1573
1574 if (tool)
1575 return (tool->m_state & wxAUI_BUTTON_STATE_DISABLED) ? false : true;
1576
1577 return false;
1578}
1579
1580wxString wxAuiToolBar::GetToolLabel(int tool_id) const
1581{
1582 wxAuiToolBarItem* tool = FindTool(tool_id);
1583 wxASSERT_MSG(tool, wxT("can't find tool in toolbar item array"));
1584 if (!tool)
1585 return wxEmptyString;
1586
1587 return tool->m_label;
1588}
1589
1590void wxAuiToolBar::SetToolLabel(int tool_id, const wxString& label)
1591{
1592 wxAuiToolBarItem* tool = FindTool(tool_id);
1593 if (tool)
1594 {
1595 tool->m_label = label;
1596 }
1597}
1598
1599wxBitmap wxAuiToolBar::GetToolBitmap(int tool_id) const
1600{
1601 wxAuiToolBarItem* tool = FindTool(tool_id);
1602 wxASSERT_MSG(tool, wxT("can't find tool in toolbar item array"));
1603 if (!tool)
1604 return wxNullBitmap;
1605
1606 return tool->m_bitmap;
1607}
1608
1609void wxAuiToolBar::SetToolBitmap(int tool_id, const wxBitmap& bitmap)
1610{
1611 wxAuiToolBarItem* tool = FindTool(tool_id);
1612 if (tool)
1613 {
1614 tool->m_bitmap = bitmap;
1615 }
1616}
1617
1618wxString wxAuiToolBar::GetToolShortHelp(int tool_id) const
1619{
1620 wxAuiToolBarItem* tool = FindTool(tool_id);
1621 wxASSERT_MSG(tool, wxT("can't find tool in toolbar item array"));
1622 if (!tool)
1623 return wxEmptyString;
1624
1625 return tool->m_shortHelp;
1626}
1627
1628void wxAuiToolBar::SetToolShortHelp(int tool_id, const wxString& help_string)
1629{
1630 wxAuiToolBarItem* tool = FindTool(tool_id);
1631 if (tool)
1632 {
1633 tool->m_shortHelp = help_string;
1634 }
1635}
1636
1637wxString wxAuiToolBar::GetToolLongHelp(int tool_id) const
1638{
1639 wxAuiToolBarItem* tool = FindTool(tool_id);
1640 wxASSERT_MSG(tool, wxT("can't find tool in toolbar item array"));
1641 if (!tool)
1642 return wxEmptyString;
1643
1644 return tool->m_longHelp;
1645}
1646
1647void wxAuiToolBar::SetToolLongHelp(int tool_id, const wxString& help_string)
1648{
1649 wxAuiToolBarItem* tool = FindTool(tool_id);
1650 if (tool)
1651 {
1652 tool->m_longHelp = help_string;
1653 }
1654}
1655
1656void wxAuiToolBar::SetCustomOverflowItems(const wxAuiToolBarItemArray& prepend,
1657 const wxAuiToolBarItemArray& append)
1658{
1659 m_customOverflowPrepend = prepend;
1660 m_customOverflowAppend = append;
1661}
1662
1663// get size of hint rectangle for a particular dock location
1664wxSize wxAuiToolBar::GetHintSize(int dock_direction) const
1665{
1666 switch (dock_direction)
1667 {
1668 case wxAUI_DOCK_TOP:
1669 case wxAUI_DOCK_BOTTOM:
1670 return m_horzHintSize;
1671 case wxAUI_DOCK_RIGHT:
1672 case wxAUI_DOCK_LEFT:
1673 return m_vertHintSize;
1674 default:
1675 wxFAIL_MSG("invalid dock location value");
1676 }
1677 return wxDefaultSize;
1678}
1679
1680bool wxAuiToolBar::IsPaneValid(const wxAuiPaneInfo& pane) const
1681{
1682 return IsPaneValid(m_windowStyle, pane);
1683}
1684
1685bool wxAuiToolBar::IsPaneValid(long style, const wxAuiPaneInfo& pane)
1686{
1687 if (style & wxAUI_TB_HORIZONTAL)
1688 {
1689 if (pane.IsLeftDockable() || pane.IsRightDockable())
1690 {
1691 return false;
1692 }
1693 }
1694 else if (style & wxAUI_TB_VERTICAL)
1695 {
1696 if (pane.IsTopDockable() || pane.IsBottomDockable())
1697 {
1698 return false;
1699 }
1700 }
1701 return true;
1702}
1703
1704bool wxAuiToolBar::IsPaneValid(long style) const
1705{
1706 wxAuiManager* manager = wxAuiManager::GetManager(const_cast<wxAuiToolBar*>(this));
1707 if (manager)
1708 {
1709 return IsPaneValid(style, manager->GetPane(const_cast<wxAuiToolBar*>(this)));
1710 }
1711 return true;
1712}
1713
1714void wxAuiToolBar::SetArtFlags() const
1715{
1716 unsigned int artflags = m_windowStyle & ~wxAUI_ORIENTATION_MASK;
1717 if (m_orientation == wxVERTICAL)
1718 {
1719 artflags |= wxAUI_TB_VERTICAL;
1720 }
1721 m_art->SetFlags(artflags);
1722}
1723
1724size_t wxAuiToolBar::GetToolCount() const
1725{
1726 return m_items.size();
1727}
1728
1729int wxAuiToolBar::GetToolIndex(int tool_id) const
1730{
1731 // this will prevent us from returning the index of the
1732 // first separator in the toolbar since its id is equal to -1
1733 if (tool_id == -1)
1734 return wxNOT_FOUND;
1735
1736 size_t i, count = m_items.GetCount();
1737 for (i = 0; i < count; ++i)
1738 {
1739 wxAuiToolBarItem& item = m_items.Item(i);
1740 if (item.m_toolId == tool_id)
1741 return i;
1742 }
1743
1744 return wxNOT_FOUND;
1745}
1746
1747bool wxAuiToolBar::GetToolFitsByIndex(int tool_idx) const
1748{
1749 if (tool_idx < 0 || tool_idx >= (int)m_items.GetCount())
1750 return false;
1751
1752 if (!m_items[tool_idx].m_sizerItem)
1753 return false;
1754
1755 int cli_w, cli_h;
1756 GetClientSize(&cli_w, &cli_h);
1757
1758 wxRect rect = m_items[tool_idx].m_sizerItem->GetRect();
1759
1760 if (m_orientation == wxVERTICAL)
1761 {
1762 // take the dropdown size into account
1763 if (m_overflowVisible)
1764 cli_h -= m_overflowSizerItem->GetSize().y;
1765
1766 if (rect.y+rect.height < cli_h)
1767 return true;
1768 }
1769 else
1770 {
1771 // take the dropdown size into account
1772 if (m_overflowVisible)
1773 cli_w -= m_overflowSizerItem->GetSize().x;
1774
1775 if (rect.x+rect.width < cli_w)
1776 return true;
1777 }
1778
1779 return false;
1780}
1781
1782
1783bool wxAuiToolBar::GetToolFits(int tool_id) const
1784{
1785 return GetToolFitsByIndex(GetToolIndex(tool_id));
1786}
1787
1788wxRect wxAuiToolBar::GetToolRect(int tool_id) const
1789{
1790 wxAuiToolBarItem* tool = FindTool(tool_id);
1791 if (tool && tool->m_sizerItem)
1792 {
1793 return tool->m_sizerItem->GetRect();
1794 }
1795
1796 return wxRect();
1797}
1798
1799bool wxAuiToolBar::GetToolBarFits() const
1800{
1801 if (m_items.GetCount() == 0)
1802 {
1803 // empty toolbar always 'fits'
1804 return true;
1805 }
1806
1807 // entire toolbar content fits if the last tool fits
1808 return GetToolFitsByIndex(m_items.GetCount() - 1);
1809}
1810
1811bool wxAuiToolBar::Realize()
1812{
1813 wxClientDC dc(this);
1814 if (!dc.IsOk())
1815 return false;
1816
1817 // calculate hint sizes for both horizontal and vertical
1818 // in the order that leaves toolbar in correct final state
1819 bool retval = false;
1820 if (m_orientation == wxHORIZONTAL)
1821 {
1822 if (RealizeHelper(dc, false))
1823 {
1824 m_vertHintSize = GetSize();
1825 if (RealizeHelper(dc, true))
1826 {
1827 m_horzHintSize = GetSize();
1828 retval = true;
1829 }
1830 }
1831 }
1832 else
1833 {
1834 if (RealizeHelper(dc, true))
1835 {
1836 m_horzHintSize = GetSize();
1837 if (RealizeHelper(dc, false))
1838 {
1839 m_vertHintSize = GetSize();
1840 retval = true;
1841 }
1842 }
1843 }
1844
1845 Refresh(false);
1846 return retval;
1847}
1848
1849bool wxAuiToolBar::RealizeHelper(wxClientDC& dc, bool horizontal)
1850{
1851 // create the new sizer to add toolbar elements to
1852 wxBoxSizer* sizer = new wxBoxSizer(horizontal ? wxHORIZONTAL : wxVERTICAL);
1853
1854 // add gripper area
1855 int separatorSize = m_art->GetElementSize(wxAUI_TBART_SEPARATOR_SIZE);
1856 int gripperSize = m_art->GetElementSize(wxAUI_TBART_GRIPPER_SIZE);
1857 if (gripperSize > 0 && m_gripperVisible)
1858 {
1859 if (horizontal)
1860 m_gripperSizerItem = sizer->Add(gripperSize, 1, 0, wxEXPAND);
1861 else
1862 m_gripperSizerItem = sizer->Add(1, gripperSize, 0, wxEXPAND);
1863 }
1864 else
1865 {
1866 m_gripperSizerItem = NULL;
1867 }
1868
1869 // add "left" padding
1870 if (m_leftPadding > 0)
1871 {
1872 if (horizontal)
1873 sizer->Add(m_leftPadding, 1);
1874 else
1875 sizer->Add(1, m_leftPadding);
1876 }
1877
1878 size_t i, count;
1879 for (i = 0, count = m_items.GetCount(); i < count; ++i)
1880 {
1881 wxAuiToolBarItem& item = m_items.Item(i);
1882 wxSizerItem* m_sizerItem = NULL;
1883
1884 switch (item.m_kind)
1885 {
1886 case wxITEM_LABEL:
1887 {
1888 wxSize size = m_art->GetLabelSize(dc, this, item);
1889 m_sizerItem = sizer->Add(size.x + (m_toolBorderPadding*2),
1890 size.y + (m_toolBorderPadding*2),
1891 item.m_proportion,
1892 item.m_alignment);
1893 if (i+1 < count)
1894 {
1895 sizer->AddSpacer(m_toolPacking);
1896 }
1897
1898 break;
1899 }
1900
1901 case wxITEM_CHECK:
1902 case wxITEM_NORMAL:
1903 case wxITEM_RADIO:
1904 {
1905 wxSize size = m_art->GetToolSize(dc, this, item);
1906 m_sizerItem = sizer->Add(size.x + (m_toolBorderPadding*2),
1907 size.y + (m_toolBorderPadding*2),
1908 0,
1909 item.m_alignment);
1910 // add tool packing
1911 if (i+1 < count)
1912 {
1913 sizer->AddSpacer(m_toolPacking);
1914 }
1915
1916 break;
1917 }
1918
1919 case wxITEM_SEPARATOR:
1920 {
1921 if (horizontal)
1922 m_sizerItem = sizer->Add(separatorSize, 1, 0, wxEXPAND);
1923 else
1924 m_sizerItem = sizer->Add(1, separatorSize, 0, wxEXPAND);
1925
1926 // add tool packing
1927 if (i+1 < count)
1928 {
1929 sizer->AddSpacer(m_toolPacking);
1930 }
1931
1932 break;
1933 }
1934
1935 case wxITEM_SPACER:
1936 if (item.m_proportion > 0)
1937 m_sizerItem = sizer->AddStretchSpacer(item.m_proportion);
1938 else
1939 m_sizerItem = sizer->Add(item.m_spacerPixels, 1);
1940 break;
1941
1942 case wxITEM_CONTROL:
1943 {
1944 //m_sizerItem = sizer->Add(item.m_window, item.m_proportion, wxEXPAND);
1945 wxSizerItem* ctrl_m_sizerItem;
1946
1947 wxBoxSizer* vert_sizer = new wxBoxSizer(wxVERTICAL);
1948 vert_sizer->AddStretchSpacer(1);
1949 ctrl_m_sizerItem = vert_sizer->Add(item.m_window, 0, wxEXPAND);
1950 vert_sizer->AddStretchSpacer(1);
1951 if ( (m_windowStyle & wxAUI_TB_TEXT) &&
1952 m_toolTextOrientation == wxAUI_TBTOOL_TEXT_BOTTOM &&
1953 !item.GetLabel().empty() )
1954 {
1955 wxSize s = GetLabelSize(item.GetLabel());
1956 vert_sizer->Add(1, s.y);
1957 }
1958
1959
1960 m_sizerItem = sizer->Add(vert_sizer, item.m_proportion, wxEXPAND);
1961
1962 wxSize min_size = item.m_minSize;
1963
1964
1965 // proportional items will disappear from the toolbar if
1966 // their min width is not set to something really small
1967 if (item.m_proportion != 0)
1968 {
1969 min_size.x = 1;
1970 }
1971
1972 if (min_size.IsFullySpecified())
1973 {
1974 m_sizerItem->SetMinSize(min_size);
1975 ctrl_m_sizerItem->SetMinSize(min_size);
1976 }
1977
1978 // add tool packing
1979 if (i+1 < count)
1980 {
1981 sizer->AddSpacer(m_toolPacking);
1982 }
1983 }
1984 }
1985
1986 item.m_sizerItem = m_sizerItem;
1987 }
1988
1989 // add "right" padding
1990 if (m_rightPadding > 0)
1991 {
1992 if (horizontal)
1993 sizer->Add(m_rightPadding, 1);
1994 else
1995 sizer->Add(1, m_rightPadding);
1996 }
1997
1998 // add drop down area
1999 m_overflowSizerItem = NULL;
2000
2001 if (m_windowStyle & wxAUI_TB_OVERFLOW)
2002 {
2003 int overflow_size = m_art->GetElementSize(wxAUI_TBART_OVERFLOW_SIZE);
2004 if (overflow_size > 0 && m_overflowVisible)
2005 {
2006 if (horizontal)
2007 m_overflowSizerItem = sizer->Add(overflow_size, 1, 0, wxEXPAND);
2008 else
2009 m_overflowSizerItem = sizer->Add(1, overflow_size, 0, wxEXPAND);
2010 }
2011 else
2012 {
2013 m_overflowSizerItem = NULL;
2014 }
2015 }
2016
2017
2018 // the outside sizer helps us apply the "top" and "bottom" padding
2019 wxBoxSizer* outside_sizer = new wxBoxSizer(horizontal ? wxVERTICAL : wxHORIZONTAL);
2020
2021 // add "top" padding
2022 if (m_topPadding > 0)
2023 {
2024 if (horizontal)
2025 outside_sizer->Add(1, m_topPadding);
2026 else
2027 outside_sizer->Add(m_topPadding, 1);
2028 }
2029
2030 // add the sizer that contains all of the toolbar elements
2031 outside_sizer->Add(sizer, 1, wxEXPAND);
2032
2033 // add "bottom" padding
2034 if (m_bottomPadding > 0)
2035 {
2036 if (horizontal)
2037 outside_sizer->Add(1, m_bottomPadding);
2038 else
2039 outside_sizer->Add(m_bottomPadding, 1);
2040 }
2041
2042 delete m_sizer; // remove old sizer
2043 m_sizer = outside_sizer;
2044
2045 // calculate the rock-bottom minimum size
2046 for (i = 0, count = m_items.GetCount(); i < count; ++i)
2047 {
2048 wxAuiToolBarItem& item = m_items.Item(i);
2049 if (item.m_sizerItem && item.m_proportion > 0 && item.m_minSize.IsFullySpecified())
2050 item.m_sizerItem->SetMinSize(0,0);
2051 }
2052
2053 m_absoluteMinSize = m_sizer->GetMinSize();
2054
2055 // reset the min sizes to what they were
2056 for (i = 0, count = m_items.GetCount(); i < count; ++i)
2057 {
2058 wxAuiToolBarItem& item = m_items.Item(i);
2059 if (item.m_sizerItem && item.m_proportion > 0 && item.m_minSize.IsFullySpecified())
2060 item.m_sizerItem->SetMinSize(item.m_minSize);
2061 }
2062
2063 // set control size
2064 wxSize size = m_sizer->GetMinSize();
2065 m_minWidth = size.x;
2066 m_minHeight = size.y;
2067
2068 if ((m_windowStyle & wxAUI_TB_NO_AUTORESIZE) == 0)
2069 {
2070 wxSize curSize = GetClientSize();
2071 wxSize new_size = GetMinSize();
2072 if (new_size != curSize)
2073 {
2074 SetClientSize(new_size);
2075 }
2076 else
2077 {
2078 m_sizer->SetDimension(0, 0, curSize.x, curSize.y);
2079 }
2080 }
2081 else
2082 {
2083 wxSize curSize = GetClientSize();
2084 m_sizer->SetDimension(0, 0, curSize.x, curSize.y);
2085 }
2086
2087 return true;
2088}
2089
2090int wxAuiToolBar::GetOverflowState() const
2091{
2092 return m_overflowState;
2093}
2094
2095wxRect wxAuiToolBar::GetOverflowRect() const
2096{
2097 wxRect cli_rect(wxPoint(0,0), GetClientSize());
2098 wxRect overflow_rect = m_overflowSizerItem->GetRect();
2099 int overflow_size = m_art->GetElementSize(wxAUI_TBART_OVERFLOW_SIZE);
2100
2101 if (m_orientation == wxVERTICAL)
2102 {
2103 overflow_rect.y = cli_rect.height - overflow_size;
2104 overflow_rect.x = 0;
2105 overflow_rect.width = cli_rect.width;
2106 overflow_rect.height = overflow_size;
2107 }
2108 else
2109 {
2110 overflow_rect.x = cli_rect.width - overflow_size;
2111 overflow_rect.y = 0;
2112 overflow_rect.width = overflow_size;
2113 overflow_rect.height = cli_rect.height;
2114 }
2115
2116 return overflow_rect;
2117}
2118
2119wxSize wxAuiToolBar::GetLabelSize(const wxString& label)
2120{
2121 wxClientDC dc(this);
2122
2123 int tx, ty;
2124 int textWidth = 0, textHeight = 0;
2125
2126 dc.SetFont(m_font);
2127
2128 // get the text height
2129 dc.GetTextExtent(wxT("ABCDHgj"), &tx, &textHeight);
2130
2131 // get the text width
2132 dc.GetTextExtent(label, &textWidth, &ty);
2133
2134 return wxSize(textWidth, textHeight);
2135}
2136
2137
2138void wxAuiToolBar::DoIdleUpdate()
2139{
2140 wxEvtHandler* handler = GetEventHandler();
2141
2142 bool need_refresh = false;
2143
2144 size_t i, count;
2145 for (i = 0, count = m_items.GetCount(); i < count; ++i)
2146 {
2147 wxAuiToolBarItem& item = m_items.Item(i);
2148
2149 if (item.m_toolId == -1)
2150 continue;
2151
2152 wxUpdateUIEvent evt(item.m_toolId);
2153 evt.SetEventObject(this);
2154
2155 if (handler->ProcessEvent(evt))
2156 {
2157 if (evt.GetSetEnabled())
2158 {
2159 bool is_enabled;
2160 if (item.m_window)
2161 is_enabled = item.m_window->IsThisEnabled();
2162 else
2163 is_enabled = (item.m_state & wxAUI_BUTTON_STATE_DISABLED) ? false : true;
2164
2165 bool new_enabled = evt.GetEnabled();
2166 if (new_enabled != is_enabled)
2167 {
2168 if (item.m_window)
2169 {
2170 item.m_window->Enable(new_enabled);
2171 }
2172 else
2173 {
2174 if (new_enabled)
2175 item.m_state &= ~wxAUI_BUTTON_STATE_DISABLED;
2176 else
2177 item.m_state |= wxAUI_BUTTON_STATE_DISABLED;
2178 }
2179 need_refresh = true;
2180 }
2181 }
2182
2183 if (evt.GetSetChecked())
2184 {
2185 // make sure we aren't checking an item that can't be
2186 if (item.m_kind != wxITEM_CHECK && item.m_kind != wxITEM_RADIO)
2187 continue;
2188
2189 bool is_checked = (item.m_state & wxAUI_BUTTON_STATE_CHECKED) ? true : false;
2190 bool new_checked = evt.GetChecked();
2191
2192 if (new_checked != is_checked)
2193 {
2194 if (new_checked)
2195 item.m_state |= wxAUI_BUTTON_STATE_CHECKED;
2196 else
2197 item.m_state &= ~wxAUI_BUTTON_STATE_CHECKED;
2198
2199 need_refresh = true;
2200 }
2201 }
2202
2203 }
2204 }
2205
2206
2207 if (need_refresh)
2208 {
2209 Refresh(false);
2210 }
2211}
2212
2213
2214void wxAuiToolBar::OnSize(wxSizeEvent& WXUNUSED(evt))
2215{
2216 int x, y;
2217 GetClientSize(&x, &y);
2218
2219 if (((x >= y) && m_absoluteMinSize.x > x) ||
2220 ((y > x) && m_absoluteMinSize.y > y))
2221 {
2222 // hide all flexible items
2223 size_t i, count;
2224 for (i = 0, count = m_items.GetCount(); i < count; ++i)
2225 {
2226 wxAuiToolBarItem& item = m_items.Item(i);
2227 if (item.m_sizerItem && item.m_proportion > 0 && item.m_sizerItem->IsShown())
2228 {
2229 item.m_sizerItem->Show(false);
2230 item.m_sizerItem->SetProportion(0);
2231 }
2232 }
2233 }
2234 else
2235 {
2236 // show all flexible items
2237 size_t i, count;
2238 for (i = 0, count = m_items.GetCount(); i < count; ++i)
2239 {
2240 wxAuiToolBarItem& item = m_items.Item(i);
2241 if (item.m_sizerItem && item.m_proportion > 0 && !item.m_sizerItem->IsShown())
2242 {
2243 item.m_sizerItem->Show(true);
2244 item.m_sizerItem->SetProportion(item.m_proportion);
2245 }
2246 }
2247 }
2248
2249 m_sizer->SetDimension(0, 0, x, y);
2250
2251 Refresh(false);
2252 Update();
2253
2254 // idle events aren't sent while user is resizing frame (why?),
2255 // but resizing toolbar here causes havoc,
2256 // so force idle handler to run after size handling complete
2257 QueueEvent(new wxIdleEvent);
2258}
2259
2260
2261
2262void wxAuiToolBar::DoSetSize(int x,
2263 int y,
2264 int width,
2265 int height,
2266 int sizeFlags)
2267{
2268 wxSize parent_size = GetParent()->GetClientSize();
2269 if (x + width > parent_size.x)
2270 width = wxMax(0, parent_size.x - x);
2271 if (y + height > parent_size.y)
2272 height = wxMax(0, parent_size.y - y);
2273
2274 wxWindow::DoSetSize(x, y, width, height, sizeFlags);
2275}
2276
2277
2278void wxAuiToolBar::OnIdle(wxIdleEvent& evt)
2279{
2280 // if orientation doesn't match dock, fix it
2281 wxAuiManager* manager = wxAuiManager::GetManager(this);
2282 if (manager)
2283 {
2284 wxAuiPaneInfo& pane = manager->GetPane(this);
2285 // pane state member is public, so it might have been changed
2286 // without going through wxPaneInfo::SetFlag() check
2287 bool ok = pane.IsOk();
2288 wxCHECK2_MSG(!ok || IsPaneValid(m_windowStyle, pane), ok = false,
2289 "window settings and pane settings are incompatible");
2290 if (ok)
2291 {
2292 wxOrientation newOrientation = m_orientation;
2293 if (pane.IsDocked())
2294 {
2295 switch (pane.dock_direction)
2296 {
2297 case wxAUI_DOCK_TOP:
2298 case wxAUI_DOCK_BOTTOM:
2299 newOrientation = wxHORIZONTAL;
2300 break;
2301 case wxAUI_DOCK_LEFT:
2302 case wxAUI_DOCK_RIGHT:
2303 newOrientation = wxVERTICAL;
2304 break;
2305 default:
2306 wxFAIL_MSG("invalid dock location value");
2307 }
2308 }
2309 else if (pane.IsResizable() &&
2310 GetOrientation(m_windowStyle) == wxBOTH)
2311 {
2312 // changing orientation in OnSize causes havoc
2313 int x, y;
2314 GetClientSize(&x, &y);
2315
2316 if (x > y)
2317 {
2318 newOrientation = wxHORIZONTAL;
2319 }
2320 else
2321 {
2322 newOrientation = wxVERTICAL;
2323 }
2324 }
2325 if (newOrientation != m_orientation)
2326 {
2327 SetOrientation(newOrientation);
2328 Realize();
2329 if (newOrientation == wxHORIZONTAL)
2330 {
2331 pane.best_size = GetHintSize(wxAUI_DOCK_TOP);
2332 }
2333 else
2334 {
2335 pane.best_size = GetHintSize(wxAUI_DOCK_LEFT);
2336 }
2337 if (pane.IsDocked())
2338 {
2339 pane.floating_size = wxDefaultSize;
2340 }
2341 else
2342 {
2343 SetSize(GetParent()->GetClientSize());
2344 }
2345 manager->Update();
2346 }
2347 }
2348 }
2349 evt.Skip();
2350}
2351
2352void wxAuiToolBar::UpdateWindowUI(long flags)
2353{
2354 if ( flags & wxUPDATE_UI_FROMIDLE )
2355 {
2356 DoIdleUpdate();
2357 }
2358
2359 wxControl::UpdateWindowUI(flags);
2360}
2361
2362void wxAuiToolBar::OnPaint(wxPaintEvent& WXUNUSED(evt))
2363{
2364 wxAutoBufferedPaintDC dc(this);
2365 wxRect cli_rect(wxPoint(0,0), GetClientSize());
2366
2367
2368 bool horizontal = m_orientation == wxHORIZONTAL;
2369
2370
2371 m_art->DrawBackground(dc, this, cli_rect);
2372
2373 int gripperSize = m_art->GetElementSize(wxAUI_TBART_GRIPPER_SIZE);
2374 int dropdown_size = m_art->GetElementSize(wxAUI_TBART_OVERFLOW_SIZE);
2375
2376 // paint the gripper
2377 if (gripperSize > 0 && m_gripperSizerItem)
2378 {
2379 wxRect gripper_rect = m_gripperSizerItem->GetRect();
2380 if (horizontal)
2381 gripper_rect.width = gripperSize;
2382 else
2383 gripper_rect.height = gripperSize;
2384 m_art->DrawGripper(dc, this, gripper_rect);
2385 }
2386
2387 // calculated how far we can draw items
2388 int last_extent;
2389 if (horizontal)
2390 last_extent = cli_rect.width;
2391 else
2392 last_extent = cli_rect.height;
2393 if (m_overflowVisible)
2394 last_extent -= dropdown_size;
2395
2396 // paint each individual tool
2397 size_t i, count = m_items.GetCount();
2398 for (i = 0; i < count; ++i)
2399 {
2400 wxAuiToolBarItem& item = m_items.Item(i);
2401
2402 if (!item.m_sizerItem)
2403 continue;
2404
2405 wxRect item_rect = item.m_sizerItem->GetRect();
2406
2407
2408 if ((horizontal && item_rect.x + item_rect.width >= last_extent) ||
2409 (!horizontal && item_rect.y + item_rect.height >= last_extent))
2410 {
2411 break;
2412 }
2413
2414 switch ( item.m_kind )
2415 {
2416 case wxITEM_NORMAL:
2417 // draw a regular or dropdown button
2418 if (!item.m_dropDown)
2419 m_art->DrawButton(dc, this, item, item_rect);
2420 else
2421 m_art->DrawDropDownButton(dc, this, item, item_rect);
2422 break;
2423
2424 case wxITEM_CHECK:
2425 case wxITEM_RADIO:
2426 // draw a toggle button
2427 m_art->DrawButton(dc, this, item, item_rect);
2428 break;
2429
2430 case wxITEM_SEPARATOR:
2431 // draw a separator
2432 m_art->DrawSeparator(dc, this, item_rect);
2433 break;
2434
2435 case wxITEM_LABEL:
2436 // draw a text label only
2437 m_art->DrawLabel(dc, this, item, item_rect);
2438 break;
2439
2440 case wxITEM_CONTROL:
2441 // draw the control's label
2442 m_art->DrawControlLabel(dc, this, item, item_rect);
2443 break;
2444 }
2445
2446 // fire a signal to see if the item wants to be custom-rendered
2447 OnCustomRender(dc, item, item_rect);
2448 }
2449
2450 // paint the overflow button
2451 if (dropdown_size > 0 && m_overflowSizerItem)
2452 {
2453 wxRect dropDownRect = GetOverflowRect();
2454 m_art->DrawOverflowButton(dc, this, dropDownRect, m_overflowState);
2455 }
2456}
2457
2458void wxAuiToolBar::OnEraseBackground(wxEraseEvent& WXUNUSED(evt))
2459{
2460 // empty
2461}
2462
2463void wxAuiToolBar::OnLeftDown(wxMouseEvent& evt)
2464{
2465 wxRect cli_rect(wxPoint(0,0), GetClientSize());
2466
2467 if (m_gripperSizerItem)
2468 {
2469 wxRect gripper_rect = m_gripperSizerItem->GetRect();
2470 if (gripper_rect.Contains(evt.GetX(), evt.GetY()))
2471 {
2472 // find aui manager
2473 wxAuiManager* manager = wxAuiManager::GetManager(this);
2474 if (!manager)
2475 return;
2476
2477 int x_drag_offset = evt.GetX() - gripper_rect.GetX();
2478 int y_drag_offset = evt.GetY() - gripper_rect.GetY();
2479
2480 // gripper was clicked
2481 manager->StartPaneDrag(this, wxPoint(x_drag_offset, y_drag_offset));
2482 return;
2483 }
2484 }
2485
2486 if (m_overflowSizerItem)
2487 {
2488 wxRect overflow_rect = GetOverflowRect();
2489
2490 if (m_art &&
2491 m_overflowVisible &&
2492 overflow_rect.Contains(evt.m_x, evt.m_y))
2493 {
2494 wxAuiToolBarEvent e(wxEVT_COMMAND_AUITOOLBAR_OVERFLOW_CLICK, -1);
2495 e.SetEventObject(this);
2496 e.SetToolId(-1);
2497 e.SetClickPoint(wxPoint(evt.GetX(), evt.GetY()));
2498 bool processed = GetEventHandler()->ProcessEvent(e);
2499
2500 if (processed)
2501 {
2502 DoIdleUpdate();
2503 }
2504 else
2505 {
2506 size_t i, count;
2507 wxAuiToolBarItemArray overflow_items;
2508
2509
2510 // add custom overflow prepend items, if any
2511 count = m_customOverflowPrepend.GetCount();
2512 for (i = 0; i < count; ++i)
2513 overflow_items.Add(m_customOverflowPrepend[i]);
2514
2515 // only show items that don't fit in the dropdown
2516 count = m_items.GetCount();
2517 for (i = 0; i < count; ++i)
2518 {
2519 if (!GetToolFitsByIndex(i))
2520 overflow_items.Add(m_items[i]);
2521 }
2522
2523 // add custom overflow append items, if any
2524 count = m_customOverflowAppend.GetCount();
2525 for (i = 0; i < count; ++i)
2526 overflow_items.Add(m_customOverflowAppend[i]);
2527
2528 int res = m_art->ShowDropDown(this, overflow_items);
2529 m_overflowState = 0;
2530 Refresh(false);
2531 if (res != -1)
2532 {
2533 wxCommandEvent event(wxEVT_COMMAND_MENU_SELECTED, res);
2534 event.SetEventObject(this);
2535 GetParent()->GetEventHandler()->ProcessEvent(event);
2536 }
2537 }
2538
2539 return;
2540 }
2541 }
2542
2543 m_dragging = false;
2544 m_actionPos = wxPoint(evt.GetX(), evt.GetY());
2545 m_actionItem = FindToolByPosition(evt.GetX(), evt.GetY());
2546
2547 if (m_actionItem)
2548 {
2549 if (m_actionItem->m_state & wxAUI_BUTTON_STATE_DISABLED)
2550 {
2551 m_actionPos = wxPoint(-1,-1);
2552 m_actionItem = NULL;
2553 return;
2554 }
2555
2556 UnsetToolTip();
2557
2558 // fire the tool dropdown event
2559 wxAuiToolBarEvent e(wxEVT_COMMAND_AUITOOLBAR_TOOL_DROPDOWN, m_actionItem->m_toolId);
2560 e.SetEventObject(this);
2561 e.SetToolId(m_actionItem->m_toolId);
2562
2563 int mouse_x = evt.GetX();
2564 wxRect rect = m_actionItem->m_sizerItem->GetRect();
2565 const bool dropDownHit = m_actionItem->m_dropDown &&
2566 mouse_x >= (rect.x+rect.width-BUTTON_DROPDOWN_WIDTH-1) &&
2567 mouse_x < (rect.x+rect.width);
2568 e.SetDropDownClicked(dropDownHit);
2569
2570 e.SetClickPoint(evt.GetPosition());
2571 e.SetItemRect(rect);
2572
2573 // we only set the 'pressed button' state if we hit the actual button
2574 // and not just the drop-down
2575 SetPressedItem(dropDownHit ? 0 : m_actionItem);
2576
2577 if(dropDownHit)
2578 {
2579 m_actionPos = wxPoint(-1,-1);
2580 m_actionItem = NULL;
2581 }
2582
2583 if(!GetEventHandler()->ProcessEvent(e) || e.GetSkipped())
2584 CaptureMouse();
2585
2586 // Ensure hovered item is really ok, as mouse may have moved during
2587 // event processing
2588 wxPoint cursor_pos_after_evt = ScreenToClient(wxGetMousePosition());
2589 SetHoverItem(FindToolByPosition(cursor_pos_after_evt.x, cursor_pos_after_evt.y));
2590
2591 DoIdleUpdate();
2592 }
2593}
2594
2595void wxAuiToolBar::OnLeftUp(wxMouseEvent& evt)
2596{
2597 if (!HasCapture())
2598 return;
2599
2600 SetPressedItem(NULL);
2601
2602 wxAuiToolBarItem* hitItem;
2603 hitItem = FindToolByPosition(evt.GetX(), evt.GetY());
2604 SetHoverItem(hitItem);
2605
2606 if (m_dragging)
2607 {
2608 // TODO: it would make sense to send out an 'END_DRAG' event here,
2609 // otherwise a client would never know what to do with the 'BEGIN_DRAG'
2610 // event
2611
2612 // OnCaptureLost() will be called now and this will reset all our state
2613 // tracking variables
2614 ReleaseMouse();
2615 }
2616 else
2617 {
2618 if (m_actionItem && hitItem == m_actionItem)
2619 {
2620 UnsetToolTip();
2621
2622 wxCommandEvent e(wxEVT_COMMAND_MENU_SELECTED, m_actionItem->m_toolId);
2623 e.SetEventObject(this);
2624
2625 if (hitItem->m_kind == wxITEM_CHECK || hitItem->m_kind == wxITEM_RADIO)
2626 {
2627 const bool toggle = !(m_actionItem->m_state & wxAUI_BUTTON_STATE_CHECKED);
2628
2629 ToggleTool(m_actionItem->m_toolId, toggle);
2630
2631 // repaint immediately
2632 Refresh(false);
2633 Update();
2634
2635 e.SetInt(toggle);
2636 }
2637
2638 // we have to release the mouse *before* sending the event, because
2639 // we don't know what a handler might do. It could open up a popup
2640 // menu for example and that would make us lose our capture anyway.
2641
2642 ReleaseMouse();
2643
2644 GetEventHandler()->ProcessEvent(e);
2645
2646 // Ensure hovered item is really ok, as mouse may have moved during
2647 // event processing
2648 wxPoint cursor_pos_after_evt = ScreenToClient(wxGetMousePosition());
2649 SetHoverItem(FindToolByPosition(cursor_pos_after_evt.x, cursor_pos_after_evt.y));
2650
2651 DoIdleUpdate();
2652 }
2653 else
2654 ReleaseMouse();
2655 }
2656}
2657
2658void wxAuiToolBar::OnRightDown(wxMouseEvent& evt)
2659{
2660 wxRect cli_rect(wxPoint(0,0), GetClientSize());
2661
2662 if (m_gripperSizerItem)
2663 {
2664 wxRect gripper_rect = m_gripperSizerItem->GetRect();
2665 if (gripper_rect.Contains(evt.GetX(), evt.GetY()))
2666 return;
2667 }
2668
2669 if (m_overflowSizerItem && m_art)
2670 {
2671 int dropdown_size = m_art->GetElementSize(wxAUI_TBART_OVERFLOW_SIZE);
2672 if (dropdown_size > 0 &&
2673 evt.m_x > cli_rect.width - dropdown_size &&
2674 evt.m_y >= 0 &&
2675 evt.m_y < cli_rect.height)
2676 {
2677 return;
2678 }
2679 }
2680
2681 m_actionPos = wxPoint(evt.GetX(), evt.GetY());
2682 m_actionItem = FindToolByPosition(evt.GetX(), evt.GetY());
2683
2684 if (m_actionItem && m_actionItem->m_state & wxAUI_BUTTON_STATE_DISABLED)
2685 {
2686 m_actionPos = wxPoint(-1,-1);
2687 m_actionItem = NULL;
2688 return;
2689 }
2690
2691 UnsetToolTip();
2692}
2693
2694void wxAuiToolBar::OnRightUp(wxMouseEvent& evt)
2695{
2696 wxAuiToolBarItem* hitItem;
2697 hitItem = FindToolByPosition(evt.GetX(), evt.GetY());
2698
2699 if (m_actionItem && hitItem == m_actionItem)
2700 {
2701 wxAuiToolBarEvent e(wxEVT_COMMAND_AUITOOLBAR_RIGHT_CLICK, m_actionItem->m_toolId);
2702 e.SetEventObject(this);
2703 e.SetToolId(m_actionItem->m_toolId);
2704 e.SetClickPoint(m_actionPos);
2705 GetEventHandler()->ProcessEvent(e);
2706 DoIdleUpdate();
2707 }
2708 else
2709 {
2710 // right-clicked on the invalid area of the toolbar
2711 wxAuiToolBarEvent e(wxEVT_COMMAND_AUITOOLBAR_RIGHT_CLICK, -1);
2712 e.SetEventObject(this);
2713 e.SetToolId(-1);
2714 e.SetClickPoint(m_actionPos);
2715 GetEventHandler()->ProcessEvent(e);
2716 DoIdleUpdate();
2717 }
2718
2719 // reset member variables
2720 m_actionPos = wxPoint(-1,-1);
2721 m_actionItem = NULL;
2722}
2723
2724void wxAuiToolBar::OnMiddleDown(wxMouseEvent& evt)
2725{
2726 wxRect cli_rect(wxPoint(0,0), GetClientSize());
2727
2728 if (m_gripperSizerItem)
2729 {
2730 wxRect gripper_rect = m_gripperSizerItem->GetRect();
2731 if (gripper_rect.Contains(evt.GetX(), evt.GetY()))
2732 return;
2733 }
2734
2735 if (m_overflowSizerItem && m_art)
2736 {
2737 int dropdown_size = m_art->GetElementSize(wxAUI_TBART_OVERFLOW_SIZE);
2738 if (dropdown_size > 0 &&
2739 evt.m_x > cli_rect.width - dropdown_size &&
2740 evt.m_y >= 0 &&
2741 evt.m_y < cli_rect.height)
2742 {
2743 return;
2744 }
2745 }
2746
2747 m_actionPos = wxPoint(evt.GetX(), evt.GetY());
2748 m_actionItem = FindToolByPosition(evt.GetX(), evt.GetY());
2749
2750 if (m_actionItem)
2751 {
2752 if (m_actionItem->m_state & wxAUI_BUTTON_STATE_DISABLED)
2753 {
2754 m_actionPos = wxPoint(-1,-1);
2755 m_actionItem = NULL;
2756 return;
2757 }
2758 }
2759
2760 UnsetToolTip();
2761}
2762
2763void wxAuiToolBar::OnMiddleUp(wxMouseEvent& evt)
2764{
2765 wxAuiToolBarItem* hitItem;
2766 hitItem = FindToolByPosition(evt.GetX(), evt.GetY());
2767
2768 if (m_actionItem && hitItem == m_actionItem)
2769 {
2770 if (hitItem->m_kind == wxITEM_NORMAL)
2771 {
2772 wxAuiToolBarEvent e(wxEVT_COMMAND_AUITOOLBAR_MIDDLE_CLICK, m_actionItem->m_toolId);
2773 e.SetEventObject(this);
2774 e.SetToolId(m_actionItem->m_toolId);
2775 e.SetClickPoint(m_actionPos);
2776 GetEventHandler()->ProcessEvent(e);
2777 DoIdleUpdate();
2778 }
2779 }
2780
2781 // reset member variables
2782 m_actionPos = wxPoint(-1,-1);
2783 m_actionItem = NULL;
2784}
2785
2786void wxAuiToolBar::OnMotion(wxMouseEvent& evt)
2787{
2788 const bool button_pressed = HasCapture();
2789
2790 // start a drag event
2791 if (!m_dragging && button_pressed &&
2792 abs(evt.GetX() - m_actionPos.x) + abs(evt.GetY() - m_actionPos.y) > 5)
2793 {
2794 // TODO: sending this event only makes sense if there is an 'END_DRAG'
2795 // event sent sometime in the future (see OnLeftUp())
2796 wxAuiToolBarEvent e(wxEVT_COMMAND_AUITOOLBAR_BEGIN_DRAG, GetId());
2797 e.SetEventObject(this);
2798 e.SetToolId(m_actionItem->m_toolId);
2799 m_dragging = GetEventHandler()->ProcessEvent(e) && !e.GetSkipped();
2800
2801 DoIdleUpdate();
2802 }
2803
2804 if(m_dragging)
2805 return;
2806
2807 wxAuiToolBarItem* hitItem = FindToolByPosition(evt.GetX(), evt.GetY());
2808 if(button_pressed)
2809 {
2810 // if we have a button pressed we want it to be shown in 'depressed'
2811 // state unless we move the mouse outside the button, then we want it
2812 // to show as just 'highlighted'
2813 if (hitItem == m_actionItem)
2814 SetPressedItem(m_actionItem);
2815 else
2816 {
2817 SetPressedItem(NULL);
2818 SetHoverItem(m_actionItem);
2819 }
2820 }
2821 else
2822 {
2823 SetHoverItem(hitItem);
2824
2825 // tooltips handling
2826 wxAuiToolBarItem* packingHitItem;
2827 packingHitItem = FindToolByPositionWithPacking(evt.GetX(), evt.GetY());
2828 if (packingHitItem)
2829 {
2830 if (packingHitItem != m_tipItem)
2831 {
2832 m_tipItem = packingHitItem;
2833
2834 if ( !packingHitItem->m_shortHelp.empty() )
2835 SetToolTip(packingHitItem->m_shortHelp);
2836 else
2837 UnsetToolTip();
2838 }
2839 }
2840 else
2841 {
2842 UnsetToolTip();
2843 m_tipItem = NULL;
2844 }
2845
2846 // figure out the dropdown button state (are we hovering or pressing it?)
2847 RefreshOverflowState();
2848 }
2849}
2850
2851void wxAuiToolBar::DoResetMouseState()
2852{
2853 RefreshOverflowState();
2854 SetHoverItem(NULL);
2855 SetPressedItem(NULL);
2856
2857 m_tipItem = NULL;
2858
2859 // we have to reset those here, because the mouse-up handlers which do
2860 // it usually won't be called if we let go of a mouse button while we
2861 // are outside of the window
2862 m_actionPos = wxPoint(-1,-1);
2863 m_actionItem = NULL;
2864}
2865
2866void wxAuiToolBar::OnLeaveWindow(wxMouseEvent& evt)
2867{
2868 if(HasCapture())
2869 {
2870 evt.Skip();
2871 return;
2872 }
2873
2874 DoResetMouseState();
2875}
2876
2877void wxAuiToolBar::OnCaptureLost(wxMouseCaptureLostEvent& WXUNUSED(evt))
2878{
2879 m_dragging = false;
2880
2881 DoResetMouseState();
2882}
2883
2884void wxAuiToolBar::OnSetCursor(wxSetCursorEvent& evt)
2885{
2886 wxCursor cursor = wxNullCursor;
2887
2888 if (m_gripperSizerItem)
2889 {
2890 wxRect gripper_rect = m_gripperSizerItem->GetRect();
2891 if (gripper_rect.Contains(evt.GetX(), evt.GetY()))
2892 {
2893 cursor = wxCursor(wxCURSOR_SIZING);
2894 }
2895 }
2896
2897 evt.SetCursor(cursor);
2898}
2899
2900
2901#endif // wxUSE_AUI
2902