///////////////////////////////////////////////////////////////////////////// // Name: src/generic/bmpcboxg.cpp // Purpose: wxBitmapComboBox // Author: Jaakko Salli // Modified by: // Created: Aug-31-2006 // RCS-ID: $Id$ // Copyright: (c) 2005 Jaakko Salli // Licence: wxWindows licence ///////////////////////////////////////////////////////////////////////////// // ============================================================================ // declarations // ============================================================================ // ---------------------------------------------------------------------------- // headers // ---------------------------------------------------------------------------- #include "wx/wxprec.h" #ifdef __BORLANDC__ #pragma hdrstop #endif #if wxUSE_BITMAPCOMBOBOX #include "wx/bmpcbox.h" #if defined(wxGENERIC_BITMAPCOMBOBOX) #ifndef WX_PRECOMP #include "wx/log.h" #endif #include "wx/odcombo.h" #include "wx/settings.h" #include "wx/dc.h" #if wxUSE_IMAGE #include "wx/image.h" #endif const wxChar wxBitmapComboBoxNameStr[] = wxT("bitmapComboBox"); // These macros allow wxArrayPtrVoid to be used in more convenient manner #define GetBitmapPtr(n) ((wxBitmap*)m_bitmaps[n]) #define IMAGE_SPACING_RIGHT 4 // Space left of image #define IMAGE_SPACING_LEFT 4 // Space right of image, left of text #define IMAGE_SPACING_VERTICAL 2 // Space top and bottom of image #define IMAGE_SPACING_CTRL_VERTICAL 7 // Spacing used in control size calculation #define EXTRA_FONT_HEIGHT 0 // Add to increase min. height of list items // ============================================================================ // implementation // ============================================================================ BEGIN_EVENT_TABLE(wxBitmapComboBox, wxOwnerDrawnComboBox) EVT_SIZE(wxBitmapComboBox::OnSize) END_EVENT_TABLE() IMPLEMENT_DYNAMIC_CLASS(wxBitmapComboBox, wxOwnerDrawnComboBox) void wxBitmapComboBox::Init() { m_fontHeight = 0; m_imgAreaWidth = 0; m_inResize = false; } wxBitmapComboBox::wxBitmapComboBox(wxWindow *parent, wxWindowID id, const wxString& value, const wxPoint& pos, const wxSize& size, const wxArrayString& choices, long style, const wxValidator& validator, const wxString& name) : wxOwnerDrawnComboBox(), wxBitmapComboBoxBase() { Init(); Create(parent,id,value,pos,size,choices,style,validator,name); } bool wxBitmapComboBox::Create(wxWindow *parent, wxWindowID id, const wxString& value, const wxPoint& pos, const wxSize& size, const wxArrayString& choices, long style, const wxValidator& validator, const wxString& name) { if ( !wxOwnerDrawnComboBox::Create(parent, id, value, pos, size, choices, style, validator, name) ) { return false; } PostCreate(); return true; } bool wxBitmapComboBox::Create(wxWindow *parent, wxWindowID id, const wxString& value, const wxPoint& pos, const wxSize& size, int n, const wxString choices[], long style, const wxValidator& validator, const wxString& name) { if ( !wxOwnerDrawnComboBox::Create(parent, id, value, pos, size, n, choices, style, validator, name) ) { return false; } PostCreate(); return true; } void wxBitmapComboBox::PostCreate() { m_fontHeight = GetCharHeight() + EXTRA_FONT_HEIGHT; while ( m_bitmaps.GetCount() < GetCount() ) m_bitmaps.Add( new wxBitmap() ); } wxBitmapComboBox::~wxBitmapComboBox() { Clear(); } // ---------------------------------------------------------------------------- // Item manipulation // ---------------------------------------------------------------------------- void wxBitmapComboBox::SetItemBitmap(unsigned int n, const wxBitmap& bitmap) { wxCHECK_RET( n < GetCount(), wxT("invalid item index") ); OnAddBitmap(bitmap); *GetBitmapPtr(n) = bitmap; if ( (int)n == GetSelection() ) Refresh(); } wxBitmap wxBitmapComboBox::GetItemBitmap(unsigned int n) const { wxCHECK_MSG( n < GetCount(), wxNullBitmap, wxT("invalid item index") ); return *GetBitmapPtr(n); } int wxBitmapComboBox::DoInsertItems(const wxArrayStringsAdapter & items, unsigned int pos, void **clientData, wxClientDataType type) { const unsigned int numItems = items.GetCount(); const unsigned int countNew = GetCount() + numItems; m_bitmaps.Alloc(countNew); for ( unsigned int i = 0; i < numItems; ++i ) { m_bitmaps.Insert(new wxBitmap(wxNullBitmap), pos + i); } const int index = wxOwnerDrawnComboBox::DoInsertItems(items, pos, clientData, type); if ( index == wxNOT_FOUND ) { for ( int i = countNew - GetCount(); i > 0; --i ) { wxBitmap *bmp = GetBitmapPtr(pos); m_bitmaps.RemoveAt(pos); delete bmp; } } return index; } int wxBitmapComboBox::Append(const wxString& item, const wxBitmap& bitmap) { const int n = wxOwnerDrawnComboBox::Append(item); if(n != wxNOT_FOUND) SetItemBitmap(n, bitmap); return n; } int wxBitmapComboBox::Append(const wxString& item, const wxBitmap& bitmap, void *clientData) { const int n = wxOwnerDrawnComboBox::Append(item, clientData); if(n != wxNOT_FOUND) SetItemBitmap(n, bitmap); return n; } int wxBitmapComboBox::Append(const wxString& item, const wxBitmap& bitmap, wxClientData *clientData) { const int n = wxOwnerDrawnComboBox::Append(item, clientData); if(n != wxNOT_FOUND) SetItemBitmap(n, bitmap); return n; } int wxBitmapComboBox::Insert(const wxString& item, const wxBitmap& bitmap, unsigned int pos) { const int n = wxOwnerDrawnComboBox::Insert(item, pos); if(n != wxNOT_FOUND) SetItemBitmap(n, bitmap); return n; } int wxBitmapComboBox::Insert(const wxString& item, const wxBitmap& bitmap, unsigned int pos, void *clientData) { const int n = wxOwnerDrawnComboBox::Insert(item, pos, clientData); if(n != wxNOT_FOUND) SetItemBitmap(n, bitmap); return n; } int wxBitmapComboBox::Insert(const wxString& item, const wxBitmap& bitmap, unsigned int pos, wxClientData *clientData) { const int n = wxOwnerDrawnComboBox::Insert(item, pos, clientData); if(n != wxNOT_FOUND) SetItemBitmap(n, bitmap); return n; } bool wxBitmapComboBox::OnAddBitmap(const wxBitmap& bitmap) { if ( bitmap.Ok() ) { int width = bitmap.GetWidth(); int height = bitmap.GetHeight(); if ( m_usedImgSize.x <= 0 ) { // // If size not yet determined, get it from this image. m_usedImgSize.x = width; m_usedImgSize.y = height; InvalidateBestSize(); wxSize newSz = GetBestSize(); wxSize sz = GetSize(); if ( newSz.y > sz.y ) SetSize(sz.x, newSz.y); else DetermineIndent(); } wxCHECK_MSG(width == m_usedImgSize.x && height == m_usedImgSize.y, false, wxT("you can only add images of same size")); } return true; } void wxBitmapComboBox::DoClear() { wxOwnerDrawnComboBox::DoClear(); unsigned int i; for ( i=0; i 0 ) { indent = m_usedImgSize.x + IMAGE_SPACING_LEFT + IMAGE_SPACING_RIGHT; m_imgAreaWidth = indent; indent -= 3; } SetCustomPaintWidth(indent); } void wxBitmapComboBox::OnSize(wxSizeEvent& event) { // Prevent infinite looping if ( !m_inResize ) { m_inResize = true; DetermineIndent(); m_inResize = false; } event.Skip(); } wxSize wxBitmapComboBox::DoGetBestSize() const { wxSize sz = wxOwnerDrawnComboBox::DoGetBestSize(); // Scale control to match height of highest image. int h2 = m_usedImgSize.y + IMAGE_SPACING_CTRL_VERTICAL; if ( h2 > sz.y ) sz.y = h2; CacheBestSize(sz); return sz; } // ---------------------------------------------------------------------------- // wxBitmapComboBox miscellaneous // ---------------------------------------------------------------------------- bool wxBitmapComboBox::SetFont(const wxFont& font) { bool res = wxOwnerDrawnComboBox::SetFont(font); m_fontHeight = GetCharHeight() + EXTRA_FONT_HEIGHT; return res; } // ---------------------------------------------------------------------------- // wxBitmapComboBox item drawing and measuring // ---------------------------------------------------------------------------- void wxBitmapComboBox::OnDrawBackground(wxDC& dc, const wxRect& rect, int item, int flags) const { if ( GetCustomPaintWidth() == 0 || !(flags & wxODCB_PAINTING_SELECTED) || item < 0 || ( (flags & wxODCB_PAINTING_CONTROL) && (GetInternalFlags() & wxCC_FULL_BUTTON)) ) { wxOwnerDrawnComboBox::OnDrawBackground(dc, rect, item, flags); return; } // // Just paint simple selection background under where is text // (ie. emulate what MSW image choice does). // int xPos = 0; // Starting x of selection rectangle const int vSizeDec = 1; // Vertical size reduction of selection rectangle edges xPos = GetCustomPaintWidth() + 2; wxCoord x, y; GetTextExtent(GetString(item), &x, &y, 0, 0); dc.SetTextForeground(wxSystemSettings::GetColour(wxSYS_COLOUR_HIGHLIGHTTEXT)); wxColour selCol = wxSystemSettings::GetColour(wxSYS_COLOUR_HIGHLIGHT); dc.SetPen(selCol); dc.SetBrush(selCol); dc.DrawRectangle(rect.x+xPos, rect.y+vSizeDec, x + 4, rect.height-(vSizeDec*2)); } void wxBitmapComboBox::OnDrawItem(wxDC& dc, const wxRect& rect, int item, int flags) const { wxString text; int imgAreaWidth = m_imgAreaWidth; bool drawText; if ( imgAreaWidth == 0 ) { wxOwnerDrawnComboBox::OnDrawItem(dc, rect, item, flags); return; } if ( flags & wxODCB_PAINTING_CONTROL ) { text = GetValue(); if ( HasFlag(wxCB_READONLY) ) drawText = true; else drawText = false; } else { text = GetString(item); drawText = true; } const wxBitmap& bmp = *GetBitmapPtr(item); if ( bmp.Ok() ) { wxCoord w = bmp.GetWidth(); wxCoord h = bmp.GetHeight(); // Draw the image centered dc.DrawBitmap(bmp, rect.x + (m_usedImgSize.x-w)/2 + IMAGE_SPACING_LEFT, rect.y + (rect.height-h)/2, true); } if ( drawText ) dc.DrawText(GetString(item), rect.x + imgAreaWidth + 1, rect.y + (rect.height-dc.GetCharHeight())/2); } wxCoord wxBitmapComboBox::OnMeasureItem(size_t WXUNUSED(item)) const { int imgHeightArea = m_usedImgSize.y + 2; return imgHeightArea > m_fontHeight ? imgHeightArea : m_fontHeight; } wxCoord wxBitmapComboBox::OnMeasureItemWidth(size_t item) const { wxCoord x, y; GetTextExtent(GetString(item), &x, &y, 0, 0); x += m_imgAreaWidth; return x; } #endif // defined(wxGENERIC_BITMAPCOMBOBOX) #endif // wxUSE_BITMAPCOMBOBOX