1 ///////////////////////////////////////////////////////////////////////////// 
   2 // Name:        src/msw/toolbar.cpp 
   4 // Author:      Julian Smart 
   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" 
  27 #if wxUSE_TOOLBAR && wxUSE_TOOLBAR_NATIVE && !defined(__SMARTPHONE__) 
  29 #include "wx/toolbar.h" 
  32     #include "wx/msw/wrapcctl.h" // include <commctrl.h> "properly" 
  33     #include "wx/dynarray.h" 
  37     #include "wx/settings.h" 
  38     #include "wx/bitmap.h" 
  39     #include "wx/region.h" 
  40     #include "wx/dcmemory.h" 
  41     #include "wx/control.h" 
  42     #include "wx/app.h"         // for GetComCtl32Version 
  44     #include "wx/stattext.h" 
  47 #include "wx/artprov.h" 
  48 #include "wx/sysopt.h" 
  49 #include "wx/dcclient.h" 
  50 #include "wx/scopedarray.h" 
  52 #include "wx/msw/private.h" 
  53 #include "wx/msw/dc.h" 
  56 #include "wx/msw/uxtheme.h" 
  59 // this define controls whether the code for button colours remapping (only 
  60 // useful for 16 or 256 colour images) is active at all, it's always turned off 
  61 // for CE where it doesn't compile (and is probably not needed anyhow) and may 
  62 // also be turned off for other systems if you always use 24bpp images and so 
  65     #define wxREMAP_BUTTON_COLOURS 
  66 #endif // !__WXWINCE__ 
  68 // ---------------------------------------------------------------------------- 
  70 // ---------------------------------------------------------------------------- 
  72 // these standard constants are not always defined in compilers headers 
  76     #define TBSTYLE_LIST            0x1000 
  77     #define TBSTYLE_FLAT            0x0800 
  80 #ifndef TBSTYLE_TRANSPARENT 
  81     #define TBSTYLE_TRANSPARENT     0x8000 
  84 #ifndef TBSTYLE_TOOLTIPS 
  85     #define TBSTYLE_TOOLTIPS        0x0100 
  90     #define TB_SETSTYLE             (WM_USER + 56) 
  91     #define TB_GETSTYLE             (WM_USER + 57) 
  95     #define TB_HITTEST              (WM_USER + 69) 
  99     #define TB_GETMAXSIZE           (WM_USER + 83) 
 102 // ---------------------------------------------------------------------------- 
 104 // ---------------------------------------------------------------------------- 
 106 IMPLEMENT_DYNAMIC_CLASS(wxToolBar
, wxControl
) 
 118         style ( wxNO_BORDER | wxTB_HORIZONTAL) 
 127 BEGIN_EVENT_TABLE(wxToolBar
, wxToolBarBase
) 
 128     EVT_MOUSE_EVENTS(wxToolBar::OnMouseEvent
) 
 129     EVT_SYS_COLOUR_CHANGED(wxToolBar::OnSysColourChanged
) 
 132 // ---------------------------------------------------------------------------- 
 134 // ---------------------------------------------------------------------------- 
 136 class wxToolBarTool 
: public wxToolBarToolBase
 
 139     wxToolBarTool(wxToolBar 
*tbar
, 
 141                   const wxString
& label
, 
 142                   const wxBitmap
& bmpNormal
, 
 143                   const wxBitmap
& bmpDisabled
, 
 145                   wxObject 
*clientData
, 
 146                   const wxString
& shortHelp
, 
 147                   const wxString
& longHelp
) 
 148         : wxToolBarToolBase(tbar
, id
, label
, bmpNormal
, bmpDisabled
, kind
, 
 149                             clientData
, shortHelp
, longHelp
) 
 155     wxToolBarTool(wxToolBar 
*tbar
, wxControl 
*control
, const wxString
& label
) 
 156         : wxToolBarToolBase(tbar
, control
, label
) 
 158         if ( IsControl() && !m_label
.empty() ) 
 160             // create a control to render the control's label 
 161             m_staticText 
= new wxStaticText
 
 168                                  wxALIGN_CENTRE 
| wxST_NO_AUTORESIZE
 
 179     virtual ~wxToolBarTool() 
 184     virtual void SetLabel(const wxString
& label
) 
 186         if ( label 
== m_label 
) 
 189         wxToolBarToolBase::SetLabel(label
); 
 192             m_staticText
->SetLabel(label
); 
 194         // we need to update the label shown in the toolbar because it has a 
 195         // pointer to the internal buffer of the old label 
 197         // TODO: use TB_SETBUTTONINFO 
 200     wxStaticText
* GetStaticText() 
 202         wxASSERT_MSG( IsControl(), 
 203                       wxT("only makes sense for embedded control tools") ); 
 208     // set/get the number of separators which we use to cover the space used by 
 209     // a control in the toolbar 
 210     void SetSeparatorsCount(size_t count
) { m_nSepCount 
= count
; } 
 211     size_t GetSeparatorsCount() const { return m_nSepCount
; } 
 213     // we need ids for the spacers which we want to modify later on, this 
 214     // function will allocate a valid/unique id for a spacer if not done yet 
 217         if ( m_id 
== wxID_SEPARATOR 
) 
 218             m_id 
= wxWindow::NewControlId(); 
 221     // this method is used for controls only and offsets the control by the 
 222     // given amount (in pixels) in horizontal direction 
 223     void MoveBy(int offset
) 
 225         wxControl 
* const control 
= GetControl(); 
 227         control
->Move(control
->GetPosition().x 
+ offset
, wxDefaultCoord
); 
 231             m_staticText
