]> git.saurik.com Git - wxWidgets.git/blob - src/common/artprov.cpp
ensure that the copies of the bitmap passed to wxMemoryDC ctor are not modified when...
[wxWidgets.git] / src / common / artprov.cpp
1 /////////////////////////////////////////////////////////////////////////////
2 // Name: src/common/artprov.cpp
3 // Purpose: wxArtProvider class
4 // Author: Vaclav Slavik
5 // Modified by:
6 // Created: 18/03/2002
7 // RCS-ID: $Id$
8 // Copyright: (c) Vaclav Slavik
9 // Licence: wxWindows licence
10 /////////////////////////////////////////////////////////////////////////////
11
12 // ---------------------------------------------------------------------------
13 // headers
14 // ---------------------------------------------------------------------------
15
16 // For compilers that support precompilation, includes "wx.h".
17 #include "wx/wxprec.h"
18
19 #if defined(__BORLANDC__)
20 #pragma hdrstop
21 #endif
22
23 #include "wx/artprov.h"
24
25 #ifndef WX_PRECOMP
26 #include "wx/list.h"
27 #include "wx/log.h"
28 #include "wx/hashmap.h"
29 #include "wx/image.h"
30 #include "wx/module.h"
31 #endif
32
33 // ===========================================================================
34 // implementation
35 // ===========================================================================
36
37 #include "wx/listimpl.cpp"
38 WX_DECLARE_LIST(wxArtProvider, wxArtProvidersList);
39 WX_DEFINE_LIST(wxArtProvidersList)
40
41 // ----------------------------------------------------------------------------
42 // Cache class - stores already requested bitmaps
43 // ----------------------------------------------------------------------------
44
45 WX_DECLARE_EXPORTED_STRING_HASH_MAP(wxBitmap, wxArtProviderBitmapsHash);
46 WX_DECLARE_EXPORTED_STRING_HASH_MAP(wxIconBundle, wxArtProviderIconBundlesHash);
47
48 class WXDLLEXPORT wxArtProviderCache
49 {
50 public:
51 bool GetBitmap(const wxString& full_id, wxBitmap* bmp);
52 void PutBitmap(const wxString& full_id, const wxBitmap& bmp)
53 { m_bitmapsHash[full_id] = bmp; }
54
55 bool GetIconBundle(const wxString& full_id, wxIconBundle* bmp);
56 void PutIconBundle(const wxString& full_id, const wxIconBundle& iconbundle)
57 { m_iconBundlesHash[full_id] = iconbundle; }
58
59 void Clear();
60
61 static wxString ConstructHashID(const wxArtID& id,
62 const wxArtClient& client,
63 const wxSize& size);
64
65 static wxString ConstructHashID(const wxArtID& id,
66 const wxArtClient& client);
67
68 private:
69 wxArtProviderBitmapsHash m_bitmapsHash; // cache of wxBitmaps
70 wxArtProviderIconBundlesHash m_iconBundlesHash; // cache of wxIconBundles
71 };
72
73 bool wxArtProviderCache::GetBitmap(const wxString& full_id, wxBitmap* bmp)
74 {
75 wxArtProviderBitmapsHash::iterator entry = m_bitmapsHash.find(full_id);
76 if ( entry == m_bitmapsHash.end() )
77 {
78 return false;
79 }
80 else
81 {
82 *bmp = entry->second;
83 return true;
84 }
85 }
86
87 bool wxArtProviderCache::GetIconBundle(const wxString& full_id, wxIconBundle* bmp)
88 {
89 wxArtProviderIconBundlesHash::iterator entry = m_iconBundlesHash.find(full_id);
90 if ( entry == m_iconBundlesHash.end() )
91 {
92 return false;
93 }
94 else
95 {
96 *bmp = entry->second;
97 return true;
98 }
99 }
100
101 void wxArtProviderCache::Clear()
102 {
103 m_bitmapsHash.clear();
104 m_iconBundlesHash.clear();
105 }
106
107 /* static */ wxString
108 wxArtProviderCache::ConstructHashID(const wxArtID& id,
109 const wxArtClient& client)
110 {
111 return id + _T('-') + client;
112 }
113
114
115 /* static */ wxString
116 wxArtProviderCache::ConstructHashID(const wxArtID& id,
117 const wxArtClient& client,
118 const wxSize& size)
119 {
120 return ConstructHashID(id, client) + _T('-') +
121 wxString::Format(_T("%d-%d"), size.x, size.y);
122 }
123
124 // ============================================================================
125 // wxArtProvider class
126 // ============================================================================
127
128 IMPLEMENT_ABSTRACT_CLASS(wxArtProvider, wxObject)
129
130 wxArtProvidersList *wxArtProvider::sm_providers = NULL;
131 wxArtProviderCache *wxArtProvider::sm_cache = NULL;
132
133 // ----------------------------------------------------------------------------
134 // wxArtProvider ctors/dtor
135 // ----------------------------------------------------------------------------
136
137 wxArtProvider::~wxArtProvider()
138 {
139 Remove(this);
140 }
141
142 // ----------------------------------------------------------------------------
143 // wxArtProvider operations on provider stack
144 // ----------------------------------------------------------------------------
145
146 /*static*/ void wxArtProvider::CommonAddingProvider()
147 {
148 if ( !sm_providers )
149 {
150 sm_providers = new wxArtProvidersList;
151 sm_cache = new wxArtProviderCache;
152 }
153
154 sm_cache->Clear();
155 }
156
157 /*static*/ void wxArtProvider::Push(wxArtProvider *provider)
158 {
159 CommonAddingProvider();
160 sm_providers->Insert(provider);
161 }
162
163 /*static*/ void wxArtProvider::PushBack(wxArtProvider *provider)
164 {
165 CommonAddingProvider();
166 sm_providers->Append(provider);
167 }
168
169 /*static*/ bool wxArtProvider::Pop()
170 {
171 wxCHECK_MSG( sm_providers, false, _T("no wxArtProvider exists") );
172 wxCHECK_MSG( !sm_providers->empty(), false, _T("wxArtProviders stack is empty") );
173
174 delete sm_providers->GetFirst()->GetData();
175 sm_cache->Clear();
176 return true;
177 }
178
179 /*static*/ bool wxArtProvider::Remove(wxArtProvider *provider)
180 {
181 wxCHECK_MSG( sm_providers, false, _T("no wxArtProvider exists") );
182
183 if ( sm_providers->DeleteObject(provider) )
184 {
185 sm_cache->Clear();
186 return true;
187 }
188
189 return false;
190 }
191
192 /*static*/ bool wxArtProvider::Delete(wxArtProvider *provider)
193 {
194 // provider will remove itself from the stack in its dtor
195 delete provider;
196
197 return true;
198 }
199
200 /*static*/ void wxArtProvider::CleanUpProviders()
201 {
202 if ( sm_providers )
203 {
204 while ( !sm_providers->empty() )
205 delete *sm_providers->begin();
206
207 delete sm_providers;
208 sm_providers = NULL;
209
210 delete sm_cache;
211 sm_cache = NULL;
212 }
213 }
214
215 // ----------------------------------------------------------------------------
216 // wxArtProvider: retrieving bitmaps/icons
217 // ----------------------------------------------------------------------------
218
219 /*static*/ wxBitmap wxArtProvider::GetBitmap(const wxArtID& id,
220 const wxArtClient& client,
221 const wxSize& size)
222 {
223 // safety-check against writing client,id,size instead of id,client,size:
224 wxASSERT_MSG( client.Last() == _T('C'), _T("invalid 'client' parameter") );
225
226 wxCHECK_MSG( sm_providers, wxNullBitmap, _T("no wxArtProvider exists") );
227
228 wxString hashId = wxArtProviderCache::ConstructHashID(id, client, size);
229
230 wxBitmap bmp;
231 if ( !sm_cache->GetBitmap(hashId, &bmp) )
232 {
233 for (wxArtProvidersList::compatibility_iterator node = sm_providers->GetFirst();
234 node; node = node->GetNext())
235 {
236 bmp = node->GetData()->CreateBitmap(id, client, size);
237 if ( bmp.Ok() )
238 {
239 #if wxUSE_IMAGE && (!defined(__WXMSW__) || wxUSE_WXDIB)
240 if ( size != wxDefaultSize &&
241 (bmp.GetWidth() != size.x || bmp.GetHeight() != size.y) )
242 {
243 wxImage img = bmp.ConvertToImage();
244 img.Rescale(size.x, size.y);
245 bmp = wxBitmap(img);
246 }
247 #endif
248 break;
249 }
250 // We could try the IconBundles here and convert what we find
251 // to a bitmap.
252 }
253 sm_cache->PutBitmap(hashId, bmp);
254 }
255
256 return bmp;
257 }
258
259 /*static*/ wxIconBundle wxArtProvider::GetIconBundle(const wxArtID& id, const wxArtClient& client)
260 {
261 // safety-check against writing client,id,size instead of id,client,size:
262 wxASSERT_MSG( client.Last() == _T('C'), _T("invalid 'client' parameter") );
263
264 wxCHECK_MSG( sm_providers, wxNullIconBundle, _T("no wxArtProvider exists") );
265
266 wxString hashId = wxArtProviderCache::ConstructHashID(id, client);
267
268 wxIconBundle iconbundle;
269 if ( !sm_cache->GetIconBundle(hashId, &iconbundle) )
270 {
271 for (wxArtProvidersList::compatibility_iterator node = sm_providers->GetFirst();
272 node; node = node->GetNext())
273 {
274 iconbundle = node->GetData()->CreateIconBundle(id, client);
275 if ( iconbundle.IsOk() )
276 break;
277 }
278
279 sm_cache->PutIconBundle(hashId, iconbundle);
280 }
281
282 return iconbundle;
283 }
284
285 /*static*/ wxIcon wxArtProvider::GetIcon(const wxArtID& id,
286 const wxArtClient& client,
287 const wxSize& size)
288 {
289 wxCHECK_MSG( sm_providers, wxNullIcon, _T("no wxArtProvider exists") );
290
291 // First look for an appropriate icon bundle - this will give us the best icon
292 wxIconBundle iconBundle = GetIconBundle(id, client);
293 if ( iconBundle.IsOk() )
294 return iconBundle.GetIcon(size);
295
296 // If there is no icon bundle then look for a bitmap
297 wxBitmap bmp = GetBitmap(id, client, size);
298 if ( !bmp.Ok() )
299 return wxNullIcon;
300
301 wxIcon icon;
302 icon.CopyFromBitmap(bmp);
303 return icon;
304 }
305
306 /* static */
307 wxIcon wxArtProvider::GetMessageBoxIcon(int flags)
308 {
309 wxIcon icon;
310 switch ( flags & wxICON_MASK )
311 {
312 default:
313 wxFAIL_MSG(_T("incorrect message box icon flags"));
314 // fall through
315
316 case wxICON_ERROR:
317 icon = wxArtProvider::GetIcon(wxART_ERROR, wxART_MESSAGE_BOX);
318 break;
319
320 case wxICON_INFORMATION:
321 icon = wxArtProvider::GetIcon(wxART_INFORMATION, wxART_MESSAGE_BOX);
322 break;
323
324 case wxICON_WARNING:
325 icon = wxArtProvider::GetIcon(wxART_WARNING, wxART_MESSAGE_BOX);
326 break;
327
328 case wxICON_QUESTION:
329 icon = wxArtProvider::GetIcon(wxART_QUESTION, wxART_MESSAGE_BOX);
330 break;
331 }
332
333 return icon;
334 }
335
336 #if defined(__WXGTK20__) && !defined(__WXUNIVERSAL__)
337 #include <gtk/gtk.h>
338 extern GtkIconSize wxArtClientToIconSize(const wxArtClient& client);
339 #endif // defined(__WXGTK20__) && !defined(__WXUNIVERSAL__)
340
341 /*static*/ wxSize wxArtProvider::GetSizeHint(const wxArtClient& client,
342 bool platform_dependent)
343 {
344 if (!platform_dependent)
345 {
346 wxArtProvidersList::compatibility_iterator node = sm_providers->GetFirst();
347 if (node)
348 return node->GetData()->DoGetSizeHint(client);
349 }
350
351 // else return platform dependent size
352
353 #if defined(__WXGTK20__) && !defined(__WXUNIVERSAL__)
354 // Gtk has specific sizes for each client, see artgtk.cpp
355 GtkIconSize gtk_size = wxArtClientToIconSize(client);
356 // no size hints for this client
357 if (gtk_size == GTK_ICON_SIZE_INVALID)
358 return wxDefaultSize;
359 gint width, height;
360 gtk_icon_size_lookup( gtk_size, &width, &height);
361 return wxSize(width, height);
362 #else // !GTK+ 2
363 // NB: These size hints may have to be adjusted per platform
364 if (client == wxART_TOOLBAR)
365 return wxSize(16, 15);
366 else if (client == wxART_MENU)
367 return wxSize(16, 15);
368 else if (client == wxART_FRAME_ICON)
369 return wxSize(16, 15);
370 else if (client == wxART_CMN_DIALOG || client == wxART_MESSAGE_BOX)
371 return wxSize(32, 32);
372 else if (client == wxART_HELP_BROWSER)
373 return wxSize(16, 15);
374 else if (client == wxART_BUTTON)
375 return wxSize(16, 15);
376 else // wxART_OTHER or perhaps a user's client, no specified size
377 return wxDefaultSize;
378 #endif // GTK+ 2/else
379 }
380
381 /* static */
382 bool wxArtProvider::HasNativeProvider()
383 {
384 #ifdef __WXGTK20__
385 return true;
386 #else
387 return false;
388 #endif
389 }
390
391 // ----------------------------------------------------------------------------
392 // deprecated wxArtProvider methods
393 // ----------------------------------------------------------------------------
394
395 #if WXWIN_COMPATIBILITY_2_6
396
397 /* static */ void wxArtProvider::PushProvider(wxArtProvider *provider)
398 {
399 Push(provider);
400 }
401
402 /* static */ void wxArtProvider::InsertProvider(wxArtProvider *provider)
403 {
404 Insert(provider);
405 }
406
407 /* static */ bool wxArtProvider::PopProvider()
408 {
409 return Pop();
410 }
411
412 /* static */ bool wxArtProvider::RemoveProvider(wxArtProvider *provider)
413 {
414 // RemoveProvider() used to delete the provider being removed so this is
415 // not a typo, we must call Delete() and not Remove() here
416 return Delete(provider);
417 }
418
419 #endif // WXWIN_COMPATIBILITY_2_6
420
421 #if WXWIN_COMPATIBILITY_2_8
422 /* static */ void wxArtProvider::Insert(wxArtProvider *provider)
423 {
424 PushBack(provider);
425 }
426 #endif // WXWIN_COMPATIBILITY_2_8
427
428 // ============================================================================
429 // wxArtProviderModule
430 // ============================================================================
431
432 class wxArtProviderModule: public wxModule
433 {
434 public:
435 bool OnInit()
436 {
437 wxArtProvider::InitStdProvider();
438 wxArtProvider::InitNativeProvider();
439 return true;
440 }
441 void OnExit()
442 {
443 wxArtProvider::CleanUpProviders();
444 }
445
446 DECLARE_DYNAMIC_CLASS(wxArtProviderModule)
447 };
448
449 IMPLEMENT_DYNAMIC_CLASS(wxArtProviderModule, wxModule)