]> git.saurik.com Git - wxWidgets.git/blame_incremental - src/msw/gdiimage.cpp
make sure we don't use uninitalized output stream in OnSysWrite() (coverity checked...
[wxWidgets.git] / src / msw / gdiimage.cpp
... / ...
CommitLineData
1///////////////////////////////////////////////////////////////////////////////
2// Name: msw/gdiimage.cpp
3// Purpose: wxGDIImage implementation
4// Author: Vadim Zeitlin
5// Modified by:
6// Created: 20.11.99
7// RCS-ID: $Id$
8// Copyright: (c) 1999 Vadim Zeitlin <zeitlin@dptmaths.ens-cachan.fr>
9// Licence: wxWindows licence
10///////////////////////////////////////////////////////////////////////////////
11
12// ============================================================================
13// declarations
14// ============================================================================
15
16// ----------------------------------------------------------------------------
17// headers
18// ----------------------------------------------------------------------------
19
20// For compilers that support precompilation, includes "wx.h".
21#include "wx/wxprec.h"
22
23#ifdef __BORLANDC__
24 #pragma hdrstop
25#endif
26
27#ifndef WX_PRECOMP
28 #include "wx/string.h"
29 #include "wx/log.h"
30#endif // WX_PRECOMP
31
32#include "wx/msw/private.h"
33
34#include "wx/app.h"
35
36#include "wx/bitmap.h"
37#include "wx/msw/gdiimage.h"
38
39#if wxUSE_WXDIB
40#include "wx/msw/dib.h"
41#endif
42
43#ifdef __WXWINCE__
44#include <winreg.h>
45#include <shellapi.h>
46#endif
47
48#include "wx/file.h"
49
50#include "wx/listimpl.cpp"
51WX_DEFINE_LIST(wxGDIImageHandlerList)
52
53// ----------------------------------------------------------------------------
54// private classes
55// ----------------------------------------------------------------------------
56
57#ifndef __WXMICROWIN__
58
59// all image handlers are declared/defined in this file because the outside
60// world doesn't have to know about them (but only about wxBITMAP_TYPE_XXX ids)
61
62class WXDLLEXPORT wxBMPFileHandler : public wxBitmapHandler
63{
64public:
65 wxBMPFileHandler() : wxBitmapHandler(_T("Windows bitmap file"), _T("bmp"),
66 wxBITMAP_TYPE_BMP)
67 {
68 }
69
70 virtual bool LoadFile(wxBitmap *bitmap,
71 const wxString& name, long flags,
72 int desiredWidth, int desiredHeight);
73 virtual bool SaveFile(wxBitmap *bitmap,
74 const wxString& name, int type,
75 const wxPalette *palette = NULL);
76
77private:
78 DECLARE_DYNAMIC_CLASS(wxBMPFileHandler)
79};
80
81class WXDLLEXPORT wxBMPResourceHandler: public wxBitmapHandler
82{
83public:
84 wxBMPResourceHandler() : wxBitmapHandler(_T("Windows bitmap resource"),
85 wxEmptyString,
86 wxBITMAP_TYPE_BMP_RESOURCE)
87 {
88 }
89
90 virtual bool LoadFile(wxBitmap *bitmap,
91 const wxString& name, long flags,
92 int desiredWidth, int desiredHeight);
93
94private:
95 DECLARE_DYNAMIC_CLASS(wxBMPResourceHandler)
96};
97
98class WXDLLEXPORT wxIconHandler : public wxGDIImageHandler
99{
100public:
101 wxIconHandler(const wxString& name, const wxString& ext, long type)
102 : wxGDIImageHandler(name, ext, type)
103 {
104 }
105
106 // creating and saving icons is not supported
107 virtual bool Create(wxGDIImage *WXUNUSED(image),
108 void *WXUNUSED(data),
109 long WXUNUSED(flags),
110 int WXUNUSED(width),
111 int WXUNUSED(height),
112 int WXUNUSED(depth) = 1)
113 {
114 return false;
115 }
116
117 virtual bool Save(wxGDIImage *WXUNUSED(image),
118 const wxString& WXUNUSED(name),
119 int WXUNUSED(type))
120 {
121 return false;
122 }
123
124 virtual bool Load(wxGDIImage *image,
125 const wxString& name,
126 long flags,
127 int desiredWidth, int desiredHeight)
128 {
129 wxIcon *icon = wxDynamicCast(image, wxIcon);
130 wxCHECK_MSG( icon, false, _T("wxIconHandler only works with icons") );
131
132 return LoadIcon(icon, name, flags, desiredWidth, desiredHeight);
133 }
134
135protected:
136 virtual bool LoadIcon(wxIcon *icon,
137 const wxString& name, long flags,
138 int desiredWidth = -1, int desiredHeight = -1) = 0;
139};
140
141class WXDLLEXPORT wxICOFileHandler : public wxIconHandler
142{
143public:
144 wxICOFileHandler() : wxIconHandler(_T("ICO icon file"),
145 _T("ico"),
146 wxBITMAP_TYPE_ICO)
147 {
148 }
149
150protected:
151 virtual bool LoadIcon(wxIcon *icon,
152 const wxString& name, long flags,
153 int desiredWidth = -1, int desiredHeight = -1);
154
155private:
156 DECLARE_DYNAMIC_CLASS(wxICOFileHandler)
157};
158
159class WXDLLEXPORT wxICOResourceHandler: public wxIconHandler
160{
161public:
162 wxICOResourceHandler() : wxIconHandler(_T("ICO resource"),
163 _T("ico"),
164 wxBITMAP_TYPE_ICO_RESOURCE)
165 {
166 }
167
168protected:
169 virtual bool LoadIcon(wxIcon *icon,
170 const wxString& name, long flags,
171 int desiredWidth = -1, int desiredHeight = -1);
172
173private:
174 DECLARE_DYNAMIC_CLASS(wxICOResourceHandler)
175};
176
177// ----------------------------------------------------------------------------
178// wxWin macros
179// ----------------------------------------------------------------------------
180
181IMPLEMENT_DYNAMIC_CLASS(wxBMPFileHandler, wxBitmapHandler)
182IMPLEMENT_DYNAMIC_CLASS(wxBMPResourceHandler, wxBitmapHandler)
183IMPLEMENT_DYNAMIC_CLASS(wxICOFileHandler, wxObject)
184IMPLEMENT_DYNAMIC_CLASS(wxICOResourceHandler, wxObject)
185
186// ----------------------------------------------------------------------------
187// private functions
188// ----------------------------------------------------------------------------
189
190#endif
191 // __MICROWIN__
192
193// ============================================================================
194// implementation
195// ============================================================================
196
197wxGDIImageHandlerList wxGDIImage::ms_handlers;
198
199// ----------------------------------------------------------------------------
200// wxGDIImage functions forwarded to wxGDIImageRefData
201// ----------------------------------------------------------------------------
202
203bool wxGDIImage::FreeResource(bool WXUNUSED(force))
204{
205 if ( !IsNull() )
206 {
207 GetGDIImageData()->Free();
208 GetGDIImageData()->m_handle = 0;
209 }
210
211 return true;
212}
213
214WXHANDLE wxGDIImage::GetResourceHandle() const
215{
216 return GetHandle();
217}
218
219// ----------------------------------------------------------------------------
220// wxGDIImage handler stuff
221// ----------------------------------------------------------------------------
222
223void wxGDIImage::AddHandler(wxGDIImageHandler *handler)
224{
225 ms_handlers.Append(handler);
226}
227
228void wxGDIImage::InsertHandler(wxGDIImageHandler *handler)
229{
230 ms_handlers.Insert(handler);
231}
232
233bool wxGDIImage::RemoveHandler(const wxString& name)
234{
235 wxGDIImageHandler *handler = FindHandler(name);
236 if ( handler )
237 {
238 ms_handlers.DeleteObject(handler);
239 return true;
240 }
241 else
242 return false;
243}
244
245wxGDIImageHandler *wxGDIImage::FindHandler(const wxString& name)
246{
247 wxGDIImageHandlerList::compatibility_iterator node = ms_handlers.GetFirst();
248 while ( node )
249 {
250 wxGDIImageHandler *handler = node->GetData();
251 if ( handler->GetName() == name )
252 return handler;
253 node = node->GetNext();
254 }
255
256 return NULL;
257}
258
259wxGDIImageHandler *wxGDIImage::FindHandler(const wxString& extension,
260 long type)
261{
262 wxGDIImageHandlerList::compatibility_iterator node = ms_handlers.GetFirst();
263 while ( node )
264 {
265 wxGDIImageHandler *handler = node->GetData();
266 if ( (handler->GetExtension() == extension) &&
267 (type == -1 || handler->GetType() == type) )
268 {
269 return handler;
270 }
271
272 node = node->GetNext();
273 }
274 return NULL;
275}
276
277wxGDIImageHandler *wxGDIImage::FindHandler(long type)
278{
279 wxGDIImageHandlerList::compatibility_iterator node = ms_handlers.GetFirst();
280 while ( node )
281 {
282 wxGDIImageHandler *handler = node->GetData();
283 if ( handler->GetType() == type )
284 return handler;
285
286 node = node->GetNext();
287 }
288
289 return NULL;
290}
291
292void wxGDIImage::CleanUpHandlers()
293{
294 wxGDIImageHandlerList::compatibility_iterator node = ms_handlers.GetFirst();
295 while ( node )
296 {
297 wxGDIImageHandler *handler = node->GetData();
298 wxGDIImageHandlerList::compatibility_iterator next = node->GetNext();
299 delete handler;
300 ms_handlers.Erase( node );
301 node = next;
302 }
303}
304
305void wxGDIImage::InitStandardHandlers()
306{
307#ifndef __WXMICROWIN__
308 AddHandler(new wxBMPResourceHandler);
309 AddHandler(new wxBMPFileHandler);
310 AddHandler(new wxICOResourceHandler);
311 AddHandler(new wxICOFileHandler);
312#endif
313}
314
315#ifndef __WXMICROWIN__
316
317// ----------------------------------------------------------------------------
318// wxBitmap handlers
319// ----------------------------------------------------------------------------
320
321bool wxBMPResourceHandler::LoadFile(wxBitmap *bitmap,
322 const wxString& name, long WXUNUSED(flags),
323 int WXUNUSED(desiredWidth),
324 int WXUNUSED(desiredHeight))
325{
326 // TODO: load colourmap.
327 bitmap->SetHBITMAP((WXHBITMAP)::LoadBitmap(wxGetInstance(), name));
328
329 if ( !bitmap->Ok() )
330 {
331 // it's probably not found
332 wxLogError(wxT("Can't load bitmap '%s' from resources! Check .rc file."),
333 name.c_str());
334
335 return false;
336 }
337
338 BITMAP bm;
339 if ( !::GetObject(GetHbitmapOf(*bitmap), sizeof(BITMAP), (LPSTR) &bm) )
340 {
341 wxLogLastError(wxT("GetObject(HBITMAP)"));
342 }
343
344 bitmap->SetWidth(bm.bmWidth);
345 bitmap->SetHeight(bm.bmHeight);
346 bitmap->SetDepth(bm.bmBitsPixel);
347
348 // use 0xc0c0c0 as transparent colour by default
349 bitmap->SetMask(new wxMask(*bitmap, *wxLIGHT_GREY));
350
351 return true;
352}
353
354bool wxBMPFileHandler::LoadFile(wxBitmap *bitmap,
355 const wxString& name, long WXUNUSED(flags),
356 int WXUNUSED(desiredWidth),
357 int WXUNUSED(desiredHeight))
358{
359#if wxUSE_WXDIB
360 wxCHECK_MSG( bitmap, false, _T("NULL bitmap in LoadFile") );
361
362 wxDIB dib(name);
363
364 return dib.IsOk() && bitmap->CopyFromDIB(dib);
365#else
366 return false;
367#endif
368}
369
370bool wxBMPFileHandler::SaveFile(wxBitmap *bitmap,
371 const wxString& name,
372 int WXUNUSED(type),
373 const wxPalette * WXUNUSED(pal))
374{
375#if wxUSE_WXDIB
376 wxCHECK_MSG( bitmap, false, _T("NULL bitmap in SaveFile") );
377
378 wxDIB dib(*bitmap);
379
380 return dib.Save(name);
381#else
382 return false;
383#endif
384}
385
386// ----------------------------------------------------------------------------
387// wxIcon handlers
388// ----------------------------------------------------------------------------
389
390bool wxICOFileHandler::LoadIcon(wxIcon *icon,
391 const wxString& name,
392 long WXUNUSED(flags),
393 int desiredWidth, int desiredHeight)
394{
395 icon->UnRef();
396
397 // actual size
398 wxSize size;
399
400 HICON hicon = NULL;
401
402 // Parse the filename: it may be of the form "filename;n" in order to
403 // specify the nth icon in the file.
404 //
405 // For the moment, ignore the issue of possible semicolons in the
406 // filename.
407 int iconIndex = 0;
408 wxString nameReal(name);
409 wxString strIconIndex = name.AfterLast(wxT(';'));
410 if (strIconIndex != name)
411 {
412 iconIndex = wxAtoi(strIconIndex);
413 nameReal = name.BeforeLast(wxT(';'));
414 }
415
416#if 0
417 // If we don't know what size icon we're looking for,
418 // try to find out what's there.
419 // Unfortunately this doesn't work, because ExtractIconEx
420 // will scale the icon to the 'desired' size, even if that
421 // size of icon isn't explicitly stored. So we would have
422 // to parse the icon file outselves.
423 if ( desiredWidth == -1 &&
424 desiredHeight == -1)
425 {
426 // Try loading a large icon first
427 if ( ::ExtractIconEx(nameReal, iconIndex, &hicon, NULL, 1) == 1)
428 {
429 }
430 // Then try loading a small icon
431 else if ( ::ExtractIconEx(nameReal, iconIndex, NULL, &hicon, 1) == 1)
432 {
433 }
434 }
435 else
436#endif
437 // were we asked for a large icon?
438 if ( desiredWidth == ::GetSystemMetrics(SM_CXICON) &&
439 desiredHeight == ::GetSystemMetrics(SM_CYICON) )
440 {
441 // get the specified large icon from file
442 if ( !::ExtractIconEx(nameReal, iconIndex, &hicon, NULL, 1) )
443 {
444 // it is not an error, but it might still be useful to be informed
445 // about it optionally
446 wxLogTrace(_T("iconload"),
447 _T("No large icons found in the file '%s'."),
448 name.c_str());
449 }
450 }
451 else if ( desiredWidth == ::GetSystemMetrics(SM_CXSMICON) &&
452 desiredHeight == ::GetSystemMetrics(SM_CYSMICON) )
453 {
454 // get the specified small icon from file
455 if ( !::ExtractIconEx(nameReal, iconIndex, NULL, &hicon, 1) )
456 {
457 wxLogTrace(_T("iconload"),
458 _T("No small icons found in the file '%s'."),
459 name.c_str());
460 }
461 }
462 //else: not standard size, load below
463
464#ifndef __WXWINCE__
465 if ( !hicon )
466 {
467 // take any size icon from the file by index
468 hicon = ::ExtractIcon(wxGetInstance(), nameReal, iconIndex);
469 }
470#endif
471
472 if ( !hicon )
473 {
474 wxLogSysError(_T("Failed to load icon from the file '%s'"),
475 name.c_str());
476
477 return false;
478 }
479
480 size = wxGetHiconSize(hicon);
481
482 if ( (desiredWidth != -1 && desiredWidth != size.x) ||
483 (desiredHeight != -1 && desiredHeight != size.y) )
484 {
485 wxLogTrace(_T("iconload"),
486 _T("Returning false from wxICOFileHandler::Load because of the size mismatch: actual (%d, %d), requested (%d, %d)"),
487 size.x, size.y,
488 desiredWidth, desiredHeight);
489
490 ::DestroyIcon(hicon);
491
492 return false;
493 }
494
495 icon->SetHICON((WXHICON)hicon);
496 icon->SetSize(size.x, size.y);
497
498 return icon->Ok();
499}
500
501bool wxICOResourceHandler::LoadIcon(wxIcon *icon,
502 const wxString& name,
503 long WXUNUSED(flags),
504 int desiredWidth, int desiredHeight)
505{
506 HICON hicon;
507
508 // do we need the icon of the specific size or would any icon do?
509 bool hasSize = desiredWidth != -1 || desiredHeight != -1;
510
511 wxASSERT_MSG( !hasSize || (desiredWidth != -1 && desiredHeight != -1),
512 _T("width and height should be either both -1 or not") );
513
514 // try to load the icon from this program first to allow overriding the
515 // standard icons (although why one would want to do it considering that
516 // we already have wxApp::GetStdIcon() is unclear)
517
518 // note that we can't just always call LoadImage() because it seems to do
519 // some icon rescaling internally which results in very ugly 16x16 icons
520 if ( hasSize )
521 {
522 hicon = (HICON)::LoadImage(wxGetInstance(), name, IMAGE_ICON,
523 desiredWidth, desiredHeight,
524 LR_DEFAULTCOLOR);
525 }
526 else
527 {
528 hicon = ::LoadIcon(wxGetInstance(), name);
529 }
530
531 // next check if it's not a standard icon
532#ifndef __WXWINCE__
533 if ( !hicon && !hasSize )
534 {
535 static const struct
536 {
537 const wxChar *name;
538 LPTSTR id;
539 } stdIcons[] =
540 {
541 { wxT("wxICON_QUESTION"), IDI_QUESTION },
542 { wxT("wxICON_WARNING"), IDI_EXCLAMATION },
543 { wxT("wxICON_ERROR"), IDI_HAND },
544 { wxT("wxICON_INFORMATION"), IDI_ASTERISK },
545 };
546
547 for ( size_t nIcon = 0; !hicon && nIcon < WXSIZEOF(stdIcons); nIcon++ )
548 {
549 if ( name == stdIcons[nIcon].name )
550 {
551 hicon = ::LoadIcon((HINSTANCE)NULL, stdIcons[nIcon].id);
552 }
553 }
554 }
555#endif
556
557 wxSize size = wxGetHiconSize(hicon);
558 icon->SetSize(size.x, size.y);
559
560 icon->SetHICON((WXHICON)hicon);
561
562 return icon->Ok();
563}
564
565// ----------------------------------------------------------------------------
566// private functions
567// ----------------------------------------------------------------------------
568
569wxSize wxGetHiconSize(HICON WXUNUSED_IN_WINCE(hicon))
570{
571 // default icon size on this hardware
572 // usually 32x32 but can be other (smaller) on pocket devices
573 wxSize size(::GetSystemMetrics(SM_CXICON), ::GetSystemMetrics(SM_CYICON));
574
575#ifndef __WXWINCE__
576 if ( hicon && wxGetOsVersion() != wxWIN32S )
577 {
578 ICONINFO info;
579 if ( !::GetIconInfo(hicon, &info) )
580 {
581 wxLogLastError(wxT("GetIconInfo"));
582 }
583 else
584 {
585 HBITMAP hbmp = info.hbmMask;
586 if ( hbmp )
587 {
588 BITMAP bm;
589 if ( ::GetObject(hbmp, sizeof(BITMAP), (LPSTR) &bm) )
590 {
591 size = wxSize(bm.bmWidth, bm.bmHeight);
592 }
593
594 ::DeleteObject(info.hbmMask);
595 }
596 if ( info.hbmColor )
597 ::DeleteObject(info.hbmColor);
598 }
599 }
600#endif
601
602 return size;
603}
604
605#endif // __WXMICROWIN__
606