]> git.saurik.com Git - wxWidgets.git/blame - src/msw/imaglist.cpp
fixing focus, fixes #11911
[wxWidgets.git] / src / msw / imaglist.cpp
CommitLineData
2bda0e17 1/////////////////////////////////////////////////////////////////////////////
4b7f2165
VZ
2// Name: src/msw/imaglist.cpp
3// Purpose: wxImageList implementation for Win32
2bda0e17
KB
4// Author: Julian Smart
5// Modified by:
6// Created: 04/01/98
7// RCS-ID: $Id$
6c9a19aa 8// Copyright: (c) Julian Smart
65571936 9// Licence: wxWindows licence
2bda0e17
KB
10/////////////////////////////////////////////////////////////////////////////
11
4b7f2165
VZ
12// ============================================================================
13// declarations
14// ============================================================================
15
16// ----------------------------------------------------------------------------
17// headers
18// ----------------------------------------------------------------------------
19
2bda0e17
KB
20// For compilers that support precompilation, includes "wx.h".
21#include "wx/wxprec.h"
22
23#ifdef __BORLANDC__
4b7f2165 24 #pragma hdrstop
2bda0e17
KB
25#endif
26
2bda0e17 27#ifndef WX_PRECOMP
57bd4c60 28 #include "wx/msw/wrapcctl.h" // include <commctrl.h> "properly"
4b7f2165
VZ
29 #include "wx/window.h"
30 #include "wx/icon.h"
31 #include "wx/dc.h"
32 #include "wx/string.h"
e0f15f4a 33 #include "wx/dcmemory.h"
88a7a4e1 34 #include "wx/intl.h"
e4db172a 35 #include "wx/log.h"
155ecd4c 36 #include "wx/image.h"
4b7f2165 37 #include <stdio.h>
2bda0e17
KB
38#endif
39
01f512d7 40#include "wx/imaglist.h"
888dde65
RR
41#include "wx/dc.h"
42#include "wx/msw/dc.h"
2bda0e17
KB
43#include "wx/msw/private.h"
44
4b7f2165
VZ
45// ----------------------------------------------------------------------------
46// wxWin macros
47// ----------------------------------------------------------------------------
48
2bda0e17 49IMPLEMENT_DYNAMIC_CLASS(wxImageList, wxObject)
2bda0e17 50
4b7f2165
VZ
51#define GetHImageList() ((HIMAGELIST)m_hImageList)
52
53// ----------------------------------------------------------------------------
54// private functions
55// ----------------------------------------------------------------------------
56
57// returns the mask if it's valid, otherwise the bitmap mask and, if it's not
58// valid neither, a "solid" mask (no transparent zones at all)
3ca6a5f0 59static HBITMAP GetMaskForImage(const wxBitmap& bitmap, const wxBitmap& mask);
4b7f2165
VZ
60
61// ============================================================================
62// implementation
63// ============================================================================
64
65// ----------------------------------------------------------------------------
66// wxImageList creation/destruction
67// ----------------------------------------------------------------------------
68
69wxImageList::wxImageList()
2bda0e17
KB
70{
71 m_hImageList = 0;
72}
73
4b7f2165
VZ
74// Creates an image list
75bool wxImageList::Create(int width, int height, bool mask, int initial)
2bda0e17 76{
a6e2b3a8
VZ
77 UINT flags = 0;
78
a6712ab9
VZ
79 // as we want to be able to use 32bpp bitmaps in the image lists, we always
80 // use ILC_COLOR32, even if the display resolution is less -- the system
81 // will make the best effort to show the bitmap if we do this resulting in
82 // quite acceptable display while using a lower depth ILC_COLOR constant
83 // (e.g. ILC_COLOR16) shows completely broken bitmaps
4676948b
JS
84#ifdef __WXWINCE__
85 flags |= ILC_COLOR;
86#else
a6712ab9 87 flags |= ILC_COLOR32;
4676948b 88#endif
a6e2b3a8 89
4b7f2165
VZ
90 if ( mask )
91 flags |= ILC_MASK;
92
93 // Grow by 1, I guess this is reasonable behaviour most of the time
94 m_hImageList = (WXHIMAGELIST) ImageList_Create(width, height, flags,
95 initial, 1);
96 if ( !m_hImageList )
97 {
f6bcfd97 98 wxLogLastError(wxT("ImageList_Create()"));
4b7f2165 99 }
2bda0e17 100
4b7f2165
VZ
101 return m_hImageList != 0;
102}
2bda0e17 103
4b7f2165 104wxImageList::~wxImageList()
2bda0e17 105{
4b7f2165
VZ
106 if ( m_hImageList )
107 {
108 ImageList_Destroy(GetHImageList());
109 m_hImageList = 0;
110 }
2bda0e17
KB
111}
112
4b7f2165
VZ
113// ----------------------------------------------------------------------------
114// wxImageList attributes
115// ----------------------------------------------------------------------------
2bda0e17 116
4b7f2165
VZ
117// Returns the number of images in the image list.
118int wxImageList::GetImageCount() const
2bda0e17 119{
9a83f860 120 wxASSERT_MSG( m_hImageList, wxT("invalid image list") );
2bda0e17 121
4b7f2165 122 return ImageList_GetImageCount(GetHImageList());
2bda0e17
KB
123}
124
f6bcfd97
BP
125// Returns the size (same for all images) of the images in the list
126bool wxImageList::GetSize(int WXUNUSED(index), int &width, int &height) const
127{
9a83f860 128 wxASSERT_MSG( m_hImageList, wxT("invalid image list") );
f6bcfd97
BP
129
130 return ImageList_GetIconSize(GetHImageList(), &width, &height) != 0;
131}
132
4b7f2165
VZ
133// ----------------------------------------------------------------------------
134// wxImageList operations
135// ----------------------------------------------------------------------------
136
2bda0e17
KB
137// Adds a bitmap, and optionally a mask bitmap.
138// Note that wxImageList creates new bitmaps, so you may delete
139// 'bitmap' and 'mask'.
140int wxImageList::Add(const wxBitmap& bitmap, const wxBitmap& mask)
141{
3ca6a5f0 142 HBITMAP hbmpMask = GetMaskForImage(bitmap, mask);
e29b83a4 143
4b7f2165
VZ
144 int index = ImageList_Add(GetHImageList(), GetHbitmapOf(bitmap), hbmpMask);
145 if ( index == -1 )
e29b83a4
VZ
146 {
147 wxLogError(_("Couldn't add an image to the image list."));
148 }
149
4b7f2165 150 ::DeleteObject(hbmpMask);
54b37e2e 151
e29b83a4 152 return index;
2bda0e17
KB
153}
154
155// Adds a bitmap, using the specified colour to create the mask bitmap
156// Note that wxImageList creates new bitmaps, so you may delete
157// 'bitmap'.
158int wxImageList::Add(const wxBitmap& bitmap, const wxColour& maskColour)
159{
4b7f2165
VZ
160 int index = ImageList_AddMasked(GetHImageList(),
161 GetHbitmapOf(bitmap),
162 wxColourToRGB(maskColour));
163 if ( index == -1 )
164 {
165 wxLogError(_("Couldn't add an image to the image list."));
166 }
167
168 return index;
2bda0e17
KB
169}
170
171// Adds a bitmap and mask from an icon.
172int wxImageList::Add(const wxIcon& icon)
173{
4b7f2165
VZ
174 int index = ImageList_AddIcon(GetHImageList(), GetHiconOf(icon));
175 if ( index == -1 )
176 {
177 wxLogError(_("Couldn't add an image to the image list."));
178 }
179
180 return index;
2bda0e17
KB
181}
182
183// Replaces a bitmap, optionally passing a mask bitmap.
184// Note that wxImageList creates new bitmaps, so you may delete
185// 'bitmap' and 'mask'.
4b7f2165
VZ
186bool wxImageList::Replace(int index,
187 const wxBitmap& bitmap, const wxBitmap& mask)
2bda0e17 188{
3ca6a5f0 189 HBITMAP hbmpMask = GetMaskForImage(bitmap, mask);
2bda0e17 190
4b7f2165
VZ
191 bool ok = ImageList_Replace(GetHImageList(), index,
192 GetHbitmapOf(bitmap), hbmpMask) != 0;
193 if ( !ok )
194 {
3103e8a9 195 wxLogLastError(wxT("ImageList_Replace()"));
4b7f2165
VZ
196 }
197
198 ::DeleteObject(hbmpMask);
199
200 return ok;
2bda0e17 201}
2bda0e17
KB
202
203// Replaces a bitmap and mask from an icon.
4b7f2165 204bool wxImageList::Replace(int i, const wxIcon& icon)
2bda0e17 205{
9fe0c2fc 206 bool ok = ImageList_ReplaceIcon(GetHImageList(), i, GetHiconOf(icon)) != -1;
4b7f2165
VZ
207 if ( !ok )
208 {
f6bcfd97 209 wxLogLastError(wxT("ImageList_ReplaceIcon()"));
4b7f2165
VZ
210 }
211
212 return ok;
2bda0e17
KB
213}
214
215// Removes the image at the given index.
debe6624 216bool wxImageList::Remove(int index)
2bda0e17 217{
4b7f2165
VZ
218 bool ok = ImageList_Remove(GetHImageList(), index) != 0;
219 if ( !ok )
220 {
f6bcfd97 221 wxLogLastError(wxT("ImageList_Remove()"));
4b7f2165
VZ
222 }
223
224 return ok;
2bda0e17
KB
225}
226
227// Remove all images
4b7f2165 228bool wxImageList::RemoveAll()
2bda0e17 229{
3c1a88d8 230 // don't use ImageList_RemoveAll() because mingw32 headers don't have it
1c304f30 231 return Remove(-1);
2bda0e17
KB
232}
233
234// Draws the given image on a dc at the specified position.
59af881e 235// If 'solidBackground' is true, Draw sets the image list background
2bda0e17
KB
236// colour to the background colour of the wxDC, to speed up
237// drawing by eliminating masked drawing where possible.
4b7f2165
VZ
238bool wxImageList::Draw(int index,
239 wxDC& dc,
240 int x, int y,
241 int flags,
242 bool solidBackground)
2bda0e17 243{
888dde65
RR
244 wxDCImpl *impl = dc.GetImpl();
245 wxMSWDCImpl *msw_impl = wxDynamicCast( impl, wxMSWDCImpl );
246 if (!msw_impl)
247 return false;
03647350 248
888dde65 249 HDC hDC = GetHdcOf(*msw_impl);
9a83f860 250 wxCHECK_MSG( hDC, false, wxT("invalid wxDC in wxImageList::Draw") );
4b7f2165
VZ
251
252 COLORREF clr = CLR_NONE; // transparent by default
253 if ( solidBackground )
254 {
b33576e1
VZ
255 const wxBrush& brush = dc.GetBackground();
256 if ( brush.Ok() )
4b7f2165 257 {
b33576e1 258 clr = wxColourToRGB(brush.GetColour());
4b7f2165
VZ
259 }
260 }
261
262 ImageList_SetBkColor(GetHImageList(), clr);
2bda0e17
KB
263
264 UINT style = 0;
4b7f2165
VZ
265 if ( flags & wxIMAGELIST_DRAW_NORMAL )
266 style |= ILD_NORMAL;
267 if ( flags & wxIMAGELIST_DRAW_TRANSPARENT )
268 style |= ILD_TRANSPARENT;
269 if ( flags & wxIMAGELIST_DRAW_SELECTED )
270 style |= ILD_SELECTED;
271 if ( flags & wxIMAGELIST_DRAW_FOCUSED )
272 style |= ILD_FOCUS;
273
274 bool ok = ImageList_Draw(GetHImageList(), index, hDC, x, y, style) != 0;
275 if ( !ok )
276 {
f6bcfd97 277 wxLogLastError(wxT("ImageList_Draw()"));
4b7f2165
VZ
278 }
279
280 return ok;
2bda0e17
KB
281}
282
49bf4e3e
JS
283// Get the bitmap
284wxBitmap wxImageList::GetBitmap(int index) const
285{
a8ff046b 286#if wxUSE_WXDIB && wxUSE_IMAGE
49bf4e3e
JS
287 int bmp_width = 0, bmp_height = 0;
288 GetSize(index, bmp_width, bmp_height);
289
290 wxBitmap bitmap(bmp_width, bmp_height);
291 wxMemoryDC dc;
292 dc.SelectObject(bitmap);
293
294 // draw it the first time to find a suitable mask colour
295 ((wxImageList*)this)->Draw(index, dc, 0, 0, wxIMAGELIST_DRAW_TRANSPARENT);
296 dc.SelectObject(wxNullBitmap);
297
298 // find the suitable mask colour
299 wxImage image = bitmap.ConvertToImage();
300 unsigned char r = 0, g = 0, b = 0;
301 image.FindFirstUnusedColour(&r, &g, &b);
302
303 // redraw whole image and bitmap in the mask colour
304 image.Create(bmp_width, bmp_height);
305 image.Replace(0, 0, 0, r, g, b);
306 bitmap = wxBitmap(image);
307
308 // redraw icon over the mask colour to actually draw it
309 dc.SelectObject(bitmap);
310 ((wxImageList*)this)->Draw(index, dc, 0, 0, wxIMAGELIST_DRAW_TRANSPARENT);
311 dc.SelectObject(wxNullBitmap);
312
313 // get the image, set the mask colour and convert back to get transparent bitmap
314 image = bitmap.ConvertToImage();
315 image.SetMaskColour(r, g, b);
316 bitmap = wxBitmap(image);
64c288fa
JS
317#else
318 wxBitmap bitmap;
64c288fa 319#endif
74aa8254 320 return bitmap;
49bf4e3e
JS
321}
322
323// Get the icon
324wxIcon wxImageList::GetIcon(int index) const
325{
326 HICON hIcon = ImageList_ExtractIcon(0, GetHImageList(), index);
327 if (hIcon)
328 {
329 wxIcon icon;
330 icon.SetHICON((WXHICON)hIcon);
a71d815b 331
49bf4e3e
JS
332 int iconW, iconH;
333 GetSize(index, iconW, iconH);
334 icon.SetSize(iconW, iconH);
a71d815b 335
49bf4e3e
JS
336 return icon;
337 }
a71d815b 338 else
49bf4e3e
JS
339 return wxNullIcon;
340}
341
4b7f2165
VZ
342// ----------------------------------------------------------------------------
343// helpers
344// ----------------------------------------------------------------------------
345
3ca6a5f0 346static HBITMAP GetMaskForImage(const wxBitmap& bitmap, const wxBitmap& mask)
4b7f2165 347{
e4a84806
VZ
348#if wxUSE_IMAGE
349 wxBitmap bitmapWithMask;
350#endif // wxUSE_IMAGE
351
3ca6a5f0 352 HBITMAP hbmpMask;
90c1530a 353 wxMask *pMask;
59af881e 354 bool deleteMask = false;
4b7f2165
VZ
355
356 if ( mask.Ok() )
357 {
3ca6a5f0 358 hbmpMask = GetHbitmapOf(mask);
90c1530a 359 pMask = NULL;
4b7f2165
VZ
360 }
361 else
362 {
90c1530a 363 pMask = bitmap.GetMask();
e4a84806
VZ
364
365#if wxUSE_IMAGE
366 // check if we don't have alpha in this bitmap -- we can create a mask
367 // from it (and we need to do it for the older systems which don't
368 // support 32bpp bitmaps natively)
369 if ( !pMask )
370 {
371 wxImage img(bitmap.ConvertToImage());
372 if ( img.HasAlpha() )
373 {
374 img.ConvertAlphaToMask();
375 bitmapWithMask = wxBitmap(img);
376 pMask = bitmapWithMask.GetMask();
377 }
378 }
379#endif // wxUSE_IMAGE
380
90c1530a 381 if ( !pMask )
4b7f2165 382 {
90c1530a
VZ
383 // use the light grey count as transparent: the trouble here is
384 // that the light grey might have been changed by Windows behind
385 // our back, so use the standard colour map to get its real value
386 wxCOLORMAP *cmap = wxGetStdColourMap();
387 wxColour col;
388 wxRGBToColour(col, cmap[wxSTD_COL_BTNFACE].from);
4b7f2165 389
90c1530a 390 pMask = new wxMask(bitmap, col);
3ca6a5f0 391
59af881e 392 deleteMask = true;
3ca6a5f0 393 }
90c1530a
VZ
394
395 hbmpMask = (HBITMAP)pMask->GetMaskBitmap();
4b7f2165
VZ
396 }
397
77ffb593 398 // windows mask convention is opposite to the wxWidgets one
3ca6a5f0 399 HBITMAP hbmpMaskInv = wxInvertMask(hbmpMask);
90c1530a
VZ
400
401 if ( deleteMask )
402 {
403 delete pMask;
404 }
3ca6a5f0
BP
405
406 return hbmpMaskInv;
4b7f2165 407}