1 /////////////////////////////////////////////////////////////////////////////
2 // Name: src/dfb/bitmap.cpp
3 // Purpose: wxBitmap implementation
4 // Author: Vaclav Slavik
7 // Copyright: (c) 2006 REA Elektronik GmbH
8 // Licence: wxWindows licence
9 /////////////////////////////////////////////////////////////////////////////
11 // For compilers that support precompilation, includes "wx.h".
12 #include "wx/wxprec.h"
23 #include "wx/bitmap.h"
24 #include "wx/colour.h"
27 #warning "move this to common"
28 #include "wx/xpmdecod.h"
30 #include "wx/dfb/private.h"
32 //-----------------------------------------------------------------------------
34 //-----------------------------------------------------------------------------
36 // Convert wxColour into it's quantized value in lower-precision
37 // pixel format (needed for masking by colour).
38 static wxColour
wxQuantizeColour(const wxColour
& clr
, const wxBitmap
& bmp
)
41 pixel_format_t
*pf
= bmp
.GetMGLbitmap_t()->pf
;
43 if ( pf
->redAdjust
== 0 && pf
->greenAdjust
== 0 && pf
->blueAdjust
== 0 )
46 return wxColour((unsigned char)((clr
.Red() >> pf
->redAdjust
) << pf
->redAdjust
),
47 (unsigned char)((clr
.Green() >> pf
->greenAdjust
) << pf
->greenAdjust
),
48 (unsigned char)((clr
.Blue() >> pf
->blueAdjust
) << pf
->blueAdjust
));
53 //-----------------------------------------------------------------------------
55 //-----------------------------------------------------------------------------
57 IMPLEMENT_DYNAMIC_CLASS(wxMask
, wxObject
)
59 wxMask
::wxMask() : m_bitmap(NULL
)
63 wxMask
::wxMask(const wxBitmap
& bitmap
)
69 wxMask
::wxMask(const wxBitmap
& bitmap
, const wxColour
& colour
)
72 Create(bitmap
, colour
);
76 wxMask
::wxMask(const wxBitmap
& bitmap
, int paletteIndex
)
79 Create(bitmap
, paletteIndex
);
81 #endif // wxUSE_PALETTE
83 wxMask
::wxMask(const wxMask
& mask
)
85 m_bitmap
= mask
.m_bitmap ?
new wxBitmap(*mask
.m_bitmap
) : NULL
;
93 #warning "move this to common code"
94 bool wxMask
::Create(const wxBitmap
& bitmap
, const wxColour
& colour
)
99 wxColour
clr(wxQuantizeColour(colour
, bitmap
));
101 wxImage
imgSrc(bitmap
.ConvertToImage());
102 imgSrc
.SetMask(false);
103 wxImage
image(imgSrc
.ConvertToMono(clr
.Red(), clr
.Green(), clr
.Blue()));
107 m_bitmap
= new wxBitmap(image
, 1);
109 return m_bitmap
->Ok();
113 bool wxMask
::Create(const wxBitmap
& bitmap
, int paletteIndex
)
116 wxPalette
*pal
= bitmap
.GetPalette();
118 wxCHECK_MSG( pal
, false, wxT("Cannot create mask from bitmap without palette") );
120 pal
->GetRGB(paletteIndex
, &r
, &g
, &b
);
122 return Create(bitmap
, wxColour(r
, g
, b
));
124 #endif // wxUSE_PALETTE
126 bool wxMask
::Create(const wxBitmap
& bitmap
)
131 wxCHECK_MSG( bitmap
.Ok(), false, wxT("Invalid bitmap") );
132 wxCHECK_MSG( bitmap
.GetDepth() == 1, false, wxT("Cannot create mask from colour bitmap") );
134 m_bitmap
= new wxBitmap(bitmap
);
138 const wxBitmap
& wxMask
::GetBitmap() const
140 return m_bitmap ?
*m_bitmap
: wxNullBitmap
;
144 //-----------------------------------------------------------------------------
146 //-----------------------------------------------------------------------------
148 class wxBitmapRefData
: public wxObjectRefData
159 wxBitmapRefData(const wxBitmapRefData
& data
)
161 if ( data
.m_surface
)
162 m_surface
= data
.m_surface
->Clone();
164 m_mask
= data
.m_mask ?
new wxMask(*data
.m_mask
) : NULL
;
166 m_palette
= data
.m_palette ?
new wxPalette(*data
.m_palette
) : NULL
;
178 wxIDirectFBSurfacePtr m_surface
;
181 wxPalette
*m_palette
;
185 #define M_BITMAP ((wxBitmapRefData *)m_refData)
187 //-----------------------------------------------------------------------------
189 //-----------------------------------------------------------------------------
191 IMPLEMENT_ABSTRACT_CLASS(wxBitmapHandler
, wxObject
)
192 IMPLEMENT_DYNAMIC_CLASS(wxBitmap
, wxBitmapBase
)
194 wxBitmap
::wxBitmap(int width
, int height
, int depth
)
196 Create(width
, height
, depth
);
199 bool wxBitmap
::Create(const wxIDirectFBSurfacePtr
& surface
)
203 wxCHECK_MSG( surface
, false, _T("invalid surface") );
205 m_refData
= new wxBitmapRefData();
206 M_BITMAP
->m_surface
= surface
;
210 bool wxBitmap
::Create(int width
, int height
, int depth
)
214 wxCHECK_MSG( width
> 0 && height
> 0, false, wxT("invalid bitmap size") );
216 DFBSurfaceDescription desc
;
217 desc
.flags
= (DFBSurfaceDescriptionFlags
)(
218 DSDESC_CAPS
| DSDESC_WIDTH
| DSDESC_HEIGHT
);
219 desc
.caps
= DSCAPS_NONE
;
221 desc
.height
= height
;
223 return Create(wxIDirectFB
::Get()->CreateSurface(&desc
));
226 #warning "FIXME: move this to common code"
227 bool wxBitmap
::CreateFromXpm(const char **bits
)
229 wxCHECK_MSG( bits
!= NULL
, false, wxT("invalid bitmap data") );
231 #if wxUSE_IMAGE && wxUSE_XPM
232 wxXPMDecoder decoder
;
233 wxImage img
= decoder
.ReadData(bits
);
234 wxCHECK_MSG( img
.Ok(), false, wxT("invalid bitmap data") );
236 *this = wxBitmap(img
);
240 wxFAIL_MSG( _T("creating bitmaps from XPMs not supported") );
242 #endif // wxUSE_IMAGE && wxUSE_XPM
246 wxBitmap
::wxBitmap(const wxImage
& image
, int depth
)
248 wxCHECK_RET( image
.Ok(), wxT("invalid image") );
251 wxImage wxBitmap
::ConvertToImage() const
253 wxCHECK_MSG( Ok(), wxNullImage
, wxT("invalid bitmap") );
255 return wxNullImage
; // FIXME
257 #endif // wxUSE_IMAGE
259 wxBitmap
::wxBitmap(const wxString
&filename
, wxBitmapType type
)
261 LoadFile(filename
, type
);
264 wxBitmap
::wxBitmap(const char bits
[], int width
, int height
, int depth
)
266 wxCHECK_RET( depth
== 1, wxT("can only create mono bitmap from XBM data") );
269 bool wxBitmap
::Ok() const
271 return (m_refData
!= NULL
&& M_BITMAP
->m_surface
);
274 bool wxBitmap
::operator==(const wxBitmap
& bmp
) const
276 // FIXME: is this the right way to compare bitmaps?
277 return (m_refData
== bmp
.m_refData
);
280 int wxBitmap
::GetHeight() const
282 wxCHECK_MSG( Ok(), -1, wxT("invalid bitmap") );
285 M_BITMAP
->m_surface
->GetSize(NULL
, &h
);
289 int wxBitmap
::GetWidth() const
291 wxCHECK_MSG( Ok(), -1, wxT("invalid bitmap") );
294 M_BITMAP
->m_surface
->GetSize(&w
, NULL
);
298 int wxBitmap
::GetDepth() const
300 wxCHECK_MSG( Ok(), -1, wxT("invalid bitmap") );
302 return M_BITMAP
->m_surface
->GetDepth();
305 wxMask
*wxBitmap
::GetMask() const
307 wxCHECK_MSG( Ok(), NULL
, wxT("invalid bitmap") );
309 return M_BITMAP
->m_mask
;
312 void wxBitmap
::SetMask(wxMask
*mask
)
314 wxCHECK_RET( Ok(), wxT("invalid bitmap") );
316 delete M_BITMAP
->m_mask
;
317 M_BITMAP
->m_mask
= mask
;
320 bool wxBitmap
::CopyFromIcon(const wxIcon
& icon
)
322 *this = *((wxBitmap
*)(&icon
));
326 wxBitmap wxBitmap
::GetSubBitmap(const wxRect
& rect
) const
329 rect
.x
>= 0 && rect
.y
>= 0 &&
330 rect
.x
+rect
.width
<= GetWidth() &&
331 rect
.y
+rect
.height
<= GetHeight(),
333 wxT("invalid bitmap or bitmap region") );
335 // NB: DirectFB subsurfaces share the same pixels buffer, so we must
336 // clone the obtained subsurface
337 DFBRectangle r
= { rect
.x
, rect
.y
, rect
.width
, rect
.height
};
338 return wxBitmap(M_BITMAP
->m_surface
->GetSubSurface(&r
)->Clone());
341 #warning "to common code"
342 bool wxBitmap
::LoadFile(const wxString
&name
, wxBitmapType type
)
346 wxBitmapHandler
*handler
= FindHandler(type
);
348 if ( handler
== NULL
)
351 if ( !image
.LoadFile(name
, type
) || !image
.Ok() )
353 wxLogError("no bitmap handler for type %d defined.", type
);
358 *this = wxBitmap(image
);
363 m_refData
= new wxBitmapRefData();
365 return handler
->LoadFile(this, name
, type
, -1, -1);
368 #warning "to common code"
369 bool wxBitmap
::SaveFile(const wxString
& filename
, wxBitmapType type
, const wxPalette
*palette
) const
371 wxCHECK_MSG( Ok(), false, wxT("invalid bitmap") );
373 wxBitmapHandler
*handler
= FindHandler(type
);
375 if ( handler
== NULL
)
377 wxImage image
= ConvertToImage();
380 image
.SetPalette(*palette
);
381 #endif // wxUSE_PALETTE
384 return image
.SaveFile(filename
, type
);
387 wxLogError("no bitmap handler for type %d defined.", type
);
392 return handler
->SaveFile(this, filename
, type
, palette
);
396 wxPalette
*wxBitmap
::GetPalette() const
398 wxCHECK_MSG( Ok(), NULL
, wxT("invalid bitmap") );
400 return M_BITMAP
->m_palette
;
403 void wxBitmap
::SetPalette(const wxPalette
& palette
)
405 wxCHECK_RET( Ok(), wxT("invalid bitmap") );
406 wxCHECK_RET( GetDepth() > 1 && GetDepth() <= 8, wxT("cannot set palette for bitmap of this depth") );
408 delete M_BITMAP
->m_palette
;
409 M_BITMAP
->m_palette
= NULL
;
411 if ( !palette
.Ok() ) return;
413 M_BITMAP
->m_palette
= new wxPalette(palette
);
415 #endif // wxUSE_PALETTE
417 void wxBitmap
::SetHeight(int height
)
423 void wxBitmap
::SetWidth(int width
)
429 void wxBitmap
::SetDepth(int depth
)
435 wxIDirectFBSurfacePtr wxBitmap
::GetDirectFBSurface() const
437 wxCHECK_MSG( Ok(), NULL
, wxT("invalid bitmap") );
439 return M_BITMAP
->m_surface
;
442 wxObjectRefData
*wxBitmap
::CreateRefData() const
444 return new wxBitmapRefData
;
447 wxObjectRefData
*wxBitmap
::CloneRefData(const wxObjectRefData
*data
) const
449 return new wxBitmapRefData(*(wxBitmapRefData
*)data
);
454 void wxBitmap
::InitStandardHandlers()
456 // not wxBitmap handlers, we rely on wxImage