1 /////////////////////////////////////////////////////////////////////////////
4 // Author: Robert Roebling
6 // Copyright: (c) Robert Roebling
7 // Licence: wxWindows licence
8 /////////////////////////////////////////////////////////////////////////////
11 #pragma implementation "image.h"
14 // For compilers that support precompilation, includes "wx.h".
15 #include "wx/wxprec.h"
22 #include "wx/bitmap.h"
25 #include "../png/png.h"
26 #include "wx/filefn.h"
28 //-----------------------------------------------------------------------------
30 //-----------------------------------------------------------------------------
32 class wxImageRefData
: public wxObjectRefData
37 ~wxImageRefData(void);
41 unsigned char *m_data
;
43 unsigned char m_maskRed
,m_maskGreen
,m_maskBlue
;
47 wxImageRefData::wxImageRefData(void)
51 m_data
= (unsigned char*) NULL
;
59 wxImageRefData::~wxImageRefData(void)
61 if (m_data
) free( m_data
);
64 wxList
wxImage::sm_handlers
;
66 //-----------------------------------------------------------------------------
68 #define M_IMGDATA ((wxImageRefData *)m_refData)
70 #if !USE_SHARED_LIBRARIES
71 IMPLEMENT_DYNAMIC_CLASS(wxImage
, wxObject
)
78 wxImage::wxImage( int width
, int height
)
80 Create( width
, height
);
83 wxImage::wxImage( const wxString
& name
, long type
)
85 LoadFile( name
, type
);
88 wxImage::wxImage( const wxImage
& image
)
93 wxImage::wxImage( const wxImage
* image
)
95 if (image
) Ref(*image
);
98 void wxImage::Create( int width
, int height
)
100 m_refData
= new wxImageRefData();
102 M_IMGDATA
->m_data
= (unsigned char *) malloc( width
*height
*3 );
103 if (M_IMGDATA
->m_data
)
105 for (int l
= 0; l
< width
*height
*3; l
++) M_IMGDATA
->m_data
[l
] = 0;
107 M_IMGDATA
->m_width
= width
;
108 M_IMGDATA
->m_height
= height
;
109 M_IMGDATA
->m_ok
= TRUE
;
117 void wxImage::Destroy()
122 wxImage
wxImage::Scale( int width
, int height
)
126 wxCHECK_MSG( Ok(), image
, "invlaid image" );
128 wxCHECK_MSG( (width
> 0) && (height
> 0), image
, "invalid image size" );
130 image
.Create( width
, height
);
132 char unsigned *data
= image
.GetData();
134 wxCHECK_MSG( data
, image
, "unable to create image" );
136 if (M_IMGDATA
->m_hasMask
)
137 image
.SetMaskColour( M_IMGDATA
->m_maskRed
, M_IMGDATA
->m_maskGreen
, M_IMGDATA
->m_maskBlue
);
139 double xscale
= (double)width
/ (double)M_IMGDATA
->m_width
;
140 double yscale
= (double)height
/ (double)M_IMGDATA
->m_height
;
142 for (int j
= 0; j
< height
; j
++)
144 for (int i
= 0; i
< width
; i
++)
146 int new_pos
= 3*(j
*width
+ i
);
147 int old_pos
= 3*((long)(j
/yscale
)*M_IMGDATA
->m_width
+ (long)(i
/xscale
));
148 data
[ new_pos
] = M_IMGDATA
->m_data
[ old_pos
];
149 data
[ new_pos
+1 ] = M_IMGDATA
->m_data
[ old_pos
+1 ];
150 data
[ new_pos
+2 ] = M_IMGDATA
->m_data
[ old_pos
+2 ];
157 void wxImage::SetRGB( int x
, int y
, unsigned char r
, unsigned char g
, unsigned char b
)
159 wxCHECK_RET( Ok(), "invalid image" );
161 int w
= M_IMGDATA
->m_width
;
162 int h
= M_IMGDATA
->m_height
;
164 wxCHECK_RET( (x
>=0) && (y
>=0) && (x
<w
) && (y
<h
), "invalid image index" );
166 long pos
= (y
* w
+ x
) * 3;
168 M_IMGDATA
->m_data
[ pos
] = r
;
169 M_IMGDATA
->m_data
[ pos
+1 ] = g
;
170 M_IMGDATA
->m_data
[ pos
+2 ] = b
;
173 unsigned char wxImage::GetRed( int x
, int y
)
175 wxCHECK_MSG( Ok(), 0, "invalid image" );
177 int w
= M_IMGDATA
->m_width
;
178 int h
= M_IMGDATA
->m_height
;
180 wxCHECK_MSG( (x
>=0) && (y
>=0) && (x
<w
) && (y
<h
), 0, "invalid image index" );
182 long pos
= (y
* w
+ x
) * 3;
184 return M_IMGDATA
->m_data
[pos
];
187 unsigned char wxImage::GetGreen( int x
, int y
)
189 wxCHECK_MSG( Ok(), 0, "invalid image" );
191 int w
= M_IMGDATA
->m_width
;
192 int h
= M_IMGDATA
->m_height
;
194 wxCHECK_MSG( (x
>=0) && (y
>=0) && (x
<w
) && (y
<h
), 0, "invalid image index" );
196 long pos
= (y
* w
+ x
) * 3;
198 return M_IMGDATA
->m_data
[pos
+1];
201 unsigned char wxImage::GetBlue( int x
, int y
)
203 wxCHECK_MSG( Ok(), 0, "invalid image" );
205 int w
= M_IMGDATA
->m_width
;
206 int h
= M_IMGDATA
->m_height
;
208 wxCHECK_MSG( (x
>=0) && (y
>=0) && (x
<w
) && (y
<h
), 0, "invalid image index" );
210 long pos
= (y
* w
+ x
) * 3;
212 return M_IMGDATA
->m_data
[pos
+2];
215 bool wxImage::Ok() const
217 return (M_IMGDATA
&& M_IMGDATA
->m_ok
);
220 char unsigned *wxImage::GetData() const
222 wxCHECK_MSG( Ok(), (char unsigned *)NULL
, "invalid image" );
224 return M_IMGDATA
->m_data
;
227 void wxImage::SetData( char unsigned *WXUNUSED(data
) )
229 wxCHECK_RET( Ok(), "invalid image" );
232 void wxImage::SetMaskColour( unsigned char r
, unsigned char g
, unsigned char b
)
234 wxCHECK_RET( Ok(), "invalid image" );
236 M_IMGDATA
->m_maskRed
= r
;
237 M_IMGDATA
->m_maskGreen
= g
;
238 M_IMGDATA
->m_maskBlue
= b
;
239 M_IMGDATA
->m_hasMask
= TRUE
;
242 unsigned char wxImage::GetMaskRed() const
244 wxCHECK_MSG( Ok(), 0, "invalid image" );
246 return M_IMGDATA
->m_maskRed
;
249 unsigned char wxImage::GetMaskGreen() const
251 wxCHECK_MSG( Ok(), 0, "invalid image" );
253 return M_IMGDATA
->m_maskGreen
;
256 unsigned char wxImage::GetMaskBlue() const
258 wxCHECK_MSG( Ok(), 0, "invalid image" );
260 return M_IMGDATA
->m_maskBlue
;
263 void wxImage::SetMask( bool mask
)
265 wxCHECK_RET( Ok(), "invalid image" );
267 M_IMGDATA
->m_hasMask
= mask
;
270 bool wxImage::HasMask() const
272 wxCHECK_MSG( Ok(), FALSE
, "invalid image" );
274 return M_IMGDATA
->m_hasMask
;
277 int wxImage::GetWidth() const
279 wxCHECK_MSG( Ok(), 0, "invalid image" );
281 return M_IMGDATA
->m_width
;
284 int wxImage::GetHeight() const
286 wxCHECK_MSG( Ok(), 0, "invalid image" );
288 return M_IMGDATA
->m_height
;
291 bool wxImage::LoadFile( const wxString
& filename
, long type
)
295 if (!wxFileExists(filename
))
297 wxLogWarning( "Image file does not exist." );
302 m_refData
= new wxImageRefData
;
304 wxImageHandler
*handler
= FindHandler(type
);
308 wxLogWarning( "No image handler for type %d defined.", type
);
313 return handler
->LoadFile( this, filename
);
316 bool wxImage::SaveFile( const wxString
& filename
, int type
)
318 wxCHECK_MSG( Ok(), FALSE
, "invalid image" );
320 wxImageHandler
*handler
= FindHandler(type
);
324 wxLogWarning( "No image handler for type %d defined.", type
);
329 return handler
->SaveFile( this, filename
);
332 void wxImage::AddHandler( wxImageHandler
*handler
)
334 sm_handlers
.Append( handler
);
337 void wxImage::InsertHandler( wxImageHandler
*handler
)
339 sm_handlers
.Insert( handler
);
342 bool wxImage::RemoveHandler( const wxString
& name
)
344 wxImageHandler
*handler
= FindHandler(name
);
347 sm_handlers
.DeleteObject(handler
);
354 wxImageHandler
*wxImage::FindHandler( const wxString
& name
)
356 wxNode
*node
= sm_handlers
.First();
359 wxImageHandler
*handler
= (wxImageHandler
*)node
->Data();
360 if (handler
->GetName() == name
) return handler
;
363 return (wxImageHandler
*)NULL
;
366 wxImageHandler
*wxImage::FindHandler( const wxString
& extension
, long bitmapType
)
368 wxNode
*node
= sm_handlers
.First();
371 wxImageHandler
*handler
= (wxImageHandler
*)node
->Data();
372 if ( handler
->GetExtension() == extension
&&
373 (bitmapType
== -1 || handler
->GetType() == bitmapType
) )
377 return (wxImageHandler
*)NULL
;
380 wxImageHandler
*wxImage::FindHandler( long bitmapType
)
382 wxNode
*node
= sm_handlers
.First();
385 wxImageHandler
*handler
= (wxImageHandler
*)node
->Data();
386 if (handler
->GetType() == bitmapType
) return handler
;
392 void wxImage::InitStandardHandlers()
394 AddHandler( new wxBMPHandler
);
395 AddHandler( new wxPNGHandler
);
398 void wxImage::CleanUpHandlers()
400 wxNode
*node
= sm_handlers
.First();
403 wxImageHandler
*handler
= (wxImageHandler
*)node
->Data();
404 wxNode
*next
= node
->Next();
411 //-----------------------------------------------------------------------------
413 //-----------------------------------------------------------------------------
415 #if !USE_SHARED_LIBRARIES
416 IMPLEMENT_DYNAMIC_CLASS(wxImageHandler
,wxObject
)
419 bool wxImageHandler::LoadFile( wxImage
*WXUNUSED(image
), const wxString
& WXUNUSED(name
) )
424 bool wxImageHandler::SaveFile( wxImage
*WXUNUSED(image
), const wxString
& WXUNUSED(name
) )
429 //-----------------------------------------------------------------------------
431 //-----------------------------------------------------------------------------
433 #if !USE_SHARED_LIBRARIES
434 IMPLEMENT_DYNAMIC_CLASS(wxPNGHandler
,wxImageHandler
)
437 bool wxPNGHandler::LoadFile( wxImage
*image
, const wxString
& name
)
442 unsigned char *ptr
, **lines
, *ptr2
;
443 int transp
,bit_depth
,color_type
,interlace_type
;
444 png_uint_32 width
, height
;
450 png_ptr
= png_create_read_struct( PNG_LIBPNG_VER_STRING
, NULL
, NULL
, NULL
);
451 if (!png_ptr
) return FALSE
;
453 info_ptr
= png_create_info_struct( png_ptr
);
456 png_destroy_read_struct( &png_ptr
, NULL
, NULL
);
460 if (setjmp(png_ptr
->jmpbuf
))
462 png_destroy_read_struct( &png_ptr
, &info_ptr
, NULL
);
466 if (info_ptr
->color_type
== PNG_COLOR_TYPE_RGB_ALPHA
)
468 png_destroy_read_struct( &png_ptr
, &info_ptr
, NULL
);
472 f
= fopen( name
, "rb" );
473 png_init_io( png_ptr
, f
);
475 png_read_info( png_ptr
, info_ptr
);
476 png_get_IHDR( png_ptr
, info_ptr
, &width
, &height
, &bit_depth
, &color_type
, &interlace_type
, NULL
, NULL
);
478 if (color_type
== PNG_COLOR_TYPE_PALETTE
) png_set_expand( png_ptr
);
480 png_set_strip_16( png_ptr
);
481 png_set_packing( png_ptr
);
482 if (png_get_valid( png_ptr
, info_ptr
, PNG_INFO_tRNS
)) png_set_expand( png_ptr
);
483 png_set_filler( png_ptr
, 0xff, PNG_FILLER_AFTER
);
485 image
->Create( width
, height
);
489 png_destroy_read_struct( &png_ptr
, &info_ptr
, NULL
);
493 lines
= (unsigned char **)malloc( height
* sizeof(unsigned char *) );
497 png_destroy_read_struct( &png_ptr
, &info_ptr
, NULL
);
501 for (i
= 0; i
< height
; i
++)
503 if ((lines
[i
] = (unsigned char *)malloc(width
* (sizeof(unsigned char) * 4))) == NULL
)
506 for (unsigned int n
= 0; n
< i
; n
++) free( lines
[n
] );
508 png_destroy_read_struct( &png_ptr
, &info_ptr
, NULL
);
513 png_read_image( png_ptr
, lines
);
514 png_destroy_read_struct( &png_ptr
, &info_ptr
, NULL
);
515 ptr
= image
->GetData();
516 if ((color_type
== PNG_COLOR_TYPE_GRAY
) ||
517 (color_type
== PNG_COLOR_TYPE_GRAY_ALPHA
))
519 for (unsigned int y
= 0; y
< height
; y
++)
522 for (unsigned int x
= 0; x
< width
; x
++)
524 unsigned char r
= *ptr2
++;
525 unsigned char a
= *ptr2
++;
544 for (unsigned int y
= 0; y
< height
; y
++)
547 for (unsigned int x
= 0; x
< width
; x
++)
549 unsigned char r
= *ptr2
++;
550 unsigned char g
= *ptr2
++;
551 unsigned char b
= *ptr2
++;
552 unsigned char a
= *ptr2
++;
562 if ((r
== 255) && (g
== 0) && (b
== 255)) r
= 254;
570 for (i
= 0; i
< height
; i
++) free( lines
[i
] );
573 image
->SetMaskColour( 255, 0, 255 );
575 image
->SetMask( FALSE
);
581 bool wxPNGHandler::SaveFile( wxImage
*image
, const wxString
& name
)
583 FILE *f
= fopen( name
, "wb" );
586 png_structp png_ptr
= png_create_write_struct( PNG_LIBPNG_VER_STRING
, NULL
, NULL
, NULL
);
593 png_infop info_ptr
= png_create_info_struct(png_ptr
);
594 if (info_ptr
== NULL
)
597 png_destroy_write_struct( &png_ptr
, (png_infopp
)NULL
);
601 if (setjmp(png_ptr
->jmpbuf
))
604 png_destroy_write_struct( &png_ptr
, (png_infopp
)NULL
);
608 png_init_io( png_ptr
, f
);
609 png_set_IHDR( png_ptr
, info_ptr
, image
->GetWidth(), image
->GetHeight(), 8,
610 PNG_COLOR_TYPE_RGB_ALPHA
, PNG_INTERLACE_NONE
,
611 PNG_COMPRESSION_TYPE_BASE
, PNG_FILTER_TYPE_BASE
);
618 png_set_sBIT( png_ptr
, info_ptr
, &sig_bit
);
619 png_write_info( png_ptr
, info_ptr
);
620 png_set_shift( png_ptr
, &sig_bit
);
621 png_set_packing( png_ptr
);
623 unsigned char *data
= (unsigned char *)malloc( image
->GetWidth()*4 );
627 png_destroy_write_struct( &png_ptr
, (png_infopp
)NULL
);
631 for (int y
= 0; y
< image
->GetHeight(); y
++)
633 unsigned char *ptr
= image
->GetData() + (y
* image
->GetWidth() * 3);
634 for (int x
= 0; x
< image
->GetWidth(); x
++)
636 data
[(x
<< 2) + 0] = *ptr
++;
637 data
[(x
<< 2) + 1] = *ptr
++;
638 data
[(x
<< 2) + 2] = *ptr
++;
639 if ((data
[(x
<< 2) + 0] == image
->GetMaskRed()) &&
640 (data
[(x
<< 2) + 1] == image
->GetMaskGreen()) &&
641 (data
[(x
<< 2) + 2] == image
->GetMaskBlue()))
642 data
[(x
<< 2) + 3] = 0;
644 data
[(x
<< 2) + 3] = 255;
646 png_bytep row_ptr
= data
;
647 png_write_rows( png_ptr
, &row_ptr
, 1 );
650 png_write_end( png_ptr
, info_ptr
);
651 png_destroy_write_struct( &png_ptr
, (png_infopp
)NULL
);
658 //-----------------------------------------------------------------------------
660 //-----------------------------------------------------------------------------
662 #if !USE_SHARED_LIBRARIES
663 IMPLEMENT_DYNAMIC_CLASS(wxBMPHandler
,wxImageHandler
)
666 bool wxBMPHandler::LoadFile( wxImage
*image
, const wxString
& name
)
669 unsigned char *data
, *ptr
;
670 int done
, i
, bpp
, planes
, comp
, ncolors
, line
, column
,
671 linesize
, linepos
, rshift
= 0, gshift
= 0, bshift
= 0;
674 long int dbuf
[4], dword
, rmask
= 0, gmask
= 0, bmask
= 0, offset
,
679 unsigned char r
, g
, b
;
689 #define BI_BITFIELDS 3
694 file
= fopen(name
, "r");
700 * Reading the bmp header
703 fread(&bbuf
, 1, 2, file
);
705 fread(dbuf
, 4, 4, file
);
710 fread(dbuf
, 4, 2, file
);
711 int width
= (int)dbuf
[0];
712 int height
= (int)dbuf
[1];
715 fprintf(stderr
, "IMLIB ERROR: Image width > 32767 pixels for file\n");
721 fprintf(stderr
, "IMLIB ERROR: Image height > 32767 pixels for file\n");
725 fread(&word
, 2, 1, file
);
727 fread(&word
, 2, 1, file
);
729 if (bpp
!= 1 && bpp
!= 4 && bpp
!= 8 && bpp
&& 16 && bpp
!= 24 && bpp
!= 32)
731 fprintf(stderr
, "IMLIB ERROR: unknown bitdepth in file\n");
735 fread(dbuf
, 4, 4, file
);
737 if (comp
!= BI_RGB
&& comp
!= BI_RLE4
&& comp
!= BI_RLE8
&& comp
!= BI_BITFIELDS
)
739 fprintf(stderr
, "IMLIB ERROR: unknown encoding in Windows BMP file\n");
743 fread(dbuf
, 4, 2, file
);
744 ncolors
= (int)dbuf
[0];
747 /* some more sanity checks */
748 if (((comp
== BI_RLE4
) && (bpp
!= 4)) || ((comp
== BI_RLE8
) && (bpp
!= 8)) || ((comp
== BI_BITFIELDS
) && (bpp
!= 16 && bpp
!= 32)))
750 fprintf(stderr
, "IMLIB ERROR: encoding of BMP doesn't match bitdepth\n");
756 cmap
= (struct _cmap
*)malloc(sizeof(struct _cmap
) * ncolors
);
760 fprintf(stderr
, "IMLIB ERROR: Cannot allocate RAM for color map in BMP file\n");
768 image
->Create( width
, height
);
769 ptr
= image
->GetData();
772 fprintf(stderr
, "IMLIB ERROR: Cannot allocate RAM for RGB data in file\n");
780 * Reading the palette, if it exists.
782 if (bpp
< 16 && ncolors
!= 0)
784 for (i
= 0; i
< ncolors
; i
++)
786 fread(bbuf
, 1, 4, file
);
792 else if (bpp
== 16 || bpp
== 32)
794 if (comp
== BI_BITFIELDS
)
798 fread(dbuf
, 4, 3, file
);
802 /* find shift amount.. ugly, but i can't think of a better way */
803 for (bit
= 0; bit
< bpp
; bit
++)
805 if (bmask
& (1 << bit
))
807 if (gmask
& (1 << bit
))
809 if (rmask
& (1 << bit
))
834 * REading the image data
836 fseek(file
, offset
, SEEK_SET
);
839 /* set the whole image to the background color */
840 if (bpp
< 16 && (comp
== BI_RLE4
|| comp
== BI_RLE8
))
842 for (i
= 0; i
< width
* height
; i
++)
852 #define poffset (line * width * 3 + column * 3)
855 * BMPs are stored upside down... hmmmmmmmmmm....
858 linesize
= ((width
* bpp
+ 31) / 32) * 4;
859 for (line
= (height
- 1); line
>= 0; line
--)
862 for (column
= 0; column
< width
;)
874 for (bit
= 0; bit
< 8; bit
++)
876 index
= ((aByte
& (0x80 >> bit
)) ? 1 : 0);
877 ptr
[poffset
] = cmap
[index
].r
;
878 ptr
[poffset
+ 1] = cmap
[index
].g
;
879 ptr
[poffset
+ 2] = cmap
[index
].b
;
887 fprintf(stderr
, "can't deal with 4bit encoded yet.\n");
896 for (nibble
= 0; nibble
< 2; nibble
++)
898 index
= ((aByte
& (0xF0 >> nibble
* 4)) >> (!nibble
* 4));
901 ptr
[poffset
] = cmap
[index
].r
;
902 ptr
[poffset
+ 1] = cmap
[index
].g
;
903 ptr
[poffset
+ 2] = cmap
[index
].b
;
920 /* column = width; */
931 linepos
= column
* bpp
/ 8;
937 int absolute
= aByte
;
939 for (i
= 0; i
< absolute
; i
++)
943 ptr
[poffset
] = cmap
[aByte
].r
;
944 ptr
[poffset
+ 1] = cmap
[aByte
].g
;
945 ptr
[poffset
+ 2] = cmap
[aByte
].b
;
954 for (i
= 0; i
< first
; i
++)
956 ptr
[poffset
] = cmap
[aByte
].r
;
957 ptr
[poffset
+ 1] = cmap
[aByte
].g
;
958 ptr
[poffset
+ 2] = cmap
[aByte
].b
;
966 ptr
[poffset
] = cmap
[aByte
].r
;
967 ptr
[poffset
+ 1] = cmap
[aByte
].g
;
968 ptr
[poffset
+ 2] = cmap
[aByte
].b
;
976 linepos
+= fread(&bbuf
, 1, 3, file
);
977 ptr
[poffset
] = (unsigned char)bbuf
[2];
978 ptr
[poffset
+ 1] = (unsigned char)bbuf
[1];
979 ptr
[poffset
+ 2] = (unsigned char)bbuf
[0];
986 linepos
+= fread(&word
, 2, 1, file
);
987 temp
= (word
& rmask
) >> rshift
;
989 temp
= (word
& gmask
) >> gshift
;
990 ptr
[poffset
+ 1] = temp
;
991 temp
= (word
& bmask
) >> gshift
;
992 ptr
[poffset
+ 2] = temp
;
999 linepos
+= fread(&dword
, 4, 1, file
);
1000 temp
= (dword
& rmask
) >> rshift
;
1001 ptr
[poffset
] = temp
;
1002 temp
= (dword
& gmask
) >> gshift
;
1003 ptr
[poffset
+ 1] = temp
;
1004 temp
= (dword
& bmask
) >> bshift
;
1005 ptr
[poffset
+ 2] = temp
;
1009 while ((linepos
< linesize
) && (comp
!= 1) && (comp
!= 2))
1011 int temp
= fread(&aByte
, 1, 1, file
);
1018 if (cmap
) free(cmap
);
1020 image
->SetMask( FALSE
);
1028 wxBitmap
wxImage::ConvertToBitmap() const
1032 wxCHECK_MSG( Ok(), bitmap
, "invalid image" );
1033 int width
= GetWidth();
1034 int height
= GetHeight();
1035 bitmap
.SetWidth( width
);
1036 bitmap
.SetHeight( height
);
1037 bitmap
.SetDepth( wxDisplayDepth() );
1039 int headersize
= sizeof(BITMAPINFOHEADER
);
1040 LPBITMAPINFO lpDIBh
= (BITMAPINFO
*) malloc( headersize
);
1041 wxCHECK_MSG( lpDIBh
, bitmap
, "could not allocate memory for DIB header" );
1043 // Fill in the DIB header
1044 lpDIBh
->bmiHeader
.biSize
= headersize
;
1045 lpDIBh
->bmiHeader
.biWidth
= width
;
1046 lpDIBh
->bmiHeader
.biHeight
= -height
;
1047 lpDIBh
->bmiHeader
.biSizeImage
= width
* height
* 3;
1049 lpDIBh
->bmiHeader
.biPlanes
= 1;
1050 lpDIBh
->bmiHeader
.biBitCount
= 24;
1051 lpDIBh
->bmiHeader
.biCompression
= BI_RGB
;
1052 lpDIBh
->bmiHeader
.biClrUsed
= 0;
1054 // These seem not needed for our purpose here.
1055 // lpDIBh->bmiHeader.biClrImportant = 0;
1056 // lpDIBh->bmiHeader.biXPelsPerMeter = 0;
1057 // lpDIBh->bmiHeader.biYPelsPerMeter = 0;
1059 unsigned char *lpBits
= (unsigned char *) malloc( width
*height
*3 );
1062 wxFAIL_MSG( "could not allocate memory for DIB" );
1067 unsigned char *data
= GetData();
1069 unsigned char *ptdata
= data
, *ptbits
= lpBits
;
1070 for( int i
=0; i
<width
*height
; i
++ )
1072 *(ptbits
++) = *(ptdata
+2);
1073 *(ptbits
++) = *(ptdata
+1);
1074 *(ptbits
++) = *(ptdata
);
1078 HDC hdc
= ::GetDC(NULL
);
1081 hbitmap
= CreateDIBitmap( hdc
, &(lpDIBh
->bmiHeader
), CBM_INIT
, lpBits
, lpDIBh
, DIB_RGB_COLORS
);
1083 // The above line is equivalent to the following two lines.
1084 // hbitmap = ::CreateCompatibleBitmap( hdc, width, height );
1085 // ::SetDIBits( hdc, hbitmap, 0, height, lpBits, lpDIBh, DIB_RGB_COLORS);
1086 // or the following lines
1087 // hbitmap = ::CreateCompatibleBitmap( hdc, width, height );
1088 // HDC memdc = ::CreateCompatibleDC( hdc );
1089 // ::SelectObject( memdc, hbitmap);
1090 // ::SetDIBitsToDevice( memdc, 0, 0, width, height,
1091 // 0, 0, 0, height, (void *)lpBits, lpDIBh, DIB_RGB_COLORS);
1092 // ::SelectObject( memdc, 0 );
1093 // ::DeleteDC( memdc );
1095 bitmap
.SetHBITMAP( (WXHBITMAP
) hbitmap
);
1099 unsigned char r
= GetMaskRed();
1100 unsigned char g
= GetMaskGreen();
1101 unsigned char b
= GetMaskBlue();
1102 unsigned char zero
= 0, one
= 255;
1105 for( int i
=0; i
<width
*height
; i
++ )
1107 if( (*(ptdata
++)!=r
) | (*(ptdata
++)!=g
) | (*(ptdata
++)!=b
) )
1120 hbitmap
= ::CreateBitmap( (WORD
)width
, (WORD
)height
, 1, 1, NULL
);
1121 ::SetDIBits( hdc
, hbitmap
, 0, (WORD
)height
, lpBits
, lpDIBh
, DIB_RGB_COLORS
);
1122 wxMask
*mask
= new wxMask();
1123 mask
->SetMaskBitmap( (WXHBITMAP
) hbitmap
);
1124 bitmap
.SetMask( mask
);
1126 /* The following can also be used but is slow to run
1127 wxColour colour( GetMaskRed(), GetMaskGreen(), GetMaskBlue());
1128 wxMask *mask = new wxMask( bitmap, colour );
1129 bitmap.SetMask( mask );
1133 ::ReleaseDC(NULL
, hdc
);
1137 if( bitmap
.GetHBITMAP() )
1138 bitmap
.SetOk( TRUE
);
1140 bitmap
.SetOk( FALSE
);
1146 wxImage::wxImage( const wxBitmap
&bitmap
)
1150 wxFAIL_MSG( "invalid bitmap" );
1154 int width
= bitmap
.GetWidth();
1155 int height
= bitmap
.GetHeight();
1156 Create( width
, height
);
1157 unsigned char *data
= GetData();
1160 wxFAIL_MSG( "could not allocate data for image" );
1164 int headersize
= sizeof(BITMAPINFOHEADER
);
1165 LPBITMAPINFO lpDIBh
= (BITMAPINFO
*) malloc( headersize
);
1168 wxFAIL_MSG( "could not allocate data for DIB header" );
1173 // Fill in the DIB header
1174 lpDIBh
->bmiHeader
.biSize
= headersize
;
1175 lpDIBh
->bmiHeader
.biWidth
= width
;
1176 lpDIBh
->bmiHeader
.biHeight
= -height
;
1177 lpDIBh
->bmiHeader
.biSizeImage
= width
* height
* 3;
1179 lpDIBh
->bmiHeader
.biPlanes
= 1;
1180 lpDIBh
->bmiHeader
.biBitCount
= 24;
1181 lpDIBh
->bmiHeader
.biCompression
= BI_RGB
;
1182 lpDIBh
->bmiHeader
.biClrUsed
= 0;
1184 // These seem not needed for our purpose here.
1185 // lpDIBh->bmiHeader.biClrImportant = 0;
1186 // lpDIBh->bmiHeader.biXPelsPerMeter = 0;
1187 // lpDIBh->bmiHeader.biYPelsPerMeter = 0;
1189 unsigned char *lpBits
= (unsigned char *) malloc( width
*height
*3 );
1192 wxFAIL_MSG( "could not allocate data for DIB" );
1199 hbitmap
= (HBITMAP
) bitmap
.GetHBITMAP();
1200 HDC hdc
= ::GetDC(NULL
);
1201 ::GetDIBits( hdc
, hbitmap
, 0, height
, lpBits
, lpDIBh
, DIB_RGB_COLORS
);
1203 unsigned char *ptdata
= data
, *ptbits
= lpBits
;
1204 for( int i
=0; i
<width
*height
; i
++ )
1206 *(ptdata
++) = *(ptbits
+2);
1207 *(ptdata
++) = *(ptbits
+1);
1208 *(ptdata
++) = *(ptbits
);
1212 if( bitmap
.GetMask() && bitmap
.GetMask()->GetMaskBitmap() )
1214 hbitmap
= (HBITMAP
) bitmap
.GetMask()->GetMaskBitmap();
1215 HDC memdc
= ::CreateCompatibleDC( hdc
);
1216 ::SetTextColor( memdc
, RGB( 0, 0, 0 ) );
1217 ::SetBkColor( memdc
, RGB( 255, 255, 255 ) );
1218 ::GetDIBits( memdc
, hbitmap
, 0, height
, lpBits
, lpDIBh
, DIB_RGB_COLORS
);
1219 ::DeleteDC( memdc
);
1220 unsigned char r
=16, g
=16, b
=16; // background set to RGB(16,16,16)
1223 for( int i
=0; i
<width
*height
; i
++ )
1233 SetMaskColour( r
, g
, b
);
1236 ::ReleaseDC(NULL
, hdc
);
1245 #include "gtk/gtk.h"
1246 #include "gdk/gdk.h"
1247 #include "gdk/gdkx.h"
1249 wxBitmap
wxImage::ConvertToBitmap() const
1253 wxCHECK_MSG( Ok(), bitmap
, "invalid image" );
1255 int width
= GetWidth();
1256 int height
= GetHeight();
1258 bitmap
.SetHeight( height
);
1259 bitmap
.SetWidth( width
);
1263 GdkImage
*data_image
=
1264 gdk_image_new( GDK_IMAGE_FASTEST
, gdk_visual_get_system(), width
, height
);
1266 bitmap
.SetPixmap( gdk_pixmap_new( (GdkWindow
*)&gdk_root_parent
, width
, height
, -1 ) );
1270 GdkImage
*mask_image
= (GdkImage
*) NULL
;
1274 unsigned char *mask_data
= (unsigned char*)malloc( ((width
>> 3)+8) * height
);
1276 mask_image
= gdk_image_new_bitmap( gdk_visual_get_system(), mask_data
, width
, height
);
1278 wxMask
*mask
= new wxMask();
1279 mask
->m_bitmap
= gdk_pixmap_new( (GdkWindow
*)&gdk_root_parent
, width
, height
, 1 );
1281 bitmap
.SetMask( mask
);
1286 GdkVisual
*visual
= gdk_window_get_visual( bitmap
.GetPixmap() );
1287 if (visual
== NULL
) visual
= gdk_window_get_visual( (GdkWindow
*) &gdk_root_parent
);
1288 int bpp
= visual
->depth
;
1289 if ((bpp
== 16) && (visual
->red_mask
!= 0xf800)) bpp
= 15;
1290 if (bpp
< 8) bpp
= 8;
1294 enum byte_order
{ RGB
, RBG
, BRG
, BGR
, GRB
, GBR
};
1295 byte_order b_o
= RGB
;
1299 GdkVisual
*visual
= gdk_visual_get_system();
1300 if ((visual
->red_mask
> visual
->green_mask
) && (visual
->green_mask
> visual
->blue_mask
)) b_o
= RGB
;
1301 else if ((visual
->red_mask
> visual
->blue_mask
) && (visual
->blue_mask
> visual
->green_mask
)) b_o
= RGB
;
1302 else if ((visual
->blue_mask
> visual
->red_mask
) && (visual
->red_mask
> visual
->green_mask
)) b_o
= BRG
;
1303 else if ((visual
->blue_mask
> visual
->green_mask
) && (visual
->green_mask
> visual
->red_mask
)) b_o
= BGR
;
1304 else if ((visual
->green_mask
> visual
->red_mask
) && (visual
->red_mask
> visual
->blue_mask
)) b_o
= GRB
;
1305 else if ((visual
->green_mask
> visual
->blue_mask
) && (visual
->blue_mask
> visual
->red_mask
)) b_o
= GBR
;
1308 int r_mask
= GetMaskRed();
1309 int g_mask
= GetMaskGreen();
1310 int b_mask
= GetMaskBlue();
1312 unsigned char* data
= GetData();
1315 for (int y
= 0; y
< height
; y
++)
1317 for (int x
= 0; x
< width
; x
++)
1319 int r
= data
[index
];
1321 int g
= data
[index
];
1323 int b
= data
[index
];
1328 if ((r
== r_mask
) && (b
== b_mask
) && (g
== g_mask
))
1329 gdk_image_put_pixel( mask_image
, x
, y
, 1 );
1331 gdk_image_put_pixel( mask_image
, x
, y
, 0 );
1338 GdkColormap
*cmap
= gtk_widget_get_default_colormap();
1339 GdkColor
*colors
= cmap
->colors
;
1340 int max
= 3 * (65536);
1343 for (int i
= 0; i
< cmap
->size
; i
++)
1345 int rdiff
= (r
<< 8) - colors
[i
].red
;
1346 int gdiff
= (g
<< 8) - colors
[i
].green
;
1347 int bdiff
= (b
<< 8) - colors
[i
].blue
;
1348 int sum
= ABS (rdiff
) + ABS (gdiff
) + ABS (bdiff
);
1349 if (sum
< max
) { index
= i
; max
= sum
; }
1352 gdk_image_put_pixel( data_image
, x
, y
, index
);
1358 guint32 pixel
= ((r
& 0xf8) << 7) | ((g
& 0xf8) << 2) | ((b
& 0xf8) >> 3);
1359 gdk_image_put_pixel( data_image
, x
, y
, pixel
);
1364 guint32 pixel
= ((r
& 0xf8) << 8) | ((g
& 0xfc) << 3) | ((b
& 0xf8) >> 3);
1365 gdk_image_put_pixel( data_image
, x
, y
, pixel
);
1374 case RGB
: pixel
= (r
<< 16) | (g
<< 8) | b
; break;
1375 case RBG
: pixel
= (r
<< 16) | (b
<< 8) | g
; break;
1376 case BRG
: pixel
= (b
<< 16) | (r
<< 8) | g
; break;
1377 case BGR
: pixel
= (b
<< 16) | (g
<< 8) | r
; break;
1378 case GRB
: pixel
= (g
<< 16) | (r
<< 8) | b
; break;
1379 case GBR
: pixel
= (g
<< 16) | (b
<< 8) | r
; break;
1381 gdk_image_put_pixel( data_image
, x
, y
, pixel
);
1390 GdkGC
*data_gc
= gdk_gc_new( bitmap
.GetPixmap() );
1392 gdk_draw_image( bitmap
.GetPixmap(), data_gc
, data_image
, 0, 0, 0, 0, width
, height
);
1394 gdk_image_destroy( data_image
);
1395 gdk_gc_unref( data_gc
);
1401 GdkGC
*mask_gc
= gdk_gc_new( bitmap
.GetMask()->GetBitmap() );
1403 gdk_draw_image( bitmap
.GetMask()->GetBitmap(), mask_gc
, mask_image
, 0, 0, 0, 0, width
, height
);
1405 gdk_image_destroy( mask_image
);
1406 gdk_gc_unref( mask_gc
);
1412 wxImage::wxImage( const wxBitmap
&bitmap
)
1414 wxCHECK_RET( bitmap
.Ok(), "invalid bitmap" );
1416 GdkImage
*gdk_image
= gdk_image_get( bitmap
.GetPixmap(),
1418 bitmap
.GetWidth(), bitmap
.GetHeight() );
1420 wxCHECK_RET( gdk_image
, "couldn't create image" );
1422 Create( bitmap
.GetWidth(), bitmap
.GetHeight() );
1423 char unsigned *data
= GetData();
1427 gdk_image_destroy( gdk_image
);
1428 wxFAIL_MSG( "couldn't create image" );
1432 GdkImage
*gdk_image_mask
= (GdkImage
*) NULL
;
1433 if (bitmap
.GetMask())
1435 gdk_image_mask
= gdk_image_get( bitmap
.GetMask()->GetBitmap(),
1437 bitmap
.GetWidth(), bitmap
.GetHeight() );
1439 SetMaskColour( 16, 16, 16 ); // anything unlikely and dividable
1442 GdkVisual
*visual
= gdk_window_get_visual( bitmap
.GetPixmap() );
1443 if (visual
== NULL
) visual
= gdk_window_get_visual( (GdkWindow
*) &gdk_root_parent
);
1444 int bpp
= visual
->depth
;
1445 if ((bpp
== 16) && (visual
->red_mask
!= 0xf800)) bpp
= 15;
1447 GdkColormap
*cmap
= gtk_widget_get_default_colormap();
1450 for (int j
= 0; j
< bitmap
.GetHeight(); j
++)
1452 for (int i
= 0; i
< bitmap
.GetWidth(); i
++)
1454 int pixel
= gdk_image_get_pixel( gdk_image
, i
, j
);
1457 data
[pos
] = cmap
->colors
[pixel
].red
>> 8;
1458 data
[pos
+1] = cmap
->colors
[pixel
].green
>> 8;
1459 data
[pos
+2] = cmap
->colors
[pixel
].blue
>> 8;
1460 } else if (bpp
== 15)
1462 data
[pos
] = (pixel
>> 7) & 0xf8;
1463 data
[pos
+1] = (pixel
>> 2) & 0xf8;
1464 data
[pos
+2] = (pixel
<< 3) & 0xf8;
1465 } else if (bpp
== 16)
1467 data
[pos
] = (pixel
>> 8) & 0xf8;
1468 data
[pos
+1] = (pixel
>> 3) & 0xfc;
1469 data
[pos
+2] = (pixel
<< 3) & 0xf8;
1472 data
[pos
] = (pixel
>> 16) & 0xff;
1473 data
[pos
+1] = (pixel
>> 8) & 0xff;
1474 data
[pos
+2] = pixel
& 0xff;
1479 int mask_pixel
= gdk_image_get_pixel( gdk_image_mask
, i
, j
);
1480 if (mask_pixel
== 0)
1492 gdk_image_destroy( gdk_image
);
1493 if (gdk_image_mask
) gdk_image_destroy( gdk_image_mask
);