1 /////////////////////////////////////////////////////////////////////////////
2 // Name: src/msw/imaglist.cpp
3 // Purpose: wxImageList implementation for Win32
4 // Author: Julian Smart
8 // Copyright: (c) Julian Smart
9 // Licence: wxWindows licence
10 /////////////////////////////////////////////////////////////////////////////
12 // ============================================================================
14 // ============================================================================
16 // ----------------------------------------------------------------------------
18 // ----------------------------------------------------------------------------
20 #if defined(__GNUG__) && !defined(NO_GCC_PRAGMA)
21 #pragma implementation "imaglist.h"
24 // For compilers that support precompilation, includes "wx.h".
25 #include "wx/wxprec.h"
31 #if defined(__WIN95__)
34 #include "wx/window.h"
37 #include "wx/string.h"
38 #include "wx/dcmemory.h"
47 #include "wx/msw/imaglist.h"
48 #include "wx/msw/private.h"
50 #if defined(__WIN95__) && !(defined(__GNUWIN32_OLD__) && !defined(__CYGWIN10__))
54 // ----------------------------------------------------------------------------
56 // ----------------------------------------------------------------------------
58 IMPLEMENT_DYNAMIC_CLASS(wxImageList
, wxObject
)
60 #define GetHImageList() ((HIMAGELIST)m_hImageList)
62 // ----------------------------------------------------------------------------
64 // ----------------------------------------------------------------------------
66 // returns the mask if it's valid, otherwise the bitmap mask and, if it's not
67 // valid neither, a "solid" mask (no transparent zones at all)
68 static HBITMAP
GetMaskForImage(const wxBitmap
& bitmap
, const wxBitmap
& mask
);
70 // ============================================================================
72 // ============================================================================
74 // ----------------------------------------------------------------------------
75 // wxImageList creation/destruction
76 // ----------------------------------------------------------------------------
78 wxImageList::wxImageList()
83 // Creates an image list
84 bool wxImageList::Create(int width
, int height
, bool mask
, int initial
)
88 // set appropriate color depth
92 int dd
= wxDisplayDepth();
94 if (dd
<= 4) flags
|= ILC_COLOR
; // 16 color
95 else if (dd
<= 8) flags
|= ILC_COLOR8
; // 256 color
96 else if (dd
<= 16) flags
|= ILC_COLOR16
; // 64k hi-color
97 else if (dd
<= 24) flags
|= ILC_COLOR24
; // 16m truecolor
98 else if (dd
<= 32) flags
|= ILC_COLOR32
; // 16m truecolor
104 // Grow by 1, I guess this is reasonable behaviour most of the time
105 m_hImageList
= (WXHIMAGELIST
) ImageList_Create(width
, height
, flags
,
109 wxLogLastError(wxT("ImageList_Create()"));
112 return m_hImageList
!= 0;
115 wxImageList::~wxImageList()
119 ImageList_Destroy(GetHImageList());
124 // ----------------------------------------------------------------------------
125 // wxImageList attributes
126 // ----------------------------------------------------------------------------
128 // Returns the number of images in the image list.
129 int wxImageList::GetImageCount() const
131 wxASSERT_MSG( m_hImageList
, _T("invalid image list") );
133 return ImageList_GetImageCount(GetHImageList());
136 // Returns the size (same for all images) of the images in the list
137 bool wxImageList::GetSize(int WXUNUSED(index
), int &width
, int &height
) const
139 wxASSERT_MSG( m_hImageList
, _T("invalid image list") );
141 return ImageList_GetIconSize(GetHImageList(), &width
, &height
) != 0;
144 // ----------------------------------------------------------------------------
145 // wxImageList operations
146 // ----------------------------------------------------------------------------
148 // Adds a bitmap, and optionally a mask bitmap.
149 // Note that wxImageList creates new bitmaps, so you may delete
150 // 'bitmap' and 'mask'.
151 int wxImageList::Add(const wxBitmap
& bitmap
, const wxBitmap
& mask
)
153 HBITMAP hbmpMask
= GetMaskForImage(bitmap
, mask
);
155 int index
= ImageList_Add(GetHImageList(), GetHbitmapOf(bitmap
), hbmpMask
);
158 wxLogError(_("Couldn't add an image to the image list."));
161 ::DeleteObject(hbmpMask
);
166 // Adds a bitmap, using the specified colour to create the mask bitmap
167 // Note that wxImageList creates new bitmaps, so you may delete
169 int wxImageList::Add(const wxBitmap
& bitmap
, const wxColour
& maskColour
)
171 int index
= ImageList_AddMasked(GetHImageList(),
172 GetHbitmapOf(bitmap
),
173 wxColourToRGB(maskColour
));
176 wxLogError(_("Couldn't add an image to the image list."));
182 // Adds a bitmap and mask from an icon.
183 int wxImageList::Add(const wxIcon
& icon
)
185 int index
= ImageList_AddIcon(GetHImageList(), GetHiconOf(icon
));
188 wxLogError(_("Couldn't add an image to the image list."));
194 // Replaces a bitmap, optionally passing a mask bitmap.
195 // Note that wxImageList creates new bitmaps, so you may delete
196 // 'bitmap' and 'mask'.
197 bool wxImageList::Replace(int index
,
198 const wxBitmap
& bitmap
, const wxBitmap
& mask
)
200 HBITMAP hbmpMask
= GetMaskForImage(bitmap
, mask
);
202 bool ok
= ImageList_Replace(GetHImageList(), index
,
203 GetHbitmapOf(bitmap
), hbmpMask
) != 0;
206 wxLogLastError(wxT("ImageList_Add()"));
209 ::DeleteObject(hbmpMask
);
214 // Replaces a bitmap and mask from an icon.
215 bool wxImageList::Replace(int i
, const wxIcon
& icon
)
217 bool ok
= ImageList_ReplaceIcon(GetHImageList(), i
, GetHiconOf(icon
)) != 0;
220 wxLogLastError(wxT("ImageList_ReplaceIcon()"));
226 // Removes the image at the given index.
227 bool wxImageList::Remove(int index
)
229 bool ok
= ImageList_Remove(GetHImageList(), index
) != 0;
232 wxLogLastError(wxT("ImageList_Remove()"));
239 bool wxImageList::RemoveAll()
241 // don't use ImageList_RemoveAll() because mingw32 headers don't have it
242 int count
= ImageList_GetImageCount(GetHImageList());
243 for ( int i
= 0; i
< count
; i
++ )
245 // the image indexes are shifted, so we should always remove the first
253 // Draws the given image on a dc at the specified position.
254 // If 'solidBackground' is true, Draw sets the image list background
255 // colour to the background colour of the wxDC, to speed up
256 // drawing by eliminating masked drawing where possible.
257 bool wxImageList::Draw(int index
,
261 bool solidBackground
)
263 HDC hDC
= GetHdcOf(dc
);
264 wxCHECK_MSG( hDC
, false, _T("invalid wxDC in wxImageList::Draw") );
266 COLORREF clr
= CLR_NONE
; // transparent by default
267 if ( solidBackground
)
269 const wxBrush
& brush
= dc
.GetBackground();
272 clr
= wxColourToRGB(brush
.GetColour());
276 ImageList_SetBkColor(GetHImageList(), clr
);
279 if ( flags
& wxIMAGELIST_DRAW_NORMAL
)
281 if ( flags
& wxIMAGELIST_DRAW_TRANSPARENT
)
282 style
|= ILD_TRANSPARENT
;
283 if ( flags
& wxIMAGELIST_DRAW_SELECTED
)
284 style
|= ILD_SELECTED
;
285 if ( flags
& wxIMAGELIST_DRAW_FOCUSED
)
288 bool ok
= ImageList_Draw(GetHImageList(), index
, hDC
, x
, y
, style
) != 0;
291 wxLogLastError(wxT("ImageList_Draw()"));
298 wxBitmap
wxImageList::GetBitmap(int index
) const
300 int bmp_width
= 0, bmp_height
= 0;
301 GetSize(index
, bmp_width
, bmp_height
);
303 wxBitmap
bitmap(bmp_width
, bmp_height
);
305 dc
.SelectObject(bitmap
);
307 // draw it the first time to find a suitable mask colour
308 ((wxImageList
*)this)->Draw(index
, dc
, 0, 0, wxIMAGELIST_DRAW_TRANSPARENT
);
309 dc
.SelectObject(wxNullBitmap
);
311 // find the suitable mask colour
312 wxImage image
= bitmap
.ConvertToImage();
313 unsigned char r
= 0, g
= 0, b
= 0;
314 image
.FindFirstUnusedColour(&r
, &g
, &b
);
316 // redraw whole image and bitmap in the mask colour
317 image
.Create(bmp_width
, bmp_height
);
318 image
.Replace(0, 0, 0, r
, g
, b
);
319 bitmap
= wxBitmap(image
);
321 // redraw icon over the mask colour to actually draw it
322 dc
.SelectObject(bitmap
);
323 ((wxImageList
*)this)->Draw(index
, dc
, 0, 0, wxIMAGELIST_DRAW_TRANSPARENT
);
324 dc
.SelectObject(wxNullBitmap
);
326 // get the image, set the mask colour and convert back to get transparent bitmap
327 image
= bitmap
.ConvertToImage();
328 image
.SetMaskColour(r
, g
, b
);
329 bitmap
= wxBitmap(image
);
335 wxIcon
wxImageList::GetIcon(int index
) const
337 HICON hIcon
= ImageList_ExtractIcon(0, GetHImageList(), index
);
341 icon
.SetHICON((WXHICON
)hIcon
);
344 GetSize(index
, iconW
, iconH
);
345 icon
.SetSize(iconW
, iconH
);
353 // ----------------------------------------------------------------------------
355 // ----------------------------------------------------------------------------
357 static HBITMAP
GetMaskForImage(const wxBitmap
& bitmap
, const wxBitmap
& mask
)
361 bool deleteMask
= false;
365 hbmpMask
= GetHbitmapOf(mask
);
370 pMask
= bitmap
.GetMask();
373 // use the light grey count as transparent: the trouble here is
374 // that the light grey might have been changed by Windows behind
375 // our back, so use the standard colour map to get its real value
376 wxCOLORMAP
*cmap
= wxGetStdColourMap();
378 wxRGBToColour(col
, cmap
[wxSTD_COL_BTNFACE
].from
);
380 pMask
= new wxMask(bitmap
, col
);
385 hbmpMask
= (HBITMAP
)pMask
->GetMaskBitmap();
388 // windows mask convention is opposite to the wxWidgets one
389 HBITMAP hbmpMaskInv
= wxInvertMask(hbmpMask
);