->Move(m_staticText
->GetPosition().x 
+ offset
, 
 238     wxStaticText 
*m_staticText
; 
 240     wxDECLARE_NO_COPY_CLASS(wxToolBarTool
); 
 243 // ---------------------------------------------------------------------------- 
 245 // ---------------------------------------------------------------------------- 
 247 // return the rectangle of the item at the given index 
 249 // returns an empty (0, 0, 0, 0) rectangle if fails so the caller may compare 
 250 // r.right or r.bottom with 0 to check for this 
 251 static RECT 
wxGetTBItemRect(HWND hwnd
, int index
) 
 255     // note that we use TB_GETITEMRECT and not TB_GETRECT because the latter 
 256     // only appeared in v4.70 of comctl32.dll 
 257     if ( !::SendMessage(hwnd
, TB_GETITEMRECT
, index
, (LPARAM
)&r
) ) 
 259         wxLogLastError(wxT("TB_GETITEMRECT")); 
 270 // ============================================================================ 
 272 // ============================================================================ 
 274 // ---------------------------------------------------------------------------- 
 276 // ---------------------------------------------------------------------------- 
 278 wxToolBarToolBase 
*wxToolBar::CreateTool(int id
, 
 279                                          const wxString
& label
, 
 280                                          const wxBitmap
& bmpNormal
, 
 281                                          const wxBitmap
& bmpDisabled
, 
 283                                          wxObject 
*clientData
, 
 284                                          const wxString
& shortHelp
, 
 285                                          const wxString
& longHelp
) 
 287     return new wxToolBarTool(this, id
, label
, bmpNormal
, bmpDisabled
, kind
, 
 288                              clientData
, shortHelp
, longHelp
); 
 292 wxToolBar::CreateTool(wxControl 
*control
, const wxString
& label
) 
 294     return new wxToolBarTool(this, control
, label
); 
 297 // ---------------------------------------------------------------------------- 
 298 // wxToolBar construction 
 299 // ---------------------------------------------------------------------------- 
 301 void wxToolBar::Init() 
 304     m_disabledImgList 
= NULL
; 
 307     m_totalFixedSize 
= 0; 
 309     // even though modern Windows applications typically use 24*24 (or even 
 310     // 32*32) size for their bitmaps, the native control itself still uses the 
 311     // old 16*15 default size (see TB_SETBITMAPSIZE documentation in MSDN), so 
 312     // default to it so that we don't call SetToolBitmapSize() unnecessarily in 
 313     // wxToolBarBase::AdjustToolBitmapSize() 
 315     m_defaultHeight 
= 15; 
 320 bool wxToolBar::Create(wxWindow 
*parent
, 
 325                        const wxString
& name
) 
 327     // common initialisation 
 328     if ( !CreateControl(parent
, id
, pos
, size
, style
, wxDefaultValidator
, name
) ) 
 333     // MSW-specific initialisation 
 334     if ( !MSWCreateToolbar(pos
, size
) ) 
 337     wxSetCCUnicodeFormat(GetHwnd()); 
 339     // we always erase our background on WM_PAINT so there is no need to do it 
 340     // in WM_ERASEBKGND too (by default this won't be done but if the toolbar 
 341     // has a non default background colour, then it would be used in both 
 342     // places resulting in flicker) 
 343     SetBackgroundStyle(wxBG_STYLE_PAINT
); 
 348 bool wxToolBar::MSWCreateToolbar(const wxPoint
& pos
, const wxSize
& size
) 
 350     if ( !MSWCreateControl(TOOLBARCLASSNAME
, wxEmptyString
, pos
, size
) ) 
 353     // toolbar-specific post initialisation 
 354     ::SendMessage(GetHwnd(), TB_BUTTONSTRUCTSIZE
, sizeof(TBBUTTON
), 0); 
 356 #ifdef TB_SETEXTENDEDSTYLE 
 357     if ( wxApp::GetComCtl32Version() >= 471 ) 
 358         ::SendMessage(GetHwnd(), TB_SETEXTENDEDSTYLE
, 0, TBSTYLE_EX_DRAWDDARROWS
); 
 364 void wxToolBar::Recreate() 
 366     const HWND hwndOld 
= GetHwnd(); 
 369         // we haven't been created yet, no need to recreate 
 373     // get the position and size before unsubclassing the old toolbar 
 374     const wxPoint pos 
= GetPosition(); 
 375     const wxSize size 
= GetSize(); 
 379     if ( !MSWCreateToolbar(pos
, size
) ) 
 382         wxFAIL_MSG( wxT("recreating the toolbar failed") ); 
 387     // reparent all our children under the new toolbar 
 388     for ( wxWindowList::compatibility_iterator node 
= m_children
.GetFirst(); 
 390           node 
= node
->GetNext() ) 
 392         wxWindow 
*win 
= node
->GetData(); 
 393         if ( !win
->IsTopLevel() ) 
 394             ::SetParent(GetHwndOf(win
), GetHwnd()); 
 397     // only destroy the old toolbar now -- 
 398     // after all the children had been reparented 
 399     ::DestroyWindow(hwndOld
); 
 401     // it is for the old bitmap control and can't be used with the new one 
 404         ::DeleteObject((HBITMAP
) m_hBitmap
); 
 408     wxDELETE(m_disabledImgList
); 
 413 wxToolBar::~wxToolBar() 
 415     // we must refresh the frame size when the toolbar is deleted but the frame 
 416     // is not - otherwise toolbar leaves a hole in the place it used to occupy 
 417     SendSizeEventToParent(); 
 420         ::DeleteObject((HBITMAP
) m_hBitmap
); 
 422     delete m_disabledImgList
; 
 425 wxSize 
wxToolBar::DoGetBestSize() const 
 430     if ( !::SendMessage(GetHwnd(), TB_GETMAXSIZE
, 0, (LPARAM
)&size
) ) 
 432         // maybe an old (< 0x400) Windows version? try to approximate the 
 433         // toolbar size ourselves 
 434         sizeBest 
= GetToolSize(); 
 435         sizeBest
.y 
+= 2 * ::GetSystemMetrics(SM_CYBORDER
); // Add borders 
 436         sizeBest
.x 
*= GetToolsCount(); 
 438         // reverse horz and vertical components if necessary 
 442             sizeBest
.x 
= sizeBest
.y
; 
 446     else // TB_GETMAXSIZE succeeded 
 448         // but it could still return an incorrect result due to what appears to 
 449         // be a bug in old comctl32.dll versions which don't handle controls in 
 450         // the toolbar correctly, so work around it (see SF patch 1902358) 
 451         if ( !IsVertical() && wxApp::GetComCtl32Version() < 600 ) 
 453             // calculate the toolbar width in alternative way 
 454             const RECT rcFirst 
= wxGetTBItemRect(GetHwnd(), 0); 
 455             const RECT rcLast 
= wxGetTBItemRect(GetHwnd(), GetToolsCount() - 1); 
 457             const int widthAlt 
= rcLast
.right 
- rcFirst
.left
; 
 458             if ( widthAlt 
> size
.cx 
) 
 462         sizeBest
.x 
= size
.cx
; 
 463         sizeBest
.y 
= size
.cy
; 
 468         // Without the extra height, DoGetBestSize can report a size that's 
 469         // smaller than the actual window, causing windows to overlap slightly 
 470         // in some circumstances, leading to missing borders (especially noticeable 
 472         if (!(GetWindowStyle() & wxTB_NODIVIDER
)) 
 477     CacheBestSize(sizeBest
); 
 482 WXDWORD 
wxToolBar::MSWGetStyle(long style
, WXDWORD 
*exstyle
) const 
 484     // toolbars never have border, giving one to them results in broken 
 486     WXDWORD msStyle 
= wxControl::MSWGetStyle
 
 488                         (style 
& ~wxBORDER_MASK
) | wxBORDER_NONE
, exstyle
 
 491     if ( !(style 
& wxTB_NO_TOOLTIPS
) ) 
 492         msStyle 
|= TBSTYLE_TOOLTIPS
; 
 494     if ( style 
& wxTB_FLAT 
&& wxApp::GetComCtl32Version() > 400 ) 
 495         msStyle 
|= TBSTYLE_FLAT
; 
 497     if ( style 
& wxTB_HORZ_LAYOUT 
&& wxApp::GetComCtl32Version() >= 470 ) 
 498         msStyle 
|= TBSTYLE_LIST
; 
 500     if ( style 
& wxTB_NODIVIDER 
) 
 501         msStyle 
|= CCS_NODIVIDER
; 
 503     if ( style 
& wxTB_NOALIGN 
) 
 504         msStyle 
|= CCS_NOPARENTALIGN
; 
 506     if ( style 
& wxTB_VERTICAL 
) 
 509     if( style 
& wxTB_BOTTOM 
) 
 510         msStyle 
|= CCS_BOTTOM
; 
 512     if ( style 
& wxTB_RIGHT 
) 
 513         msStyle 
|= CCS_RIGHT
; 
 515     // always use TBSTYLE_TRANSPARENT because the background is not drawn 
 516     // correctly without it in all themes and, for whatever reason, the control 
 517     // also flickers horribly when it is resized if this style is not used 
 519     // note that this is implicitly enabled by the native toolbar itself when 
 520     // TBSTYLE_FLAT is used (i.e. it's impossible to use TBSTYLE_FLAT without 
 521     // TBSTYLE_TRANSPARENT) but turn it on explicitly in any case 
 522     msStyle 
|= TBSTYLE_TRANSPARENT
; 
 527 // ---------------------------------------------------------------------------- 
 528 // adding/removing tools 
 529 // ---------------------------------------------------------------------------- 
 531 bool wxToolBar::DoInsertTool(size_t WXUNUSED(pos
), 
 532                              wxToolBarToolBase 
* WXUNUSED(tool
)) 
 534     // nothing special to do here - we really create the toolbar buttons in 
 536     InvalidateBestSize(); 
 540 bool wxToolBar::DoDeleteTool(size_t pos
, wxToolBarToolBase 
*tool
) 
 542     // the main difficulty we have here is with the controls in the toolbars: 
 543     // as we (sometimes) use several separators to cover up the space used by 
 544     // them, the indices are not the same for us and the toolbar 
 546     // first determine the position of the first button to delete: it may be 
 547     // different from pos if we use several separators to cover the space used 
 549     wxToolBarToolsList::compatibility_iterator node
; 
 550     for ( node 
= m_tools
.GetFirst(); node
; node 
= node
->GetNext() ) 
 552         wxToolBarToolBase 
*tool2 
= node
->GetData(); 
 555             // let node point to the next node in the list 
 556             node 
= node
->GetNext(); 
 561         if ( tool2
->IsControl() ) 
 562             pos 
+= ((wxToolBarTool 
*)tool2
)->GetSeparatorsCount() - 1; 
 565     // now determine the number of buttons to delete and the area taken by them 
 566     size_t nButtonsToDelete 
= 1; 
 568     // get the size of the button we're going to delete 
 569     const RECT r 
= wxGetTBItemRect(GetHwnd(), pos
); 
 571     int width 
= r
.right 
- r
.left
; 
 573     if ( tool
->IsControl() ) 
 575         nButtonsToDelete 
= ((wxToolBarTool 
*)tool
)->GetSeparatorsCount(); 
 576         width 
*= nButtonsToDelete
; 
 579     // do delete all buttons 
 580     m_nButtons 
-= nButtonsToDelete
; 
 581     while ( nButtonsToDelete
-- > 0 ) 
 583         if ( !::SendMessage(GetHwnd(), TB_DELETEBUTTON
, pos
, 0) ) 
 585             wxLogLastError(wxT("TB_DELETEBUTTON")); 
 591     // and finally reposition all the controls after this button (the toolbar 
 592     // takes care of all normal items) 
 593     for ( /* node -> first after deleted */ ; node
; node 
= node
->GetNext() ) 
 595         wxToolBarTool 
*tool2 
= (wxToolBarTool
*)node
->GetData(); 
 596         if ( tool2
->IsControl() ) 
 598             tool2
->MoveBy(-width
); 
 602     InvalidateBestSize(); 
 607 void wxToolBar::CreateDisabledImageList() 
 609     wxDELETE(m_disabledImgList
); 
 611     // as we can't use disabled image list with older versions of comctl32.dll, 
 612     // don't even bother creating it 
 613     if ( wxApp::GetComCtl32Version() >= 470 ) 
 615         // search for the first disabled button img in the toolbar, if any 
 616         for ( wxToolBarToolsList::compatibility_iterator
 
 617                 node 
= m_tools
.GetFirst(); node
; node 
= node
->GetNext() ) 
 619             wxToolBarToolBase 
*tool 
= node
->GetData(); 
 620             wxBitmap bmpDisabled 
= tool
->GetDisabledBitmap(); 
 621             if ( bmpDisabled
.IsOk() ) 
 623                 const wxSize sizeBitmap 
= bmpDisabled
.GetSize(); 
 624                 m_disabledImgList 
= new wxImageList
 
 628                                             bmpDisabled
.GetMask() != NULL
, 
 635         // we don't have any disabled bitmaps 
 639 bool wxToolBar::Realize() 
 641     if ( !wxToolBarBase::Realize() ) 
 644     const size_t nTools 
= GetToolsCount(); 
 646 #ifdef wxREMAP_BUTTON_COLOURS 
 647     // don't change the values of these constants, they can be set from the 
 648     // user code via wxSystemOptions 
 657     // the user-specified option overrides anything, but if it wasn't set, only 
 658     // remap the buttons on 8bpp displays as otherwise the bitmaps usually look 
 659     // much worse after remapping 
 660     static const wxChar 
*remapOption 
= wxT("msw.remap"); 
 661     const int remapValue 
= wxSystemOptions::HasOption(remapOption
) 
 662                                 ? wxSystemOptions::GetOptionInt(remapOption
) 
 663                                 : wxDisplayDepth() <= 8 ? Remap_Buttons
 
 666 #endif // wxREMAP_BUTTON_COLOURS 
 668     // delete all old buttons, if any 
 669     for ( size_t pos 
= 0; pos 
< m_nButtons
; pos
++ ) 
 671         if ( !::SendMessage(GetHwnd(), TB_DELETEBUTTON
, 0, 0) ) 
 673             wxLogDebug(wxT("TB_DELETEBUTTON failed")); 
 677     // First, add the bitmap: we use one bitmap for all toolbar buttons 
 678     // ---------------------------------------------------------------- 
 680     wxToolBarToolsList::compatibility_iterator node
; 
 683     if ( !HasFlag(wxTB_NOICONS
) ) 
 685         // if we already have a bitmap, we'll replace the existing one -- 
 686         // otherwise we'll install a new one 
 687         HBITMAP oldToolBarBitmap 
= (HBITMAP
)m_hBitmap
; 
 689         const wxCoord totalBitmapWidth  
= m_defaultWidth 
* 
 690                                           wx_truncate_cast(wxCoord
, nTools
), 
 691                       totalBitmapHeight 
= m_defaultHeight
; 
 693         // Create a bitmap and copy all the tool bitmaps into it 
 694         wxMemoryDC dcAllButtons
; 
 695         wxBitmap 
bitmap(totalBitmapWidth
, totalBitmapHeight
); 
 696         dcAllButtons
.SelectObject(bitmap
); 
 698 #ifdef wxREMAP_BUTTON_COLOURS 
 699         if ( remapValue 
!= Remap_TransparentBg 
) 
 700 #endif // wxREMAP_BUTTON_COLOURS 
 702             // VZ: why do we hardcode grey colour for CE? 
 703             dcAllButtons
.SetBackground(wxBrush( 
 705                                         wxColour(0xc0, 0xc0, 0xc0) 
 706 #else // !__WXWINCE__ 
 707                                         GetBackgroundColour() 
 708 #endif // __WXWINCE__/!__WXWINCE__ 
 710             dcAllButtons
.Clear(); 
 713         m_hBitmap 
= bitmap
.GetHBITMAP(); 
 714         HBITMAP hBitmap 
= (HBITMAP
)m_hBitmap
; 
 716 #ifdef wxREMAP_BUTTON_COLOURS 
 717         if ( remapValue 
== Remap_Bg 
) 
 719             dcAllButtons
.SelectObject(wxNullBitmap
); 
 721             // Even if we're not remapping the bitmap 
 722             // content, we still have to remap the background. 
 723             hBitmap 
= (HBITMAP
)MapBitmap((WXHBITMAP
) hBitmap
, 
 724                 totalBitmapWidth
, totalBitmapHeight
); 
 726             dcAllButtons
.SelectObject(bitmap
); 
 728 #endif // wxREMAP_BUTTON_COLOURS 
 730         // the button position 
 733         // the number of buttons (not separators) 
 736         CreateDisabledImageList(); 
 737         for ( node 
= m_tools
.GetFirst(); node
; node 
= node
->GetNext() ) 
 739             wxToolBarToolBase 
*tool 
= node
->GetData(); 
 740             if ( tool
->IsButton() ) 
 742                 const wxBitmap
& bmp 
= tool
->GetNormalBitmap(); 
 744                 const int w 
= bmp
.GetWidth(); 
 745                 const int h 
= bmp
.GetHeight(); 
 749                     int xOffset 
= wxMax(0, (m_defaultWidth 
- w
)/2); 
 750                     int yOffset 
= wxMax(0, (m_defaultHeight 
- h
)/2); 
 752                     // notice the last parameter: do use mask 
 753                     dcAllButtons
.DrawBitmap(bmp
, x 
+ xOffset
, yOffset
, true); 
 757                     wxFAIL_MSG( wxT("invalid tool button bitmap") ); 
 760                 // also deal with disabled bitmap if we want to use them 
 761                 if ( m_disabledImgList 
) 
 763                     wxBitmap bmpDisabled 
= tool
->GetDisabledBitmap(); 
 764 #if wxUSE_IMAGE && wxUSE_WXDIB 
 765                     if ( !bmpDisabled
.IsOk() ) 
 767                         // no disabled bitmap specified but we still need to 
 768                         // fill the space in the image list with something, so 
 769                         // we grey out the normal bitmap 
 771                           imgGreyed 
= bmp
.ConvertToImage().ConvertToGreyscale(); 
 773 #ifdef wxREMAP_BUTTON_COLOURS 
 774                         if ( remapValue 
== Remap_Buttons 
) 
 776                             // we need to have light grey background colour for 
 777                             // MapBitmap() to work correctly 
 778                             for ( int y 
= 0; y 
< h
; y
++ ) 
 780                                 for ( int x 
= 0; x 
< w
; x
++ ) 
 782                                     if ( imgGreyed
.IsTransparent(x
, y
) ) 
 783                                         imgGreyed
.SetRGB(x
, y
, 
 785                                                          wxLIGHT_GREY
->Green(), 
 786                                                          wxLIGHT_GREY
->Blue()); 
 790 #endif // wxREMAP_BUTTON_COLOURS 
 792                         bmpDisabled 
= wxBitmap(imgGreyed
); 
 794 #endif // wxUSE_IMAGE 
 796 #ifdef wxREMAP_BUTTON_COLOURS 
 797                     if ( remapValue 
== Remap_Buttons 
) 
 798                         MapBitmap(bmpDisabled
.GetHBITMAP(), w
, h
); 
 799 #endif // wxREMAP_BUTTON_COLOURS 
 801                     m_disabledImgList
->Add(bmpDisabled
); 
 804                 // still inc width and number of buttons because otherwise the 
 805                 // subsequent buttons will all be shifted which is rather confusing 
 806                 // (and like this you'd see immediately which bitmap was bad) 
 812         dcAllButtons
.SelectObject(wxNullBitmap
); 
 814         // don't delete this HBITMAP! 
 815         bitmap
.SetHBITMAP(0); 
 817 #ifdef wxREMAP_BUTTON_COLOURS 
 818         if ( remapValue 
== Remap_Buttons 
) 
 820             // Map to system colours 
 821             hBitmap 
= (HBITMAP
)MapBitmap((WXHBITMAP
) hBitmap
, 
 822                                          totalBitmapWidth
, totalBitmapHeight
); 
 824 #endif // wxREMAP_BUTTON_COLOURS 
 826         bool addBitmap 
= true; 
 828         if ( oldToolBarBitmap 
) 
 830 #ifdef TB_REPLACEBITMAP 
 831             if ( wxApp::GetComCtl32Version() >= 400 ) 
 833                 TBREPLACEBITMAP replaceBitmap
; 
 834                 replaceBitmap
.hInstOld 
= NULL
; 
 835                 replaceBitmap
.hInstNew 
= NULL
; 
 836                 replaceBitmap
.nIDOld 
= (UINT_PTR
)oldToolBarBitmap
; 
 837                 replaceBitmap
.nIDNew 
= (UINT_PTR
)hBitmap
; 
 838                 replaceBitmap
.nButtons 
= nButtons
; 
 839                 if ( !::SendMessage(GetHwnd(), TB_REPLACEBITMAP
, 
 840                                     0, (LPARAM
) &replaceBitmap
) ) 
 842                     wxFAIL_MSG(wxT("Could not replace the old bitmap")); 
 845                 ::DeleteObject(oldToolBarBitmap
); 
 851 #endif // TB_REPLACEBITMAP 
 853                 // we can't replace the old bitmap, so we will add another one 
 854                 // (awfully inefficient, but what else to do?) and shift the bitmap 
 855                 // indices accordingly 
 858                 bitmapId 
= m_nButtons
; 
 862         if ( addBitmap 
) // no old bitmap or we can't replace it 
 864             TBADDBITMAP addBitmap
; 
 866             addBitmap
.nID 
= (UINT_PTR
)hBitmap
; 
 867             if ( ::SendMessage(GetHwnd(), TB_ADDBITMAP
, 
 868                                (WPARAM
) nButtons
, (LPARAM
)&addBitmap
) == -1 ) 
 870                 wxFAIL_MSG(wxT("Could not add bitmap to toolbar")); 
 874         // disable image lists are only supported in comctl32.dll 4.70+ 
 875         if ( wxApp::GetComCtl32Version() >= 470 ) 
 877             HIMAGELIST hil 
= m_disabledImgList
 
 878                                 ? GetHimagelistOf(m_disabledImgList
) 
 881             // notice that we set the image list even if don't have one right 
 882             // now as we could have it before and need to reset it in this case 
 883             HIMAGELIST oldImageList 
= (HIMAGELIST
) 
 884               ::SendMessage(GetHwnd(), TB_SETDISABLEDIMAGELIST
, 0, (LPARAM
)hil
); 
 886             // delete previous image list if any 
 888                 ::DeleteObject(oldImageList
); 
 893     // Next add the buttons and separators 
 894     // ----------------------------------- 
 896     wxScopedArray
<TBBUTTON
> buttons(new TBBUTTON
[nTools
]); 
 898     // this array will hold the indices of all controls in the toolbar 
 899     wxArrayInt controlIds
; 
 901     bool lastWasRadio 
= false; 
 903     for ( node 
= m_tools
.GetFirst(); node
; node 
= node
->GetNext() ) 
 905         wxToolBarTool 
*tool 
= static_cast<wxToolBarTool 
*>(node
->GetData()); 
 907         // don't add separators to the vertical toolbar with old comctl32.dll 
 908         // versions as they didn't handle this properly 
 909         if ( IsVertical() && tool
->IsSeparator() && 
 910                 wxApp::GetComCtl32Version() <= 472 ) 
 915         TBBUTTON
& button 
= buttons
[i
]; 
 917         wxZeroMemory(button
); 
 919         bool isRadio 
= false; 
 920         switch ( tool
->GetStyle() ) 
 922             case wxTOOL_STYLE_CONTROL
: 
 923             case wxTOOL_STYLE_SEPARATOR
: 
 924                 if ( tool
->IsStretchableSpace() ) 
 926                     // we're going to modify the size of this button later and 
 927                     // so we need a valid id for it and not wxID_SEPARATOR 
 928                     // which is used by spacers by default 
 929                     tool
->AllocSpacerId(); 
 931                     // also set the number of separators so that the logic in 
 932                     // HandlePaint() works correctly 
 933                     tool
->SetSeparatorsCount(1); 
 936                 button
.idCommand 
= tool
->GetId(); 
 937                 button
.fsState 
= TBSTATE_ENABLED
; 
 938                 button
.fsStyle 
= TBSTYLE_SEP
; 
 941             case wxTOOL_STYLE_BUTTON
: 
 942                 if ( !HasFlag(wxTB_NOICONS
) ) 
 943                     button
.iBitmap 
= bitmapId
; 
 945                 if ( HasFlag(wxTB_TEXT
) ) 
 947                     const wxString
& label 
= tool
->GetLabel(); 
 948                     if ( !label
.empty() ) 
 949                         button
.iString 
= (INT_PTR
)label
.wx_str(); 
 952                 button
.idCommand 
= tool
->GetId(); 
 954                 if ( tool
->IsEnabled() ) 
 955                     button
.fsState 
|= TBSTATE_ENABLED
; 
 956                 if ( tool
->IsToggled() ) 
 957                     button
.fsState 
|= TBSTATE_CHECKED
; 
 959                 switch ( tool
->GetKind() ) 
 962                         button
.fsStyle 
= TBSTYLE_CHECKGROUP
; 
 966                             // the first item in the radio group is checked by 
 967                             // default to be consistent with wxGTK and the menu 
 969                             button
.fsState 
|= TBSTATE_CHECKED
; 
 971                             if (tool
->Toggle(true)) 
 973                                 DoToggleTool(tool
, true); 
 976                         else if ( tool
->IsToggled() ) 
 978                             wxToolBarToolsList::compatibility_iterator nodePrev 
= node
->GetPrevious(); 
 979                             int prevIndex 
= i 
- 1; 
 982                                 TBBUTTON
& prevButton 
= buttons
[prevIndex
]; 
 983                                 wxToolBarToolBase 
*tool 
= nodePrev
->GetData(); 
 984                                 if ( !tool
->IsButton() || tool
->GetKind() != wxITEM_RADIO 
) 
 987                                 if ( tool
->Toggle(false) ) 
 988                                     DoToggleTool(tool
, false); 
 990                                 prevButton
.fsState 
&= ~TBSTATE_CHECKED
; 
 991                                 nodePrev 
= nodePrev
->GetPrevious(); 
1000                         button
.fsStyle 
= TBSTYLE_CHECK
; 
1004                         button
.fsStyle 
= TBSTYLE_BUTTON
; 
1007                    case wxITEM_DROPDOWN
: 
1008                         button
.fsStyle 
= TBSTYLE_DROPDOWN
; 
1012                         wxFAIL_MSG( wxT("unexpected toolbar button kind") ); 
1013                         button
.fsStyle 
= TBSTYLE_BUTTON
; 
1021         lastWasRadio 
= isRadio
; 
1026     if ( !::SendMessage(GetHwnd(), TB_ADDBUTTONS
, i
, (LPARAM
)buttons
.get()) ) 
1028         wxLogLastError(wxT("TB_ADDBUTTONS")); 
1032     // Adjust controls and stretchable spaces 
1033     // -------------------------------------- 
1035     // adjust the controls size to fit nicely in the toolbar and compute its 
1036     // total size while doing it 
1037     m_totalFixedSize 
= 0; 
1039     for ( node 
= m_tools
.GetFirst(); node
; node 
= node
->GetNext(), toolIndex
++ ) 
1041         wxToolBarTool 
* const tool 
= (wxToolBarTool
*)node
->GetData(); 
1043         const RECT r 
= wxGetTBItemRect(GetHwnd(), toolIndex
); 
1045         if ( !tool
->IsControl() ) 
1048                 m_totalFixedSize 
+= r
.bottom 
- r
.top
; 
1050                 m_totalFixedSize 
+= r
.right 
- r
.left
; 
1057             // don't embed controls in the vertical toolbar, this doesn't look 
1058             // good and wxGTK doesn't do it neither (and the code below can't 
1059             // deal with this case) 
1063         wxControl 
* const control 
= tool
->GetControl(); 
1064         wxStaticText 
* const staticText 
= tool
->GetStaticText(); 
1066         wxSize size 
= control
->GetSize(); 
1067         wxSize staticTextSize
; 
1070             staticTextSize 
= staticText
->GetSize(); 
1071             staticTextSize
.y 
+= 3; // margin between control and its label 
1074         // TB_SETBUTTONINFO message is only supported by comctl32.dll 4.71+ 
1075 #ifdef TB_SETBUTTONINFO 
1076         // available in headers, now check whether it is available now 
1077         // (during run-time) 
1078         if ( wxApp::GetComCtl32Version() >= 471 ) 
1080             // set the (underlying) separators width to be that of the 
1083             tbbi
.cbSize 
= sizeof(tbbi
); 
1084             tbbi
.dwMask 
= TBIF_SIZE
; 
1085             tbbi
.cx 
= (WORD
)size
.x
; 
1086             if ( !::SendMessage(GetHwnd(), TB_SETBUTTONINFO
, 
1087                                 tool
->GetId(), (LPARAM
)&tbbi
) ) 
1089                 // the id is probably invalid? 
1090                 wxLogLastError(wxT("TB_SETBUTTONINFO")); 
1094 #endif // comctl32.dll 4.71 
1095         // TB_SETBUTTONINFO unavailable 
1097             // try adding several separators to fit the controls width 
1098             int widthSep 
= r
.right 
- r
.left
; 
1103             tbb
.fsState 
= TBSTATE_ENABLED
; 
1104             tbb
.fsStyle 
= TBSTYLE_SEP
; 
1106             size_t nSeparators 
= size
.x 
/ widthSep
; 
1107             for ( size_t nSep 
= 0; nSep 
< nSeparators
; nSep
++ ) 
1109                 if ( !::SendMessage(GetHwnd(), TB_INSERTBUTTON
, 
1110                                     toolIndex
, (LPARAM
)&tbb
) ) 
1112                     wxLogLastError(wxT("TB_INSERTBUTTON")); 
1118             // remember the number of separators we used - we'd have to 
1119             // delete all of them later 
1120             tool
->SetSeparatorsCount(nSeparators
); 
1122             // adjust the controls width to exactly cover the separators 
1123             size
.x 
= (nSeparators 
+ 1)*widthSep
; 
1124             control
->SetSize(size
.x
, wxDefaultCoord
); 
1127         // position the control itself correctly vertically centering it on the 
1128         // icon area of the toolbar 
1129         int height 
= r
.bottom 
- r
.top 
- staticTextSize
.y
; 
1131         int diff 
= height 
- size
.y
; 
1132         if ( diff 
< 0 || !HasFlag(wxTB_TEXT
) ) 
1134             // not enough room for the static text 
1138             // recalculate height & diff without the staticText control 
1139             height 
= r
.bottom 
- r
.top
; 
1140             diff 
= height 
- size
.y
; 
1143                 // the control is too high, resize to fit 
1144                 control
->SetSize(wxDefaultCoord
, height 
- 2); 
1149         else // enough space for both the control and the label 
1155         control
->Move(r
.left
, r
.top 
+ (diff 
+ 1) / 2); 
1158             staticText
->Move(r
.left 
+ (size
.x 
- staticTextSize
.x
)/2, 
1159                              r
.bottom 
- staticTextSize
.y
); 
1162         m_totalFixedSize 
+= size
.x
; 
1165     // the max index is the "real" number of buttons - i.e. counting even the 
1166     // separators which we added just for aligning the controls 
1167     m_nButtons 
= toolIndex
; 
1169     if ( !IsVertical() ) 
1171         if ( m_maxRows 
== 0 ) 
1172             // if not set yet, only one row 
1175     else if ( m_nButtons 
> 0 ) // vertical non empty toolbar 
1177         // if not set yet, have one column 
1179         SetRows(m_nButtons
); 
1182     InvalidateBestSize(); 
1188 void wxToolBar::UpdateStretchableSpacersSize() 
1190 #ifdef TB_SETBUTTONINFO 
1191     // we can't resize the spacers if TB_SETBUTTONINFO is not supported (we 
1192     // could try to do it with multiple separators as for the controls but this 
1193     // is too painful and it just doesn't seem to be worth doing for the 
1195     if ( wxApp::GetComCtl32Version() < 471 ) 
1198     // check if we have any stretchable spacers in the first place 
1199     unsigned numSpaces 
= 0; 
1200     wxToolBarToolsList::compatibility_iterator node
; 
1201     for ( node 
= m_tools
.GetFirst(); node
; node 
= node
->GetNext() ) 
1203         wxToolBarTool 
* const tool 
= (wxToolBarTool
*)node
->GetData(); 
1204         if ( tool
->IsStretchableSpace() ) 
1211     // we do, adjust their size: either distribute the extra size among them or 
1212     // reduce their size if there is not enough place for all tools 
1213     const int totalSize 
= IsVertical() ? GetClientSize().y 
: GetClientSize().x
; 
1214     const int extraSize 
= totalSize 
- m_totalFixedSize
; 
1215     const int sizeSpacer 
= extraSize 
> 0 ? extraSize 
/ numSpaces 
: 1; 
1217     // the last spacer should consume all remaining space if we have too much 
1218     // of it (which can be greater than sizeSpacer because of the rounding) 
1219     const int sizeLastSpacer 
= extraSize 
> 0 
1220                                 ? extraSize 
- (numSpaces 
- 1)*sizeSpacer
 
1223     // cumulated offset by which we need to move all the following controls to 
1224     // the right: while the toolbar takes care of the normal items, we must 
1225     // move the controls manually ourselves to ensure they remain at the 
1229     for ( node 
= m_tools
.GetFirst(); node
; node 
= node
->GetNext(), toolIndex
++ ) 
1231         wxToolBarTool 
* const tool 
= (wxToolBarTool
*)node
->GetData(); 
1233         if ( tool
->IsControl() && offset 
) 
1235             tool
->MoveBy(offset
); 
1240         if ( !tool
->IsStretchableSpace() ) 
1243         const RECT rcOld 
= wxGetTBItemRect(GetHwnd(), toolIndex
); 
1245         WinStruct
<TBBUTTONINFO
> tbbi
; 
1246         tbbi
.dwMask 
= TBIF_SIZE
; 
1247         tbbi
.cx 
= --numSpaces 
? sizeSpacer 
: sizeLastSpacer
; 
1249         if ( !::SendMessage(GetHwnd(), TB_SETBUTTONINFO
, 
1250                             tool
->GetId(), (LPARAM
)&tbbi
) ) 
1252             wxLogLastError(wxT("TB_SETBUTTONINFO")); 
1256             // we successfully resized this one, move all the controls after it 
1257             // by the corresponding amount (may be positive or negative) 
1258             offset 
+= tbbi
.cx 
- (rcOld
.right 
- rcOld
.left
); 
1261 #endif // TB_SETBUTTONINFO 
1264 // ---------------------------------------------------------------------------- 
1266 // ---------------------------------------------------------------------------- 
1268 bool wxToolBar::MSWCommand(WXUINT 
WXUNUSED(cmd
), WXWORD id_
) 
1270     // cast to signed is important as we compare this id with (signed) ints in 
1271     // FindById() and without the cast we'd get a positive int from a 
1272     // "negative" (i.e. > 32767) WORD 
1273     const int id 
= (signed short)id_
; 
1275     wxToolBarToolBase 
*tool 
= FindById(id
); 
1279     bool toggled 
= false; // just to suppress warnings 
1281     LRESULT state 
= ::SendMessage(GetHwnd(), TB_GETSTATE
, id
, 0); 
1283     if ( tool
->CanBeToggled() ) 
1285         toggled 
= (state 
& TBSTATE_CHECKED
) != 0; 
1287         // ignore the event when a radio button is released, as this doesn't 
1288         // seem to happen at all, and is handled otherwise 
1289         if ( tool
->GetKind() == wxITEM_RADIO 
&& !toggled 
) 
1292         tool
->Toggle(toggled
); 
1293         UnToggleRadioGroup(tool
); 
1296     // Without the two lines of code below, if the toolbar was repainted during 
1297     // OnLeftClick(), then it could end up without the tool bitmap temporarily 
1298     // (see http://lists.nongnu.org/archive/html/lmi/2008-10/msg00014.html). 
1299     // The Update() call below ensures that this won't happen, by repainting 
1300     // invalidated areas of the toolbar immediately. 
1302     // To complicate matters, the tool would be drawn in depressed state (this 
1303     // code is called when mouse button is released, not pressed). That's not 
1304     // ideal, having the tool pressed for the duration of OnLeftClick() 
1305     // provides the user with useful visual clue that the app is busy reacting 
1306     // to the event. So we manually put the tool into pressed state, handle the 
1307     // event and then finally restore tool's original state. 
1308     ::SendMessage(GetHwnd(), TB_SETSTATE
, id
, MAKELONG(state 
| TBSTATE_PRESSED
, 0)); 
1311     bool allowLeftClick 
= OnLeftClick(id
, toggled
); 
1313     // Restore the unpressed state. Enabled/toggled state might have been 
1314     // changed since so take care of it. 
1315     if (tool
->IsEnabled()) 
1316         state 
|= TBSTATE_ENABLED
; 
1318         state 
&= ~TBSTATE_ENABLED
; 
1319     if (tool
->IsToggled()) 
1320         state 
|= TBSTATE_CHECKED
; 
1322         state 
&= ~TBSTATE_CHECKED
; 
1323     ::SendMessage(GetHwnd(), TB_SETSTATE
, id
, MAKELONG(state
, 0)); 
1325     // OnLeftClick() can veto the button state change - for buttons which 
1326     // may be toggled only, of course. 
1327     if ( !allowLeftClick 
&& tool
->CanBeToggled() ) 
1330         tool
->Toggle(!toggled
); 
1332         ::SendMessage(GetHwnd(), TB_CHECKBUTTON
, id
, MAKELONG(!toggled
, 0)); 
1338 bool wxToolBar::MSWOnNotify(int WXUNUSED(idCtrl
), 
1340                             WXLPARAM 
*WXUNUSED(result
)) 
1342     LPNMHDR hdr 
= (LPNMHDR
)lParam
; 
1343     if ( hdr
->code 
== TBN_DROPDOWN 
) 
1345         LPNMTOOLBAR tbhdr 
= (LPNMTOOLBAR
)lParam
; 
1347         wxCommandEvent 
evt(wxEVT_COMMAND_TOOL_DROPDOWN_CLICKED
, tbhdr
->iItem
); 
1348         if ( HandleWindowEvent(evt
) ) 
1350             // Event got handled, don't display default popup menu 
1354         const wxToolBarToolBase 
* const tool 
= FindById(tbhdr
->iItem
); 
1355         wxCHECK_MSG( tool
, false, wxT("drop down message for unknown tool") ); 
1357         wxMenu 
* const menu 
= tool
->GetDropdownMenu(); 
1361         // Display popup menu below button 
1362         const RECT r 
= wxGetTBItemRect(GetHwnd(), GetToolPos(tbhdr
->iItem
)); 
1364             PopupMenu(menu
, r
.left
, r
.bottom
); 
1370     if( !HasFlag(wxTB_NO_TOOLTIPS
) ) 
1373         // First check if this applies to us 
1375         // the tooltips control created by the toolbar is sometimes Unicode, even 
1376         // in an ANSI application - this seems to be a bug in comctl32.dll v5 
1377         UINT code 
= hdr
->code
; 
1378         if ( (code 
!= (UINT
) TTN_NEEDTEXTA
) && (code 
!= (UINT
) TTN_NEEDTEXTW
) ) 
1381         HWND toolTipWnd 
= (HWND
)::SendMessage(GetHwnd(), TB_GETTOOLTIPS
, 0, 0); 
1382         if ( toolTipWnd 
!= hdr
->hwndFrom 
) 
1385         LPTOOLTIPTEXT ttText 
= (LPTOOLTIPTEXT
)lParam
; 
1386         int id 
= (int)ttText
->hdr
.idFrom
; 
1388         wxToolBarToolBase 
*tool 
= FindById(id
); 
1390             return HandleTooltipNotify(code
, lParam
, tool
->GetShortHelp()); 
1392         wxUnusedVar(lParam
); 
1399 // ---------------------------------------------------------------------------- 
1401 // ---------------------------------------------------------------------------- 
1403 void wxToolBar::SetToolBitmapSize(const wxSize
& size
) 
1405     wxToolBarBase::SetToolBitmapSize(size
); 
1407     ::SendMessage(GetHwnd(), TB_SETBITMAPSIZE
, 0, MAKELONG(size
.x
, size
.y
)); 
1410 void wxToolBar::SetRows(int nRows
) 
1412     if ( nRows 
== m_maxRows 
) 
1414         // avoid resizing the frame uselessly 
1418     // TRUE in wParam means to create at least as many rows, FALSE - 
1421     ::SendMessage(GetHwnd(), TB_SETROWS
, 
1422                   MAKEWPARAM(nRows
, !(GetWindowStyle() & wxTB_VERTICAL
)), 
1430 // The button size is bigger than the bitmap size 
1431 wxSize 
wxToolBar::GetToolSize() const 
1433     // TB_GETBUTTONSIZE is supported from version 4.70 
1434 #if defined(_WIN32_IE) && (_WIN32_IE >= 0x300 ) \ 
1435     && !( defined(__GNUWIN32__) && !wxCHECK_W32API_VERSION( 1, 0 ) ) \ 
1436     && !defined (__DIGITALMARS__) 
1437     if ( wxApp::GetComCtl32Version() >= 470 ) 
1439         DWORD dw 
= ::SendMessage(GetHwnd(), TB_GETBUTTONSIZE
, 0, 0); 
1441         return wxSize(LOWORD(dw
), HIWORD(dw
)); 
1444 #endif // comctl32.dll 4.70+ 
1447         return wxSize(m_defaultWidth 
+ 8, m_defaultHeight 
+ 7); 
1452 wxToolBarToolBase 
*GetItemSkippingDummySpacers(const wxToolBarToolsList
& tools
, 
1455     wxToolBarToolsList::compatibility_iterator current 
= tools
.GetFirst(); 
1457     for ( ; current 
; current 
= current
->GetNext() ) 
1460             return current
->GetData(); 
1462         wxToolBarTool 
*tool 
= (wxToolBarTool 
*)current
->GetData(); 
1463         size_t separators 
= tool
->GetSeparatorsCount(); 
1465         // if it is a normal button, sepcount == 0, so skip 1 item (the button) 
1466         // otherwise, skip as many items as the separator count, plus the 
1468         index 
-= separators 
? separators 
+ 1 : 1; 
1474 wxToolBarToolBase 
*wxToolBar::FindToolForPosition(wxCoord x
, wxCoord y
) const 
1479     int index 
= (int)::SendMessage(GetHwnd(), TB_HITTEST
, 0, (LPARAM
)&pt
); 
1481     // MBN: when the point ( x, y ) is close to the toolbar border 
1482     //      TB_HITTEST returns m_nButtons ( not -1 ) 
1483     if ( index 
< 0 || (size_t)index 
>= m_nButtons 
) 
1484         // it's a separator or there is no tool at all there 
1487     // when TB_SETBUTTONINFO is available (both during compile- and run-time), 
1488     // we don't use the dummy separators hack 
1489 #ifdef TB_SETBUTTONINFO 
1490     if ( wxApp::GetComCtl32Version() >= 471 ) 
1492         return m_tools
.Item((size_t)index
)->GetData(); 
1495 #endif // TB_SETBUTTONINFO 
1497         return GetItemSkippingDummySpacers( m_tools
, (size_t) index 
); 
1501 void wxToolBar::UpdateSize() 
1503     wxPoint pos 
= GetPosition(); 
1504     ::SendMessage(GetHwnd(), TB_AUTOSIZE
, 0, 0); 
1505     if (pos 
!= GetPosition()) 
1508     // In case Realize is called after the initial display (IOW the programmer 
1509     // may have rebuilt the toolbar) give the frame the option of resizing the 
1510     // toolbar to full width again, but only if the parent is a frame and the 
1511     // toolbar is managed by the frame.  Otherwise assume that some other 
1512     // layout mechanism is controlling the toolbar size and leave it alone. 
1513     SendSizeEventToParent(); 
1516 // ---------------------------------------------------------------------------- 
1518 // --------------------------------------------------------------------------- 
1520 // get the TBSTYLE of the given toolbar window 
1521 long wxToolBar::GetMSWToolbarStyle() const 
1523     return ::SendMessage(GetHwnd(), TB_GETSTYLE
, 0, 0L); 
1526 void wxToolBar::SetWindowStyleFlag(long style
) 
1528     // the style bits whose changes force us to recreate the toolbar 
1529     static const long MASK_NEEDS_RECREATE 
= wxTB_TEXT 
| wxTB_NOICONS
; 
1531     const long styleOld 
= GetWindowStyle(); 
1533     wxToolBarBase::SetWindowStyleFlag(style
); 
1535     // don't recreate an empty toolbar: not only this is unnecessary, but it is 
1536     // also fatal as we'd then try to recreate the toolbar when it's just being 
1538     if ( GetToolsCount() && 
1539             (style 
& MASK_NEEDS_RECREATE
) != (styleOld 
& MASK_NEEDS_RECREATE
) ) 
1541         // to remove the text labels, simply re-realizing the toolbar is enough 
1542         // but I don't know of any way to add the text to an existing toolbar 
1543         // other than by recreating it entirely 
1548 // ---------------------------------------------------------------------------- 
1550 // ---------------------------------------------------------------------------- 
1552 void wxToolBar::DoEnableTool(wxToolBarToolBase 
*tool
, bool enable
) 
1554     ::SendMessage(GetHwnd(), TB_ENABLEBUTTON
, 
1555                   (WPARAM
)tool
->GetId(), (LPARAM
)MAKELONG(enable
, 0)); 
1558 void wxToolBar::DoToggleTool(wxToolBarToolBase 
*tool
, bool toggle
) 
1560     ::SendMessage(GetHwnd(), TB_CHECKBUTTON
, 
1561                   (WPARAM
)tool
->GetId(), (LPARAM
)MAKELONG(toggle
, 0)); 
1564 void wxToolBar::DoSetToggle(wxToolBarToolBase 
*WXUNUSED(tool
), bool WXUNUSED(toggle
)) 
1566     // VZ: AFAIK, the button has to be created either with TBSTYLE_CHECK or 
1567     //     without, so we really need to delete the button and recreate it here 
1568     wxFAIL_MSG( wxT("not implemented") ); 
1571 void wxToolBar::SetToolNormalBitmap( int id
, const wxBitmap
& bitmap 
) 
1573     wxToolBarTool
* tool 
= static_cast<wxToolBarTool
*>(FindById(id
)); 
1576         wxCHECK_RET( tool
->IsButton(), wxT("Can only set bitmap on button tools.")); 
1578         tool
->SetNormalBitmap(bitmap
); 
1583 void wxToolBar::SetToolDisabledBitmap( int id
, const wxBitmap
& bitmap 
) 
1585     wxToolBarTool
* tool 
= static_cast<wxToolBarTool
*>(FindById(id
)); 
1588         wxCHECK_RET( tool
->IsButton(), wxT("Can only set bitmap on button tools.")); 
1590         tool
->SetDisabledBitmap(bitmap
); 
1595 // ---------------------------------------------------------------------------- 
1597 // ---------------------------------------------------------------------------- 
1599 // Responds to colour changes, and passes event on to children. 
1600 void wxToolBar::OnSysColourChanged(wxSysColourChangedEvent
& event
) 
1602     wxRGBToColour(m_backgroundColour
, ::GetSysColor(COLOR_BTNFACE
)); 
1604     // Remap the buttons 
1607     // Relayout the toolbar 
1608     int nrows 
= m_maxRows
; 
1609     m_maxRows 
= 0;      // otherwise SetRows() wouldn't do anything 
1614     // let the event propagate further 
1618 void wxToolBar::OnMouseEvent(wxMouseEvent
& event
) 
1620     if ( event
.Leaving() ) 
1624             OnMouseEnter(wxID_ANY
); 
1632     if ( event
.RightDown() ) 
1634         // find the tool under the mouse 
1635         wxCoord x 
= 0, y 
= 0; 
1636         event
.GetPosition(&x
, &y
); 
1638         wxToolBarToolBase 
*tool 
= FindToolForPosition(x
, y
); 
1639         OnRightClick(tool 
? tool
->GetId() : -1, x
, y
); 
1647 bool wxToolBar::HandleSize(WXWPARAM 
WXUNUSED(wParam
), WXLPARAM lParam
) 
1649     // wait until we have some tools 
1650     if ( !GetToolsCount() ) 
1653     // calculate our minor dimension ourselves - we're confusing the standard 
1654     // logic (TB_AUTOSIZE) with our horizontal toolbars and other hacks 
1655     const RECT r 
= wxGetTBItemRect(GetHwnd(), 0); 
1663         w 
= r
.right 
- r
.left
; 
1666             w 
*= (m_nButtons 
+ m_maxRows 
- 1)/m_maxRows
; 
1673         if (HasFlag( wxTB_FLAT 
)) 
1674             h 
= r
.bottom 
- r
.top 
- 3; 
1676             h 
= r
.bottom 
- r
.top
; 
1679             // FIXME: hardcoded separator line height... 
1680             h 
+= HasFlag(wxTB_NODIVIDER
) ? 4 : 6; 
1685     if ( MAKELPARAM(w
, h
) != lParam 
) 
1687         // size really changed 
1691     UpdateStretchableSpacersSize(); 
1693     // message processed 
1697 #ifdef wxHAS_MSW_BACKGROUND_ERASE_HOOK 
1699 bool wxToolBar::HandlePaint(WXWPARAM wParam
, WXLPARAM lParam
) 
1701     // we must prevent the dummy separators corresponding to controls or 
1702     // stretchable spaces from being seen: we used to do it by painting over 
1703     // them but this, unsurprisingly, resulted in a lot of flicker so now we 
1704     // prevent the toolbar from painting them at all 
1706     // compute the region containing all dummy separators which we don't want 
1708     wxRegion rgnDummySeps
; 
1709     const wxRect rectTotal 
= GetClientRect(); 
1711     for ( wxToolBarToolsList::compatibility_iterator node 
= m_tools
.GetFirst(); 
1713           node 
= node
->GetNext() ) 
1715         wxToolBarTool 
* const 
1716             tool 
= static_cast<wxToolBarTool 
*>(node
->GetData()); 
1718         if ( tool
->IsControl() || tool
->IsStretchableSpace() ) 
1720             const size_t numSeps 
= tool
->GetSeparatorsCount(); 
1721             for ( size_t n 
= 0; n 
< numSeps
; n
++, toolIndex
++ ) 
1723                 // for some reason TB_GETITEMRECT returns a rectangle 1 pixel 
1724                 // shorter than the full window size (at least under Windows 7) 
1725                 // but we need to erase the full width/height below 
1726                 RECT rcItem 
= wxGetTBItemRect(GetHwnd(), toolIndex
); 
1730                     rcItem
.right 
= rectTotal
.width
; 
1735                     rcItem
.bottom 
= rectTotal
.height
; 
1738                 rgnDummySeps
.Union(wxRectFromRECT(rcItem
)); 
1743             // normal tools never correspond to more than one native button 
1748     if ( rgnDummySeps
.IsOk() ) 
1750         // exclude the area occupied by the controls and stretchable spaces 
1751         // from the update region to prevent the toolbar from drawing 
1753         if ( !::ValidateRgn(GetHwnd(), GetHrgnOf(rgnDummySeps
)) ) 
1755             wxLogLastError(wxT("ValidateRgn()")); 
1759     // still let the native control draw everything else normally but set up a 
1760     // hook to be able to process the next WM_ERASEBKGND sent to our parent 
1761     // because toolbar will ask it to erase its background from its WM_PAINT 
1762     // handler (when using TBSTYLE_TRANSPARENT which we do always use) 
1764     // installing hook is not completely trivial as all kinds of strange 
1765     // situations are possible: sometimes we can be called recursively from 
1766     // inside the native toolbar WM_PAINT handler so the hook might already be 
1767     // installed and sometimes the native toolbar might not send WM_ERASEBKGND 
1768     // to the parent at all for whatever reason, so deal with all these cases 
1769     wxWindow 
* const parent 
= GetParent(); 
1770     const bool hadHook 
= parent
->MSWHasEraseBgHook(); 
1772         GetParent()->MSWSetEraseBgHook(this); 
1774     MSWDefWindowProc(WM_PAINT
, wParam
, lParam
); 
1777         GetParent()->MSWSetEraseBgHook(NULL
); 
1780     if ( rgnDummySeps
.IsOk() ) 
1782         // erase the dummy separators region ourselves now as nobody painted 
1784         WindowHDC 
hdc(GetHwnd()); 
1785         ::SelectClipRgn(hdc
, GetHrgnOf(rgnDummySeps
)); 
1786         MSWDoEraseBackground(hdc
); 
1792 WXHBRUSH 
wxToolBar::MSWGetToolbarBgBrush() 
1794     // we conservatively use a solid brush here but we could also use a themed 
1795     // brush by using DrawThemeBackground() to create a bitmap brush (it'd need 
1796     // to be invalidated whenever the toolbar is resized and, also, correctly 
1797     // aligned using SetBrushOrgEx() before each use -- there is code for doing 
1798     // this in wxNotebook already so it'd need to be refactored into wxWindow) 
1800     // however inasmuch as there is a default background for the toolbar at all 
1801     // (and this is not a trivial question as different applications use very 
1802     // different colours), it seems to be a solid one and using REBAR 
1803     // background brush as we used to do before doesn't look good at all under 
1804     // Windows 7 (and probably Vista too), so for now we just keep it simple 
1806         colBg 
= m_hasBgCol 
? GetBackgroundColour() 
1807                            : wxSystemSettings::GetColour(wxSYS_COLOUR_BTNFACE
); 
1809         brush 
= wxTheBrushList
->FindOrCreateBrush(colBg
); 
1811     return brush 
? static_cast<WXHBRUSH
>(brush
->GetResourceHandle()) : 0; 
1814 WXHBRUSH 
wxToolBar::MSWGetBgBrushForChild(WXHDC hDC
, wxWindowMSW 
*child
) 
1816     WXHBRUSH hbr 
= wxToolBarBase::MSWGetBgBrushForChild(hDC
, child
); 
1820     // the base class version only returns a brush for erasing children 
1821     // background if we have a non-default background colour but as the toolbar 
1822     // doesn't erase its own background by default, we need to always do it for 
1823     // (semi-)transparent children 
1824     if ( child
->GetParent() == this && child
->HasTransparentBackground() ) 
1825         return MSWGetToolbarBgBrush(); 
1830 void wxToolBar::MSWDoEraseBackground(WXHDC hDC
) 
1832     wxFillRect(GetHwnd(), (HDC
)hDC
, (HBRUSH
)MSWGetToolbarBgBrush()); 
1835 bool wxToolBar::MSWEraseBgHook(WXHDC hDC
) 
1837     // toolbar WM_PAINT handler offsets the DC origin before sending 
1838     // WM_ERASEBKGND to the parent but as we handle it in the toolbar itself, 
1839     // we need to reset it back 
1842     if ( !::SetWindowOrgEx(hdc
, 0, 0, &ptOldOrg
) ) 
1844         wxLogLastError(wxT("SetWindowOrgEx(tbar-bg-hdc)")); 
1848     MSWDoEraseBackground(hDC
); 
1850     ::SetWindowOrgEx(hdc
, ptOldOrg
.x
, ptOldOrg
.y
, NULL
); 
1855 #endif // wxHAS_MSW_BACKGROUND_ERASE_HOOK 
1857 void wxToolBar::HandleMouseMove(WXWPARAM 
WXUNUSED(wParam
), WXLPARAM lParam
) 
1859     wxCoord x 
= GET_X_LPARAM(lParam
), 
1860             y 
= GET_Y_LPARAM(lParam
); 
1861     wxToolBarToolBase
* tool 
= FindToolForPosition( x
, y 
); 
1863     // has the current tool changed? 
1864     if ( tool 
!= m_pInTool 
) 
1867         OnMouseEnter(tool 
? tool
->GetId() : wxID_ANY
); 
1871 WXLRESULT 
wxToolBar::MSWWindowProc(WXUINT nMsg
, WXWPARAM wParam
, WXLPARAM lParam
) 
1876             // we don't handle mouse moves, so always pass the message to 
1877             // wxControl::MSWWindowProc (HandleMouseMove just calls OnMouseEnter) 
1878             HandleMouseMove(wParam
, lParam
); 
1882             if ( HandleSize(wParam
, lParam
) ) 
1886 #ifdef wxHAS_MSW_BACKGROUND_ERASE_HOOK 
1888             // refreshing the controls in the toolbar inside a composite window 
1889             // results in an endless stream of WM_PAINT messages -- and seems 
1890             // to be unnecessary anyhow as everything works just fine without 
1891             // any special workarounds in this case 
1892             if ( !IsDoubleBuffered() && HandlePaint(wParam
, lParam
) ) 
1895 #endif // wxHAS_MSW_BACKGROUND_ERASE_HOOK 
1898     return wxControl::MSWWindowProc(nMsg
, wParam
, lParam
); 
1901 // ---------------------------------------------------------------------------- 
1902 // private functions 
1903 // ---------------------------------------------------------------------------- 
1905 #ifdef wxREMAP_BUTTON_COLOURS 
1907 WXHBITMAP 
wxToolBar::MapBitmap(WXHBITMAP bitmap
, int width
, int height
) 
1913         wxLogLastError(wxT("CreateCompatibleDC")); 
1918     SelectInHDC 
bmpInHDC(hdcMem
, (HBITMAP
)bitmap
); 
1922         wxLogLastError(wxT("SelectObject")); 
1927     wxCOLORMAP 
*cmap 
= wxGetStdColourMap(); 
1929     for ( int i 
= 0; i 
< width
; i
++ ) 
1931         for ( int j 
= 0; j 
< height
; j
++ ) 
1933             COLORREF pixel 
= ::GetPixel(hdcMem
, i
, j
); 
1935             for ( size_t k 
= 0; k 
< wxSTD_COL_MAX
; k
++ ) 
1937                 COLORREF col 
= cmap
[k
].from
; 
1938                 if ( abs(GetRValue(pixel
) - GetRValue(col
)) < 10 && 
1939                      abs(GetGValue(pixel
) - GetGValue(col
)) < 10 && 
1940                      abs(GetBValue(pixel
) - GetBValue(col
)) < 10 ) 
1942                     if ( cmap
[k
].to 
!= pixel 
) 
1943                         ::SetPixel(hdcMem
, i
, j
, cmap
[k
].to
); 
1953 #endif // wxREMAP_BUTTON_COLOURS 
1955 #endif // wxUSE_TOOLBAR