]>
git.saurik.com Git - wxWidgets.git/blob - src/gtk/bitmap.cpp
1 /////////////////////////////////////////////////////////////////////////////
4 // Author: Robert Roebling
6 // Copyright: (c) 1998 Robert Roebling
7 // Licence: wxWindows licence
8 /////////////////////////////////////////////////////////////////////////////
11 #pragma implementation "bitmap.h"
14 #include "wx/bitmap.h"
16 #include "wx/filefn.h"
17 #include "gdk/gdkprivate.h"
20 //-----------------------------------------------------------------------------
22 //-----------------------------------------------------------------------------
24 IMPLEMENT_DYNAMIC_CLASS(wxMask
,wxObject
)
28 m_bitmap
= (GdkBitmap
*) NULL
;
31 wxMask::wxMask( const wxBitmap
& WXUNUSED(bitmap
), const wxColour
& WXUNUSED(colour
) )
35 wxMask::wxMask( const wxBitmap
& WXUNUSED(bitmap
), int WXUNUSED(paletteIndex
) )
39 wxMask::wxMask( const wxBitmap
& WXUNUSED(bitmap
) )
45 if (m_bitmap
) gdk_bitmap_unref( m_bitmap
);
48 GdkBitmap
*wxMask::GetBitmap(void) const
53 //-----------------------------------------------------------------------------
55 //-----------------------------------------------------------------------------
57 class wxBitmapRefData
: public wxObjectRefData
60 wxBitmapRefData(void);
61 ~wxBitmapRefData(void);
72 wxBitmapRefData::wxBitmapRefData(void)
74 m_pixmap
= (GdkPixmap
*) NULL
;
75 m_bitmap
= (GdkBitmap
*) NULL
;
76 m_mask
= (wxMask
*) NULL
;
80 m_palette
= (wxPalette
*) NULL
;
83 wxBitmapRefData::~wxBitmapRefData(void)
85 if (m_pixmap
) gdk_pixmap_unref( m_pixmap
);
86 if (m_bitmap
) gdk_bitmap_unref( m_bitmap
);
87 if (m_mask
) delete m_mask
;
88 if (m_palette
) delete m_palette
;
91 //-----------------------------------------------------------------------------
93 #define M_BMPDATA ((wxBitmapRefData *)m_refData)
95 IMPLEMENT_DYNAMIC_CLASS(wxBitmap
,wxGDIObject
)
97 wxBitmap::wxBitmap(void)
99 if (wxTheBitmapList
) wxTheBitmapList
->AddBitmap(this);
102 wxBitmap::wxBitmap( int width
, int height
, int depth
)
104 wxCHECK_RET( (width
> 0) && (height
> 0), "invalid bitmap size" )
105 wxCHECK_RET( (depth
> 0) || (depth
== -1), "invalid bitmap depth" )
107 m_refData
= new wxBitmapRefData();
109 GdkWindow
*parent
= (GdkWindow
*) &gdk_root_parent
;
111 M_BMPDATA
->m_mask
= (wxMask
*) NULL
;
112 M_BMPDATA
->m_pixmap
= gdk_pixmap_new( parent
, width
, height
, depth
);
113 M_BMPDATA
->m_width
= width
;
114 M_BMPDATA
->m_height
= height
;
115 M_BMPDATA
->m_bpp
= gdk_window_get_visual( parent
)->depth
;
117 if (wxTheBitmapList
) wxTheBitmapList
->AddBitmap(this);
120 wxBitmap::wxBitmap( const char **bits
)
122 wxCHECK_RET( bits
!= NULL
, "invalid bitmap data" )
124 m_refData
= new wxBitmapRefData();
126 GdkBitmap
*mask
= (GdkBitmap
*) NULL
;
127 GdkWindow
*parent
= (GdkWindow
*) &gdk_root_parent
;
129 M_BMPDATA
->m_pixmap
= gdk_pixmap_create_from_xpm_d( parent
, &mask
, NULL
, (gchar
**) bits
);
133 M_BMPDATA
->m_mask
= new wxMask();
134 M_BMPDATA
->m_mask
->m_bitmap
= mask
;
137 gdk_window_get_size( M_BMPDATA
->m_pixmap
, &(M_BMPDATA
->m_width
), &(M_BMPDATA
->m_height
) );
139 M_BMPDATA
->m_bpp
= gdk_window_get_visual( parent
)->depth
; // ?
140 if (wxTheBitmapList
) wxTheBitmapList
->AddBitmap(this);
143 wxBitmap::wxBitmap( char **bits
)
145 wxCHECK_RET( bits
!= NULL
, "invalid bitmap data" )
147 m_refData
= new wxBitmapRefData();
149 GdkBitmap
*mask
= (GdkBitmap
*) NULL
;
150 GdkWindow
*parent
= (GdkWindow
*) &gdk_root_parent
;
152 M_BMPDATA
->m_pixmap
= gdk_pixmap_create_from_xpm_d( parent
, &mask
, NULL
, (gchar
**) bits
);
156 M_BMPDATA
->m_mask
= new wxMask();
157 M_BMPDATA
->m_mask
->m_bitmap
= mask
;
160 gdk_window_get_size( M_BMPDATA
->m_pixmap
, &(M_BMPDATA
->m_width
), &(M_BMPDATA
->m_height
) );
162 M_BMPDATA
->m_bpp
= gdk_window_get_visual( parent
)->depth
; // ?
163 if (wxTheBitmapList
) wxTheBitmapList
->AddBitmap(this);
166 wxBitmap::wxBitmap( const wxBitmap
& bmp
)
170 if (wxTheBitmapList
) wxTheBitmapList
->AddBitmap(this);
173 wxBitmap::wxBitmap( const wxBitmap
* bmp
)
175 if (bmp
) Ref( *bmp
);
177 if (wxTheBitmapList
) wxTheBitmapList
->AddBitmap(this);
180 wxBitmap::wxBitmap( const wxString
&filename
, int type
)
182 LoadFile( filename
, type
);
184 if (wxTheBitmapList
) wxTheBitmapList
->AddBitmap(this);
187 wxBitmap::wxBitmap( const char bits
[], int width
, int height
, int WXUNUSED(depth
))
189 m_refData
= new wxBitmapRefData();
191 M_BMPDATA
->m_mask
= (wxMask
*) NULL
;
192 M_BMPDATA
->m_bitmap
=
193 gdk_bitmap_create_from_data( (GdkWindow
*) &gdk_root_parent
, (gchar
*) bits
, width
, height
);
194 M_BMPDATA
->m_width
= width
;
195 M_BMPDATA
->m_height
= height
;
196 M_BMPDATA
->m_bpp
= 1;
198 if (wxTheBitmapList
) wxTheBitmapList
->AddBitmap(this);
201 wxBitmap::~wxBitmap(void)
203 if (wxTheBitmapList
) wxTheBitmapList
->DeleteObject(this);
206 wxBitmap
& wxBitmap::operator = ( const wxBitmap
& bmp
)
208 if (*this == bmp
) return (*this);
213 bool wxBitmap::operator == ( const wxBitmap
& bmp
)
215 return m_refData
== bmp
.m_refData
;
218 bool wxBitmap::operator != ( const wxBitmap
& bmp
)
220 return m_refData
!= bmp
.m_refData
;
223 bool wxBitmap::Ok(void) const
225 return (m_refData
!= NULL
);
228 int wxBitmap::GetHeight(void) const
230 wxCHECK_MSG( Ok(), -1, "invalid bitmap" );
232 return M_BMPDATA
->m_height
;
235 int wxBitmap::GetWidth(void) const
237 wxCHECK_MSG( Ok(), -1, "invalid bitmap" );
239 return M_BMPDATA
->m_width
;
242 int wxBitmap::GetDepth(void) const
244 wxCHECK_MSG( Ok(), -1, "invalid bitmap" );
246 return M_BMPDATA
->m_bpp
;
249 void wxBitmap::SetHeight( int height
)
253 wxFAIL_MSG( "wxBitmap::SetHeight not implemented" );
255 M_BMPDATA
->m_height
= height
;
258 void wxBitmap::SetWidth( int width
)
262 wxFAIL_MSG( "wxBitmap::SetWidth not implemented" );
264 M_BMPDATA
->m_width
= width
;
267 void wxBitmap::SetDepth( int depth
)
271 wxFAIL_MSG( "wxBitmap::SetDepth not implemented" );
273 M_BMPDATA
->m_bpp
= depth
;
276 wxMask
*wxBitmap::GetMask(void) const
278 wxCHECK_MSG( Ok(), (wxMask
*) NULL
, "invalid bitmap" );
280 return M_BMPDATA
->m_mask
;
283 void wxBitmap::SetMask( wxMask
*mask
)
285 wxCHECK_RET( Ok(), "invalid bitmap" );
287 if (M_BMPDATA
->m_mask
) delete M_BMPDATA
->m_mask
;
289 M_BMPDATA
->m_mask
= mask
;
292 bool wxBitmap::SaveFile( const wxString
&name
, int type
, wxPalette
*WXUNUSED(palette
) )
294 wxCHECK_MSG( Ok(), FALSE
, "invalid bitmap" );
296 if (type
== wxBITMAP_TYPE_PNG
)
298 wxImage image
= ConvertToImage();
299 if (image
.Ok()) return image
.SaveFile( name
, type
);
305 bool wxBitmap::LoadFile( const wxString
&name
, int type
)
309 if (!wxFileExists(name
)) return FALSE
;
311 if (type
== wxBITMAP_TYPE_XPM
)
313 m_refData
= new wxBitmapRefData();
315 GdkBitmap
*mask
= (GdkBitmap
*) NULL
;
316 GdkWindow
*parent
= (GdkWindow
*) &gdk_root_parent
;
318 M_BMPDATA
->m_pixmap
= gdk_pixmap_create_from_xpm( parent
, &mask
, NULL
, name
);
322 M_BMPDATA
->m_mask
= new wxMask();
323 M_BMPDATA
->m_mask
->m_bitmap
= mask
;
326 gdk_window_get_size( M_BMPDATA
->m_pixmap
, &(M_BMPDATA
->m_width
), &(M_BMPDATA
->m_height
) );
327 M_BMPDATA
->m_bpp
= gdk_window_get_visual( parent
)->depth
;
329 else if (type
== wxBITMAP_TYPE_PNG
)
332 image
.LoadFile( name
, type
);
333 if (image
.Ok()) *this = wxBitmap( image
);
335 else if (type
== wxBITMAP_TYPE_BMP
)
338 image
.LoadFile( name
, type
);
339 if (image
.Ok()) *this = wxBitmap( image
);
347 wxPalette
*wxBitmap::GetPalette(void) const
349 if (!Ok()) return (wxPalette
*) NULL
;
351 return M_BMPDATA
->m_palette
;
354 GdkPixmap
*wxBitmap::GetPixmap(void) const
356 wxCHECK_MSG( Ok(), (GdkPixmap
*) NULL
, "invalid bitmap" );
358 return M_BMPDATA
->m_pixmap
;
361 GdkBitmap
*wxBitmap::GetBitmap(void) const
363 wxCHECK_MSG( Ok(), (GdkBitmap
*) NULL
, "invalid bitmap" );
365 return M_BMPDATA
->m_bitmap
;
368 wxBitmap::wxBitmap( const wxImage
&image
)
370 if (wxTheBitmapList
) wxTheBitmapList
->AddBitmap(this);
372 if (!image
.Ok()) return;
374 m_refData
= new wxBitmapRefData();
376 M_BMPDATA
->m_height
= image
.GetHeight();
377 M_BMPDATA
->m_width
= image
.GetWidth();
378 int width
= image
.GetWidth();
379 int height
= image
.GetHeight();
383 GdkImage
*data_image
=
384 gdk_image_new( GDK_IMAGE_FASTEST
, gdk_visual_get_system(), width
, height
);
386 M_BMPDATA
->m_pixmap
=
387 gdk_pixmap_new( (GdkWindow
*)&gdk_root_parent
, width
, height
, -1 );
391 GdkImage
*mask_image
= (GdkImage
*) NULL
;
395 unsigned char *mask_data
= (unsigned char*)malloc( ((width
>> 3)+8) * height
);
397 mask_image
= gdk_image_new_bitmap( gdk_visual_get_system(), mask_data
, width
, height
);
399 M_BMPDATA
->m_mask
= new wxMask();
400 M_BMPDATA
->m_mask
->m_bitmap
= gdk_pixmap_new( (GdkWindow
*)&gdk_root_parent
, width
, height
, 1 );
405 GdkVisual
*visual
= gdk_window_get_visual( M_BMPDATA
->m_pixmap
);
406 if (visual
== NULL
) visual
= gdk_window_get_visual( (GdkWindow
*) &gdk_root_parent
);
407 int bpp
= visual
->depth
;
408 if ((bpp
== 16) && (visual
->red_mask
!= 0xf800)) bpp
= 15;
409 if (bpp
< 8) bpp
= 8;
413 enum byte_order
{ RGB
, RBG
, BRG
, BGR
, GRB
, GBR
};
414 byte_order b_o
= RGB
;
418 GdkVisual
*visual
= gdk_visual_get_system();
419 if ((visual
->red_mask
> visual
->green_mask
) && (visual
->green_mask
> visual
->blue_mask
)) b_o
= RGB
;
420 else if ((visual
->red_mask
> visual
->blue_mask
) && (visual
->blue_mask
> visual
->green_mask
)) b_o
= RGB
;
421 else if ((visual
->blue_mask
> visual
->red_mask
) && (visual
->red_mask
> visual
->green_mask
)) b_o
= BRG
;
422 else if ((visual
->blue_mask
> visual
->green_mask
) && (visual
->green_mask
> visual
->red_mask
)) b_o
= BGR
;
423 else if ((visual
->green_mask
> visual
->red_mask
) && (visual
->red_mask
> visual
->blue_mask
)) b_o
= GRB
;
424 else if ((visual
->green_mask
> visual
->blue_mask
) && (visual
->blue_mask
> visual
->red_mask
)) b_o
= GBR
;
427 int r_mask
= image
.GetMaskRed();
428 int g_mask
= image
.GetMaskGreen();
429 int b_mask
= image
.GetMaskBlue();
431 unsigned char* data
= image
.GetData();
434 for (int y
= 0; y
< height
; y
++)
436 for (int x
= 0; x
< width
; x
++)
447 if ((r
== r_mask
) && (b
== b_mask
) && (g
== g_mask
))
448 gdk_image_put_pixel( mask_image
, x
, y
, 1 );
450 gdk_image_put_pixel( mask_image
, x
, y
, 0 );
457 GdkColormap
*cmap
= gtk_widget_get_default_colormap();
458 GdkColor
*colors
= cmap
->colors
;
459 int max
= 3 * (65536);
462 for (int i
= 0; i
< cmap
->size
; i
++)
464 int rdiff
= (r
<< 8) - colors
[i
].red
;
465 int gdiff
= (g
<< 8) - colors
[i
].green
;
466 int bdiff
= (b
<< 8) - colors
[i
].blue
;
467 int sum
= ABS (rdiff
) + ABS (gdiff
) + ABS (bdiff
);
468 if (sum
< max
) { index
= i
; max
= sum
; }
471 gdk_image_put_pixel( data_image
, x
, y
, index
);
477 guint32 pixel
= ((r
& 0xf8) << 7) | ((g
& 0xf8) << 2) | ((b
& 0xf8) >> 3);
478 gdk_image_put_pixel( data_image
, x
, y
, pixel
);
483 guint32 pixel
= ((r
& 0xf8) << 8) | ((g
& 0xfc) << 3) | ((b
& 0xf8) >> 3);
484 gdk_image_put_pixel( data_image
, x
, y
, pixel
);
493 case RGB
: pixel
= (r
<< 16) | (g
<< 8) | b
; break;
494 case RBG
: pixel
= (r
<< 16) | (b
<< 8) | g
; break;
495 case BRG
: pixel
= (b
<< 16) | (r
<< 8) | g
; break;
496 case BGR
: pixel
= (b
<< 16) | (g
<< 8) | r
; break;
497 case GRB
: pixel
= (g
<< 16) | (r
<< 8) | b
; break;
498 case GBR
: pixel
= (g
<< 16) | (b
<< 8) | r
; break;
500 gdk_image_put_pixel( data_image
, x
, y
, pixel
);
509 GdkGC
*data_gc
= gdk_gc_new( M_BMPDATA
->m_pixmap
);
511 gdk_draw_image( M_BMPDATA
->m_pixmap
, data_gc
, data_image
, 0, 0, 0, 0, width
, height
);
513 gdk_image_destroy( data_image
);
514 gdk_gc_unref( data_gc
);
520 GdkGC
*mask_gc
= gdk_gc_new( M_BMPDATA
->m_mask
->m_bitmap
);
522 gdk_draw_image( M_BMPDATA
->m_mask
->m_bitmap
, mask_gc
, mask_image
, 0, 0, 0, 0, width
, height
);
524 gdk_image_destroy( mask_image
);
525 gdk_gc_unref( mask_gc
);
529 wxImage
wxBitmap::ConvertToImage() const
533 wxCHECK_MSG( Ok(), image
, "invalid bitmap" );
535 GdkImage
*gdk_image
= gdk_image_get( M_BMPDATA
->m_pixmap
, 0, 0, M_BMPDATA
->m_width
, M_BMPDATA
->m_height
);
537 if (!gdk_image
) return image
;
539 image
.Create( M_BMPDATA
->m_width
, M_BMPDATA
->m_height
);
540 char unsigned *data
= image
.GetData();
542 GdkVisual
*visual
= gdk_window_get_visual( M_BMPDATA
->m_pixmap
);
543 if (visual
== NULL
) visual
= gdk_window_get_visual( (GdkWindow
*) &gdk_root_parent
);
544 int bpp
= visual
->depth
;
545 if ((bpp
== 16) && (visual
->red_mask
!= 0xf800)) bpp
= 15;
547 GdkColormap
*cmap
= gtk_widget_get_default_colormap();
550 for (int j
= 0; j
< M_BMPDATA
->m_height
; j
++)
552 for (int i
= 0; i
< M_BMPDATA
->m_width
; i
++)
554 int pixel
= gdk_image_get_pixel( gdk_image
, i
, j
);
557 data
[pos
] = cmap
->colors
[pixel
].red
>> 8;
558 data
[pos
+1] = cmap
->colors
[pixel
].green
>> 8;
559 data
[pos
+2] = cmap
->colors
[pixel
].blue
>> 8;
560 } else if (bpp
== 15)
562 data
[pos
] = (pixel
>> 7) & 0xf8;
563 data
[pos
+1] = (pixel
>> 2) & 0xf8;
564 data
[pos
+2] = (pixel
<< 3) & 0xf8;
565 } else if (bpp
== 16)
567 data
[pos
] = (pixel
>> 8) & 0xf8;
568 data
[pos
+1] = (pixel
>> 3) & 0xfc;
569 data
[pos
+2] = (pixel
<< 3) & 0xf8;
572 data
[pos
] = (pixel
>> 16) & 0xff;
573 data
[pos
+1] = (pixel
>> 8) & 0xff;
574 data
[pos
+2] = pixel
& 0xff;
581 gdk_image_destroy( gdk_image
);