]> git.saurik.com Git - wxWidgets.git/blame - src/common/artprov.cpp
Optimize pixels rotation in wxImage::Rotate90().
[wxWidgets.git] / src / common / artprov.cpp
CommitLineData
2aca4a98 1/////////////////////////////////////////////////////////////////////////////
8ecff181 2// Name: src/common/artprov.cpp
2aca4a98
VS
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
65571936 9// Licence: wxWindows licence
2aca4a98
VS
10/////////////////////////////////////////////////////////////////////////////
11
12// ---------------------------------------------------------------------------
13// headers
14// ---------------------------------------------------------------------------
15
2aca4a98
VS
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
02761f6c
WS
23#include "wx/artprov.h"
24
2aca4a98 25#ifndef WX_PRECOMP
2aca4a98 26 #include "wx/list.h"
8ecff181 27 #include "wx/log.h"
df69528b 28 #include "wx/hashmap.h"
155ecd4c 29 #include "wx/image.h"
02761f6c 30 #include "wx/module.h"
2aca4a98
VS
31#endif
32
2aca4a98
VS
33// ===========================================================================
34// implementation
35// ===========================================================================
36
37#include "wx/listimpl.cpp"
38WX_DECLARE_LIST(wxArtProvider, wxArtProvidersList);
259c43f6 39WX_DEFINE_LIST(wxArtProvidersList)
2aca4a98
VS
40
41// ----------------------------------------------------------------------------
42// Cache class - stores already requested bitmaps
43// ----------------------------------------------------------------------------
44
3f5c62f9 45WX_DECLARE_EXPORTED_STRING_HASH_MAP(wxBitmap, wxArtProviderBitmapsHash);
52734360 46WX_DECLARE_EXPORTED_STRING_HASH_MAP(wxIconBundle, wxArtProviderIconBundlesHash);
2aca4a98
VS
47
48class WXDLLEXPORT wxArtProviderCache
49{
50public:
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
52734360
VZ
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
2aca4a98 59 void Clear();
9e7ed2b0 60
57b0987b
VS
61 static wxString ConstructHashID(const wxArtID& id,
62 const wxArtClient& client,
2aca4a98
VS
63 const wxSize& size);
64
52734360
VZ
65 static wxString ConstructHashID(const wxArtID& id,
66 const wxArtClient& client);
67
2aca4a98 68private:
52734360
VZ
69 wxArtProviderBitmapsHash m_bitmapsHash; // cache of wxBitmaps
70 wxArtProviderIconBundlesHash m_iconBundlesHash; // cache of wxIconBundles
2aca4a98
VS
71};
72
73bool 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 {
4629016d 78 return false;
2aca4a98
VS
79 }
80 else
81 {
82 *bmp = entry->second;
4629016d 83 return true;
2aca4a98
VS
84 }
85}
86
52734360
VZ
87bool 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
2aca4a98
VS
101void wxArtProviderCache::Clear()
102{
103 m_bitmapsHash.clear();
52734360 104 m_iconBundlesHash.clear();
2aca4a98
VS
105}
106
52734360
VZ
107/* static */ wxString
108wxArtProviderCache::ConstructHashID(const wxArtID& id,
109 const wxArtClient& client)
2aca4a98 110{
9a83f860 111 return id + wxT('-') + client;
2aca4a98
VS
112}
113
114
52734360
VZ
115/* static */ wxString
116wxArtProviderCache::ConstructHashID(const wxArtID& id,
117 const wxArtClient& client,
118 const wxSize& size)
119{
9a83f860
VZ
120 return ConstructHashID(id, client) + wxT('-') +
121 wxString::Format(wxT("%d-%d"), size.x, size.y);
52734360
VZ
122}
123
571d2e0f 124// ============================================================================
2aca4a98 125// wxArtProvider class
571d2e0f 126// ============================================================================
2aca4a98
VS
127
128IMPLEMENT_ABSTRACT_CLASS(wxArtProvider, wxObject)
129
130wxArtProvidersList *wxArtProvider::sm_providers = NULL;
131wxArtProviderCache *wxArtProvider::sm_cache = NULL;
132
571d2e0f
VZ
133// ----------------------------------------------------------------------------
134// wxArtProvider ctors/dtor
135// ----------------------------------------------------------------------------
136
137wxArtProvider::~wxArtProvider()
138{
139 Remove(this);
140}
141
142// ----------------------------------------------------------------------------
143// wxArtProvider operations on provider stack
144// ----------------------------------------------------------------------------
145
dd7d379e 146/*static*/ void wxArtProvider::CommonAddingProvider()
2aca4a98
VS
147{
148 if ( !sm_providers )
149 {
150 sm_providers = new wxArtProvidersList;
2aca4a98
VS
151 sm_cache = new wxArtProviderCache;
152 }
153
c3357a03 154 sm_cache->Clear();
2aca4a98
VS
155}
156
571d2e0f 157/*static*/ void wxArtProvider::Push(wxArtProvider *provider)
dd7d379e
VS
158{
159 CommonAddingProvider();
160 sm_providers->Insert(provider);
161}
162
14440cc6 163/*static*/ void wxArtProvider::PushBack(wxArtProvider *provider)
dd7d379e
VS
164{
165 CommonAddingProvider();
166 sm_providers->Append(provider);
167}
168
571d2e0f 169/*static*/ bool wxArtProvider::Pop()
2aca4a98 170{
9a83f860
VZ
171 wxCHECK_MSG( sm_providers, false, wxT("no wxArtProvider exists") );
172 wxCHECK_MSG( !sm_providers->empty(), false, wxT("wxArtProviders stack is empty") );
2aca4a98 173
222ed1d6 174 delete sm_providers->GetFirst()->GetData();
2aca4a98 175 sm_cache->Clear();
4629016d 176 return true;
2aca4a98
VS
177}
178
571d2e0f 179/*static*/ bool wxArtProvider::Remove(wxArtProvider *provider)
2aca4a98 180{
9a83f860 181 wxCHECK_MSG( sm_providers, false, wxT("no wxArtProvider exists") );
2aca4a98
VS
182
183 if ( sm_providers->DeleteObject(provider) )
184 {
185 sm_cache->Clear();
4629016d 186 return true;
2aca4a98 187 }
9e7ed2b0 188
4629016d 189 return false;
2aca4a98
VS
190}
191
571d2e0f
VZ
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
2aca4a98
VS
200/*static*/ void wxArtProvider::CleanUpProviders()
201{
571d2e0f
VZ
202 if ( sm_providers )
203 {
204 while ( !sm_providers->empty() )
205 delete *sm_providers->begin();
206
5276b0a5
VZ
207 wxDELETE(sm_providers);
208 wxDELETE(sm_cache);
571d2e0f 209 }
2aca4a98
VS
210}
211
571d2e0f
VZ
212// ----------------------------------------------------------------------------
213// wxArtProvider: retrieving bitmaps/icons
214// ----------------------------------------------------------------------------
215
57b0987b
VS
216/*static*/ wxBitmap wxArtProvider::GetBitmap(const wxArtID& id,
217 const wxArtClient& client,
e53a95bc 218 const wxSize& size)
2aca4a98 219{
57b0987b 220 // safety-check against writing client,id,size instead of id,client,size:
9a83f860 221 wxASSERT_MSG( client.Last() == wxT('C'), wxT("invalid 'client' parameter") );
57b0987b 222
9a83f860 223 wxCHECK_MSG( sm_providers, wxNullBitmap, wxT("no wxArtProvider exists") );
2aca4a98 224
e53a95bc 225 wxString hashId = wxArtProviderCache::ConstructHashID(id, client, size);
2aca4a98
VS
226
227 wxBitmap bmp;
228 if ( !sm_cache->GetBitmap(hashId, &bmp) )
229 {
222ed1d6 230 for (wxArtProvidersList::compatibility_iterator node = sm_providers->GetFirst();
2aca4a98
VS
231 node; node = node->GetNext())
232 {
e53a95bc 233 bmp = node->GetData()->CreateBitmap(id, client, size);
2aca4a98 234 if ( bmp.Ok() )
b5c2a334
VS
235 break;
236 }
237
1d3dfc57 238 wxSize sizeNeeded = size;
b5c2a334
VS
239 if ( !bmp.Ok() )
240 {
241 // no bitmap created -- as a fallback, try if we can find desired
242 // icon in a bundle
243 wxIconBundle iconBundle = DoGetIconBundle(id, client);
244 if ( iconBundle.IsOk() )
3242feb1 245 {
1d3dfc57
VZ
246 if ( sizeNeeded == wxDefaultSize )
247 sizeNeeded = GetNativeSizeHint(client);
248
249 wxIcon icon(iconBundle.GetIcon(sizeNeeded));
b5c2a334 250 if ( icon.IsOk() )
1d3dfc57
VZ
251 {
252 // this icon may be not of the correct size, it will be
253 // rescaled below in such case
b5c2a334 254 bmp.CopyFromIcon(icon);
1d3dfc57 255 }
b5c2a334
VS
256 }
257 }
258
1d3dfc57 259 // if we didn't get the correct size, resize the bitmap
64c288fa 260#if wxUSE_IMAGE && (!defined(__WXMSW__) || wxUSE_WXDIB)
1d3dfc57
VZ
261 if ( bmp.IsOk() && sizeNeeded != wxDefaultSize )
262 {
263 if ( bmp.GetSize() != sizeNeeded )
b5c2a334
VS
264 {
265 wxImage img = bmp.ConvertToImage();
1d3dfc57 266 img.Rescale(sizeNeeded.x, sizeNeeded.y);
b5c2a334 267 bmp = wxBitmap(img);
3242feb1 268 }
2aca4a98 269 }
1d3dfc57 270#endif // wxUSE_IMAGE
b5c2a334 271
2aca4a98 272 sm_cache->PutBitmap(hashId, bmp);
b5c2a334 273 }
2aca4a98
VS
274
275 return bmp;
276}
277
b5c2a334
VS
278/*static*/
279wxIconBundle wxArtProvider::GetIconBundle(const wxArtID& id, const wxArtClient& client)
280{
281 wxIconBundle iconbundle(DoGetIconBundle(id, client));
282
283 if ( iconbundle.IsOk() )
284 {
285 return iconbundle;
286 }
287 else
288 {
289 // fall back to single-icon bundle
290 return wxIconBundle(GetIcon(id, client));
291 }
292}
293
294/*static*/
295wxIconBundle wxArtProvider::DoGetIconBundle(const wxArtID& id, const wxArtClient& client)
52734360
VZ
296{
297 // safety-check against writing client,id,size instead of id,client,size:
9a83f860 298 wxASSERT_MSG( client.Last() == wxT('C'), wxT("invalid 'client' parameter") );
52734360 299
9a83f860 300 wxCHECK_MSG( sm_providers, wxNullIconBundle, wxT("no wxArtProvider exists") );
52734360
VZ
301
302 wxString hashId = wxArtProviderCache::ConstructHashID(id, client);
303
304 wxIconBundle iconbundle;
305 if ( !sm_cache->GetIconBundle(hashId, &iconbundle) )
306 {
307 for (wxArtProvidersList::compatibility_iterator node = sm_providers->GetFirst();
308 node; node = node->GetNext())
309 {
310 iconbundle = node->GetData()->CreateIconBundle(id, client);
311 if ( iconbundle.IsOk() )
312 break;
313 }
314
315 sm_cache->PutIconBundle(hashId, iconbundle);
316 }
317
318 return iconbundle;
319}
320
57b0987b
VS
321/*static*/ wxIcon wxArtProvider::GetIcon(const wxArtID& id,
322 const wxArtClient& client,
2aca4a98
VS
323 const wxSize& size)
324{
57b0987b 325 wxBitmap bmp = GetBitmap(id, client, size);
b5c2a334
VS
326
327 if ( !bmp.IsOk() )
2aca4a98 328 return wxNullIcon;
2aca4a98 329
25dd6997
VZ
330 wxIcon icon;
331 icon.CopyFromBitmap(bmp);
332 return icon;
333}
2aca4a98 334
1970409e 335/* static */
39f6b6ba 336wxArtID wxArtProvider::GetMessageBoxIconId(int flags)
1970409e 337{
1970409e
VZ
338 switch ( flags & wxICON_MASK )
339 {
340 default:
9a83f860 341 wxFAIL_MSG(wxT("incorrect message box icon flags"));
1970409e
VZ
342 // fall through
343
344 case wxICON_ERROR:
39f6b6ba 345 return wxART_ERROR;
1970409e
VZ
346
347 case wxICON_INFORMATION:
39f6b6ba 348 return wxART_INFORMATION;
1970409e
VZ
349
350 case wxICON_WARNING:
39f6b6ba 351 return wxART_WARNING;
1970409e
VZ
352
353 case wxICON_QUESTION:
39f6b6ba 354 return wxART_QUESTION;
1970409e 355 }
1970409e
VZ
356}
357
e53a95bc 358/*static*/ wxSize wxArtProvider::GetSizeHint(const wxArtClient& client,
b737ad10
RR
359 bool platform_dependent)
360{
361 if (!platform_dependent)
362 {
363 wxArtProvidersList::compatibility_iterator node = sm_providers->GetFirst();
364 if (node)
e53a95bc
RR
365 return node->GetData()->DoGetSizeHint(client);
366 }
f6d43eda 367
a158acac
VS
368 return GetNativeSizeHint(client);
369}
370
371#ifndef wxHAS_NATIVE_ART_PROVIDER_IMPL
372/*static*/
6f23f80b 373wxSize wxArtProvider::GetNativeSizeHint(const wxArtClient& WXUNUSED(client))
a158acac
VS
374{
375 // rather than returning some arbitrary value that doesn't make much
376 // sense (as 2.8 used to do), tell the caller that we don't have a clue:
377 return wxDefaultSize;
378}
379
380/*static*/
381void wxArtProvider::InitNativeProvider()
382{
b737ad10 383}
a158acac
VS
384#endif // !wxHAS_NATIVE_ART_PROVIDER_IMPL
385
b737ad10 386
14440cc6
VS
387/* static */
388bool wxArtProvider::HasNativeProvider()
389{
390#ifdef __WXGTK20__
391 return true;
392#else
393 return false;
394#endif
395}
396
571d2e0f
VZ
397// ----------------------------------------------------------------------------
398// deprecated wxArtProvider methods
399// ----------------------------------------------------------------------------
400
401#if WXWIN_COMPATIBILITY_2_6
402
403/* static */ void wxArtProvider::PushProvider(wxArtProvider *provider)
404{
405 Push(provider);
406}
407
408/* static */ void wxArtProvider::InsertProvider(wxArtProvider *provider)
409{
410 Insert(provider);
411}
412
413/* static */ bool wxArtProvider::PopProvider()
414{
415 return Pop();
416}
417
418/* static */ bool wxArtProvider::RemoveProvider(wxArtProvider *provider)
419{
420 // RemoveProvider() used to delete the provider being removed so this is
421 // not a typo, we must call Delete() and not Remove() here
422 return Delete(provider);
423}
424
425#endif // WXWIN_COMPATIBILITY_2_6
426
14440cc6
VS
427#if WXWIN_COMPATIBILITY_2_8
428/* static */ void wxArtProvider::Insert(wxArtProvider *provider)
429{
430 PushBack(provider);
431}
432#endif // WXWIN_COMPATIBILITY_2_8
433
571d2e0f
VZ
434// ============================================================================
435// wxArtProviderModule
436// ============================================================================
2aca4a98
VS
437
438class wxArtProviderModule: public wxModule
439{
440public:
2b5f62a0
VZ
441 bool OnInit()
442 {
443 wxArtProvider::InitStdProvider();
7bebedd8 444 wxArtProvider::InitNativeProvider();
4629016d 445 return true;
2b5f62a0
VZ
446 }
447 void OnExit()
448 {
449 wxArtProvider::CleanUpProviders();
450 }
2aca4a98
VS
451
452 DECLARE_DYNAMIC_CLASS(wxArtProviderModule)
453};
454
455IMPLEMENT_DYNAMIC_CLASS(wxArtProviderModule, wxModule)