1 ///////////////////////////////////////////////////////////////////////////// 
   2 // Name:        src/common/imagbmp.cpp 
   3 // Purpose:     wxImage BMP,ICO and CUR handlers 
   4 // Author:      Robert Roebling, Chris Elliott 
   6 // Copyright:   (c) Robert Roebling, Chris Elliott 
   7 // Licence:     wxWindows licence 
   8 ///////////////////////////////////////////////////////////////////////////// 
  10 // For compilers that support precompilation, includes "wx.h". 
  11 #include "wx/wxprec.h" 
  19 #include "wx/imagbmp.h" 
  23         #include "wx/msw/wrapwin.h" 
  27     #include "wx/bitmap.h" 
  28     #include "wx/palette.h" 
  33 #include "wx/filefn.h" 
  34 #include "wx/wfstream.h" 
  35 #include "wx/quantize.h" 
  36 #include "wx/scopeguard.h" 
  37 #include "wx/anidecod.h" 
  42 // ---------------------------------------------------------------------------- 
  44 // ---------------------------------------------------------------------------- 
  48 static bool CanReadICOOrCUR(wxInputStream 
*stream
, wxUint16 resourceType
); 
  50 #endif // wxUSE_ICO_CUR 
  52 //----------------------------------------------------------------------------- 
  54 //----------------------------------------------------------------------------- 
  56 IMPLEMENT_DYNAMIC_CLASS(wxBMPHandler
,wxImageHandler
) 
  73 #define BI_BITFIELDS 3 
  76 #define poffset (line * width * 3 + column * 3) 
  78 bool wxBMPHandler::SaveFile(wxImage 
*image
, 
  79                             wxOutputStream
& stream
, 
  82     return SaveDib(image
, stream
, verbose
, true/*IsBmp*/, false/*IsMask*/); 
  85 bool wxBMPHandler::SaveDib(wxImage 
*image
, 
  86                            wxOutputStream
& stream
, 
  92     wxCHECK_MSG( image
, false, wxT("invalid pointer in wxBMPHandler::SaveFile") ); 
  98             wxLogError(_("BMP: Couldn't save invalid image.")); 
 103     // get the format of the BMP file to save, else use 24bpp 
 104     unsigned format 
= wxBMP_24BPP
; 
 105     if ( image
->HasOption(wxIMAGE_OPTION_BMP_FORMAT
) ) 
 106         format 
= image
->GetOptionInt(wxIMAGE_OPTION_BMP_FORMAT
); 
 108     wxUint16 bpp
;     // # of bits per pixel 
 109     int palette_size
; // # of color map entries, ie. 2^bpp colors 
 111     // set the bpp and appropriate palette_size, and do additional checks 
 112     if ( (format 
== wxBMP_1BPP
) || (format 
== wxBMP_1BPP_BW
) ) 
 117     else if ( format 
== wxBMP_4BPP 
) 
 122     else if ( (format 
== wxBMP_8BPP
) || (format 
== wxBMP_8BPP_GREY
) || 
 123               (format 
== wxBMP_8BPP_RED
) || (format 
== wxBMP_8BPP_PALETTE
) ) 
 125         // need to set a wxPalette to use this, HOW TO CHECK IF VALID, SIZE? 
 126         if ((format 
== wxBMP_8BPP_PALETTE
) 
 128                 && !image
->HasPalette() 
 129 #endif // wxUSE_PALETTE 
 134                 wxLogError(_("BMP: wxImage doesn't have own wxPalette.")); 
 141     else  // you get 24bpp 
 143         format 
= wxBMP_24BPP
; 
 148     unsigned width 
= image
->GetWidth(); 
 149     unsigned row_padding 
= (4 - ((width 
* bpp 
+ 7) / 8) % 4) % 4; // # bytes to pad to dword 
 150     unsigned row_width 
= (width 
* bpp 
+ 7) / 8 + row_padding
; // # of bytes per row 
 155         wxUint16  magic
;          // format magic, always 'BM' 
 156         wxUint32  filesize
;       // total file size, inc. headers 
 157         wxUint32  reserved
;       // for future use 
 158         wxUint32  data_offset
;    // image data offset in the file 
 161         wxUint32  bih_size
;       // 2nd part's size 
 162         wxUint32  width
, height
;  // bitmap's dimensions 
 163         wxUint16  planes
;         // num of planes 
 164         wxUint16  bpp
;            // bits per pixel 
 165         wxUint32  compression
;    // compression method 
 166         wxUint32  size_of_bmp
;    // size of the bitmap 
 167         wxUint32  h_res
, v_res
;   // image resolution in pixels-per-meter 
 168         wxUint32  num_clrs
;       // number of colors used 
 169         wxUint32  num_signif_clrs
;// number of significant colors 
 172     wxUint32 hdr_size 
= 14/*BitmapHeader*/ + 40/*BitmapInfoHeader*/; 
 174     hdr
.magic 
= wxUINT16_SWAP_ON_BE(0x4D42/*'BM'*/); 
 175     hdr
.filesize 
= wxUINT32_SWAP_ON_BE( hdr_size 
+ palette_size
*4 + 
 176                                         row_width 
* image
->GetHeight() ); 
 178     hdr
.data_offset 
= wxUINT32_SWAP_ON_BE(hdr_size 
+ palette_size
*4); 
 180     hdr
.bih_size 
= wxUINT32_SWAP_ON_BE(hdr_size 
- 14); 
 181     hdr
.width 
= wxUINT32_SWAP_ON_BE(image
->GetWidth()); 
 184         hdr
.height 
= wxUINT32_SWAP_ON_BE(image
->GetHeight()); 
 188         hdr
.height 
= wxUINT32_SWAP_ON_BE(2 * image
->GetHeight()); 
 190     hdr
.planes 
= wxUINT16_SWAP_ON_BE(1); // always 1 plane 
 191     hdr
.bpp 
= wxUINT16_SWAP_ON_BE(bpp
); 
 192     hdr
.compression 
= 0; // RGB uncompressed 
 193     hdr
.size_of_bmp 
= wxUINT32_SWAP_ON_BE(row_width 
* image
->GetHeight()); 
 195     // get the resolution from the image options  or fall back to 72dpi standard 
 196     // for the BMP format if not specified 
 198     switch ( GetResolutionFromOptions(*image
, &hres
, &vres
) ) 
 201             wxFAIL_MSG( wxT("unexpected image resolution units") ); 
 204         case wxIMAGE_RESOLUTION_NONE
: 
 207             // fall through to convert it to correct units 
 209         case wxIMAGE_RESOLUTION_INCHES
: 
 210             // convert resolution in inches to resolution in centimeters 
 211             hres 
= (int)(10*mm2inches
*hres
); 
 212             vres 
= (int)(10*mm2inches
*vres
); 
 213             // fall through to convert it to resolution in meters 
 215         case wxIMAGE_RESOLUTION_CM
: 
 216             // convert resolution in centimeters to resolution in meters 
 222     hdr
.h_res 
= wxUINT32_SWAP_ON_BE(hres
); 
 223     hdr
.v_res 
= wxUINT32_SWAP_ON_BE(vres
); 
 224     hdr
.num_clrs 
= wxUINT32_SWAP_ON_BE(palette_size
); // # colors in colormap 
 225     hdr
.num_signif_clrs 
= 0;     // all colors are significant 
 229         if (// VS: looks ugly but compilers tend to do ugly things with structs, 
 230             //     like aligning hdr.filesize's ofset to dword :( 
 231             // VZ: we should add padding then... 
 232             !stream
.Write(&hdr
.magic
, 2) || 
 233             !stream
.Write(&hdr
.filesize
, 4) || 
 234             !stream
.Write(&hdr
.reserved
, 4) || 
 235             !stream
.Write(&hdr
.data_offset
, 4) 
 240                 wxLogError(_("BMP: Couldn't write the file (Bitmap) header.")); 
 248             !stream
.Write(&hdr
.bih_size
, 4) || 
 249             !stream
.Write(&hdr
.width
, 4) || 
 250             !stream
.Write(&hdr
.height
, 4) || 
 251             !stream
.Write(&hdr
.planes
, 2) || 
 252             !stream
.Write(&hdr
.bpp
, 2) || 
 253             !stream
.Write(&hdr
.compression
, 4) || 
 254             !stream
.Write(&hdr
.size_of_bmp
, 4) || 
 255             !stream
.Write(&hdr
.h_res
, 4) || 
 256             !stream
.Write(&hdr
.v_res
, 4) || 
 257             !stream
.Write(&hdr
.num_clrs
, 4) || 
 258             !stream
.Write(&hdr
.num_signif_clrs
, 4) 
 263                 wxLogError(_("BMP: Couldn't write the file (BitmapInfo) header.")); 
 269     wxPalette 
*palette 
= NULL
; // entries for quantized images 
 270     wxUint8 
*rgbquad 
= NULL
;   // for the RGBQUAD bytes for the colormap 
 271     wxImage 
*q_image 
= NULL
;   // destination for quantized image 
 273     // if <24bpp use quantization to reduce colors for *some* of the formats 
 274     if ( (format 
== wxBMP_1BPP
) || (format 
== wxBMP_4BPP
) || 
 275          (format 
== wxBMP_8BPP
) || (format 
== wxBMP_8BPP_PALETTE
) ) 
 277         // make a new palette and quantize the image 
 278         if (format 
!= wxBMP_8BPP_PALETTE
) 
 280             q_image 
= new wxImage(); 
 282             // I get a delete error using Quantize when desired colors > 236 
 283             int quantize 
= ((palette_size 
> 236) ? 236 : palette_size
); 
 284             // fill the destination too, it gives much nicer 4bpp images 
 285             wxQuantize::Quantize( *image
, *q_image
, &palette
, quantize
, 0, 
 286                                   wxQUANTIZE_FILL_DESTINATION_IMAGE 
); 
 291             palette 
= new wxPalette(image
->GetPalette()); 
 292 #endif // wxUSE_PALETTE 
 296         unsigned char r
, g
, b
; 
 297         rgbquad 
= new wxUint8 
[palette_size
*4]; 
 299         for (i 
= 0; i 
< palette_size
; i
++) 
 302             if ( !palette
->GetRGB(i
, &r
, &g
, &b
) ) 
 303 #endif // wxUSE_PALETTE 
 312     // make a 256 entry greyscale colormap or 2 entry black & white 
 313     else if ( (format 
== wxBMP_8BPP_GREY
) || (format 
== wxBMP_8BPP_RED
) || 
 314               (format 
== wxBMP_1BPP_BW
) ) 
 316         rgbquad 
= new wxUint8 
[palette_size
*4]; 
 318         for ( int i 
= 0; i 
< palette_size
; i
++ ) 
 320             // if 1BPP_BW then the value should be either 0 or 255 
 321             wxUint8 c 
= (wxUint8
)((i 
> 0) && (format 
== wxBMP_1BPP_BW
) ? 255 : i
); 
 330     // if the colormap was made, then it needs to be written 
 335             if ( !stream
.Write(rgbquad
, palette_size
*4) ) 
 339                     wxLogError(_("BMP: Couldn't write RGB color map.")); 
 344 #endif // wxUSE_PALETTE 
 352     // pointer to the image data, use quantized if available 
 353     wxUint8 
*data 
= (wxUint8
*) image
->GetData(); 
 354     if (q_image
) if (q_image
->Ok()) data 
= (wxUint8
*) q_image
->GetData(); 
 356     wxUint8 
*buffer 
= new wxUint8
[row_width
]; 
 357     memset(buffer
, 0, row_width
); 
 361     for (y 
= image
->GetHeight() -1; y 
>= 0; y
--) 
 363         if ( format 
== wxBMP_24BPP 
) // 3 bytes per pixel red,green,blue 
 365             for ( x 
= 0; x 
< width
; x
++ ) 
 367                 pixel 
= 3*(y
*width 
+ x
); 
 369                 buffer
[3*x    
] = data
[pixel
+2]; 
 370                 buffer
[3*x 
+ 1] = data
[pixel
+1]; 
 371                 buffer
[3*x 
+ 2] = data
[pixel
]; 
 374         else if ((format 
== wxBMP_8BPP
) ||       // 1 byte per pixel in color 
 375                  (format 
== wxBMP_8BPP_PALETTE
)) 
 377             for (x 
= 0; x 
< width
; x
++) 
 379                 pixel 
= 3*(y
*width 
+ x
); 
 381                 buffer
[x
] = (wxUint8
)palette
->GetPixel( data
[pixel
], 
 385                 // FIXME: what should this be? use some std palette maybe? 
 387 #endif // wxUSE_PALETTE 
 390         else if ( format 
== wxBMP_8BPP_GREY 
) // 1 byte per pix, rgb ave to grey 
 392             for (x 
= 0; x 
< width
; x
++) 
 394                 pixel 
= 3*(y
*width 
+ x
); 
 395                 buffer
[x
] = (wxUint8
)(.299*data
[pixel
] + 
 400         else if ( format 
== wxBMP_8BPP_RED 
) // 1 byte per pixel, red as greys 
 402             for (x 
= 0; x 
< width
; x
++) 
 404                 buffer
[x
] = (wxUint8
)data
[3*(y
*width 
+ x
)]; 
 407         else if ( format 
== wxBMP_4BPP 
) // 4 bpp in color 
 409             for (x 
= 0; x 
< width
; x
+=2) 
 411                 pixel 
= 3*(y
*width 
+ x
); 
 413                 // fill buffer, ignore if > width 
 415                 buffer
[x
/2] = (wxUint8
)( 
 416                     ((wxUint8
)palette
->GetPixel(data
[pixel
], 
 418                                                 data
[pixel
+2]) << 4) | 
 421                      : ((wxUint8
)palette
->GetPixel(data
[pixel
+3], 
 425                 // FIXME: what should this be? use some std palette maybe? 
 427 #endif // wxUSE_PALETTE 
 430         else if ( format 
== wxBMP_1BPP 
) // 1 bpp in "color" 
 432             for (x 
= 0; x 
< width
; x
+=8) 
 434                 pixel 
= 3*(y
*width 
+ x
); 
 437                 buffer
[x
/8] = (wxUint8
)( 
 438                                            ((wxUint8
)palette
->GetPixel(data
[pixel
], data
[pixel
+1], data
[pixel
+2]) << 7) | 
 439                     (((x
+1) > width
) ? 0 : ((wxUint8
)palette
->GetPixel(data
[pixel
+3], data
[pixel
+4], data
[pixel
+5]) << 6)) | 
 440                     (((x
+2) > width
) ? 0 : ((wxUint8
)palette
->GetPixel(data
[pixel
+6], data
[pixel
+7], data
[pixel
+8]) << 5)) | 
 441                     (((x
+3) > width
) ? 0 : ((wxUint8
)palette
->GetPixel(data
[pixel
+9], data
[pixel
+10], data
[pixel
+11]) << 4)) | 
 442                     (((x
+4) > width
) ? 0 : ((wxUint8
)palette
->GetPixel(data
[pixel
+12], data
[pixel
+13], data
[pixel
+14]) << 3)) | 
 443                     (((x
+5) > width
) ? 0 : ((wxUint8
)palette
->GetPixel(data
[pixel
+15], data
[pixel
+16], data
[pixel
+17]) << 2)) | 
 444                     (((x
+6) > width
) ? 0 : ((wxUint8
)palette
->GetPixel(data
[pixel
+18], data
[pixel
+19], data
[pixel
+20]) << 1)) | 
 445                     (((x
+7) > width
) ? 0 : ((wxUint8
)palette
->GetPixel(data
[pixel
+21], data
[pixel
+22], data
[pixel
+23])     ))    ); 
 447                 // FIXME: what should this be? use some std palette maybe? 
 449 #endif // wxUSE_PALETTE 
 452         else if ( format 
== wxBMP_1BPP_BW 
) // 1 bpp B&W colormap from red color ONLY 
 454             for (x 
= 0; x 
< width
; x
+=8) 
 456                 pixel 
= 3*(y
*width 
+ x
); 
 458                 buffer
[x
/8] = (wxUint8
)( 
 459                                           (((wxUint8
)(data
[pixel
]   /128.)) << 7) | 
 460                    (((x
+1) > width
) ? 0 : (((wxUint8
)(data
[pixel
+3] /128.)) << 6)) | 
 461                    (((x
+2) > width
) ? 0 : (((wxUint8
)(data
[pixel
+6] /128.)) << 5)) | 
 462                    (((x
+3) > width
) ? 0 : (((wxUint8
)(data
[pixel
+9] /128.)) << 4)) | 
 463                    (((x
+4) > width
) ? 0 : (((wxUint8
)(data
[pixel
+12]/128.)) << 3)) | 
 464                    (((x
+5) > width
) ? 0 : (((wxUint8
)(data
[pixel
+15]/128.)) << 2)) | 
 465                    (((x
+6) > width
) ? 0 : (((wxUint8
)(data
[pixel
+18]/128.)) << 1)) | 
 466                    (((x
+7) > width
) ? 0 : (((wxUint8
)(data
[pixel
+21]/128.))     ))    ); 
 470         if ( !stream
.Write(buffer
, row_width
) ) 
 474                 wxLogError(_("BMP: Couldn't write data.")); 
 479 #endif // wxUSE_PALETTE 
 487 #endif // wxUSE_PALETTE 
 496     static void Free(BMPPalette
* pal
) { delete [] pal
; } 
 498     unsigned char r
, g
, b
; 
 501 bool wxBMPHandler::DoLoadDib(wxImage 
* image
, int width
, int height
, 
 502                              int bpp
, int ncolors
, int comp
, 
 503                              wxFileOffset bmpOffset
, wxInputStream
& stream
, 
 504                              bool verbose
, bool IsBmp
, bool hasPalette
) 
 506     wxInt32         aDword
, rmask 
= 0, gmask 
= 0, bmask 
= 0, amask 
= 0; 
 507     int             rshift 
= 0, gshift 
= 0, bshift 
= 0, ashift 
= 0; 
 508     int             rbits 
= 0, gbits 
= 0, bbits 
= 0; 
 514     // allocate space for palette if needed: 
 519         cmap 
= new BMPPalette
[ncolors
]; 
 524                 wxLogError(_("BMP: Couldn't allocate memory.")); 
 534     wxON_BLOCK_EXIT1(&BMPPalette::Free
, cmap
); 
 536     // destroy existing here instead of: 
 538     image
->Create(width
, height
); 
 540     unsigned char *ptr 
= image
->GetData(); 
 546             wxLogError( _("BMP: Couldn't allocate memory.") ); 
 551     unsigned char *alpha
; 
 554         // tell the image to allocate an alpha buffer 
 556         alpha 
= image
->GetAlpha(); 
 561                 wxLogError(_("BMP: Couldn't allocate memory.")); 
 571     // Reading the palette, if it exists: 
 572     if ( bpp 
< 16 && ncolors 
!= 0 ) 
 574         unsigned char* r 
= new unsigned char[ncolors
]; 
 575         unsigned char* g 
= new unsigned char[ncolors
]; 
 576         unsigned char* b 
= new unsigned char[ncolors
]; 
 577         for (int j 
= 0; j 
< ncolors
; j
++) 
 581                 stream
.Read(bbuf
, 4); 
 592                 //used in reading .ico file mask 
 595                 b
[j
] = cmap
[j
].b 
= ( j 
? 255 : 0 ); 
 600         // Set the palette for the wxImage 
 601         image
->SetPalette(wxPalette(ncolors
, r
, g
, b
)); 
 602 #endif // wxUSE_PALETTE 
 608     else if ( bpp 
== 16 || bpp 
== 32 ) 
 610         if ( comp 
== BI_BITFIELDS 
) 
 613             stream
.Read(dbuf
, 4 * 3); 
 614             rmask 
= wxINT32_SWAP_ON_BE(dbuf
[0]); 
 615             gmask 
= wxINT32_SWAP_ON_BE(dbuf
[1]); 
 616             bmask 
= wxINT32_SWAP_ON_BE(dbuf
[2]); 
 617             // find shift amount (Least significant bit of mask) 
 618             for (bit 
= bpp
-1; bit
>=0; bit
--) 
 620                 if (bmask 
& (1 << bit
)) 
 622                 if (gmask 
& (1 << bit
)) 
 624                 if (rmask 
& (1 << bit
)) 
 627             // Find number of bits in mask (MSB-LSB+1) 
 628             for (bit 
= 0; bit 
< bpp
; bit
++) 
 630                 if (bmask 
& (1 << bit
)) 
 631                     bbits 
= bit
-bshift
+1; 
 632                 if (gmask 
& (1 << bit
)) 
 633                     gbits 
= bit
-gshift
+1; 
 634                 if (rmask 
& (1 << bit
)) 
 635                     rbits 
= bit
-rshift
+1; 
 638         else if ( bpp 
== 16 ) 
 650         else if ( bpp 
== 32 ) 
 668      * Reading the image data 
 672         // NOTE: seeking a positive amount in wxFromCurrent mode allows us to 
 673         //       load even non-seekable streams (see wxInputStream::SeekI docs)! 
 674         const wxFileOffset pos 
= stream
.TellI(); 
 675         if (pos 
!= wxInvalidOffset 
&& bmpOffset 
> pos
) 
 676             if (stream
.SeekI(bmpOffset 
- pos
, wxFromCurrent
) == wxInvalidOffset
) 
 678         //else: icon, just carry on 
 681     unsigned char *data 
= ptr
; 
 683     /* set the whole image to the background color */ 
 684     if ( bpp 
< 16 && (comp 
== BI_RLE4 
|| comp 
== BI_RLE8
) ) 
 686         for (int i 
= 0; i 
< width 
* height
; i
++) 
 695     int linesize 
= ((width 
* bpp 
+ 31) / 32) * 4; 
 697     // flag indicating if we have any not fully transparent alpha values: this 
 698     // is used to account for the bitmaps which use 32bpp format (normally 
 699     // meaning that they have alpha channel) but have only zeroes in it so that 
 700     // without this hack they appear fully transparent -- and as this is 
 701     // unlikely intentional, we consider that they don't have alpha at all in 
 702     // this case (see #10915) 
 703     bool hasValidAlpha 
= false; 
 705     /* BMPs are stored upside down */ 
 706     for ( int line 
= (height 
- 1); line 
>= 0; line
-- ) 
 709         for ( int column 
= 0; column 
< width 
; ) 
 714                 aByte 
= stream
.GetC(); 
 717                     for (int bit 
= 0; bit 
< 8 && column 
< width
; bit
++) 
 719                         int index 
= ((aByte 
& (0x80 >> bit
)) ? 1 : 0); 
 720                         ptr
[poffset
] = cmap
[index
].r
; 
 721                         ptr
[poffset 
+ 1] = cmap
[index
].g
; 
 722                         ptr
[poffset 
+ 2] = cmap
[index
].b
; 
 728                     if ( comp 
== BI_RLE4 
) 
 732                         aByte 
= stream
.GetC(); 
 740                             else if ( aByte 
== 1 ) 
 745                             else if ( aByte 
== 2 ) 
 747                                 aByte 
= stream
.GetC(); 
 749                                 linepos 
= column 
* bpp 
/ 4; 
 750                                 aByte 
= stream
.GetC(); 
 751                                 line 
-= aByte
; // upside down 
 755                                 int absolute 
= aByte
; 
 758                                 for (int k 
= 0; k 
< absolute
; k
++) 
 763                                         aByte 
= stream
.GetC(); 
 764                                         nibble
[0] = (wxUint8
)( (aByte 
& 0xF0) >> 4 ) ; 
 765                                         nibble
[1] = (wxUint8
)( aByte 
& 0x0F ) ; 
 767                                     ptr
[poffset    
] = cmap
[nibble
[k%2
]].r
; 
 768                                     ptr
[poffset 
+ 1] = cmap
[nibble
[k%2
]].g
; 
 769                                     ptr
[poffset 
+ 2] = cmap
[nibble
[k%2
]].b
; 
 774                                 if ( readBytes 
& 0x01 ) 
 775                                     aByte 
= stream
.GetC(); 
 781                             nibble
[0] = (wxUint8
)( (aByte 
& 0xF0) >> 4 ) ; 
 782                             nibble
[1] = (wxUint8
)( aByte 
& 0x0F ) ; 
 784                             for ( int l 
= 0; l 
< first 
&& column 
< width
; l
++ ) 
 786                                 ptr
[poffset    
] = cmap
[nibble
[l%2
]].r
; 
 787                                 ptr
[poffset 
+ 1] = cmap
[nibble
[l%2
]].g
; 
 788                                 ptr
[poffset 
+ 2] = cmap
[nibble
[l%2
]].b
; 
 797                         for (int nibble 
= 0; nibble 
< 2 && column 
< width
; nibble
++) 
 799                             int index 
= ((aByte 
& (0xF0 >> (nibble 
* 4))) >> (!nibble 
* 4)); 
 802                             ptr
[poffset
] = cmap
[index
].r
; 
 803                             ptr
[poffset 
+ 1] = cmap
[index
].g
; 
 804                             ptr
[poffset 
+ 2] = cmap
[index
].b
; 
 811                     if ( comp 
== BI_RLE8 
) 
 815                         aByte 
= stream
.GetC(); 
 820                                 /* column = width; */ 
 822                             else if ( aByte 
== 1 ) 
 827                             else if ( aByte 
== 2 ) 
 829                                 aByte 
= stream
.GetC(); 
 831                                 linepos 
= column 
* bpp 
/ 8; 
 832                                 aByte 
= stream
.GetC(); 
 837                                 int absolute 
= aByte
; 
 838                                 for (int k 
= 0; k 
< absolute
; k
++) 
 841                                     aByte 
= stream
.GetC(); 
 842                                     ptr
[poffset    
] = cmap
[aByte
].r
; 
 843                                     ptr
[poffset 
+ 1] = cmap
[aByte
].g
; 
 844                                     ptr
[poffset 
+ 2] = cmap
[aByte
].b
; 
 847                                 if ( absolute 
& 0x01 ) 
 848                                     aByte 
= stream
.GetC(); 
 853                             for ( int l 
= 0; l 
< first 
&& column 
< width
; l
++ ) 
 855                                 ptr
[poffset    
] = cmap
[aByte
].r
; 
 856                                 ptr
[poffset 
+ 1] = cmap
[aByte
].g
; 
 857                                 ptr
[poffset 
+ 2] = cmap
[aByte
].b
; 
 865                         ptr
[poffset    
] = cmap
[aByte
].r
; 
 866                         ptr
[poffset 
+ 1] = cmap
[aByte
].g
; 
 867                         ptr
[poffset 
+ 2] = cmap
[aByte
].b
; 
 869                         // linepos += size;    seems to be wrong, RR 
 873             else if ( bpp 
== 24 ) 
 875                 stream
.Read(bbuf
, 3); 
 877                 ptr
[poffset    
] = (unsigned char)bbuf
[2]; 
 878                 ptr
[poffset 
+ 1] = (unsigned char)bbuf
[1]; 
 879                 ptr
[poffset 
+ 2] = (unsigned char)bbuf
[0]; 
 882             else if ( bpp 
== 16 ) 
 885                 stream
.Read(&aWord
, 2); 
 886                 aWord 
= wxUINT16_SWAP_ON_BE(aWord
); 
 888                 /* Use the masks and calculated amount of shift 
 889                    to retrieve the color data out of the word.  Then 
 890                    shift it left by (8 - number of bits) such that 
 891                    the image has the proper dynamic range */ 
 892                 temp 
= (unsigned char)(((aWord 
& rmask
) >> rshift
) << (8-rbits
)); 
 894                 temp 
= (unsigned char)(((aWord 
& gmask
) >> gshift
) << (8-gbits
)); 
 895                 ptr
[poffset 
+ 1] = temp
; 
 896                 temp 
= (unsigned char)(((aWord 
& bmask
) >> bshift
) << (8-bbits
)); 
 897                 ptr
[poffset 
+ 2] = temp
; 
 903                 stream
.Read(&aDword
, 4); 
 904                 aDword 
= wxINT32_SWAP_ON_BE(aDword
); 
 906                 temp 
= (unsigned char)((aDword 
& rmask
) >> rshift
); 
 908                 temp 
= (unsigned char)((aDword 
& gmask
) >> gshift
); 
 909                 ptr
[poffset 
+ 1] = temp
; 
 910                 temp 
= (unsigned char)((aDword 
& bmask
) >> bshift
); 
 911                 ptr
[poffset 
+ 2] = temp
; 
 914                     temp 
= (unsigned char)((aDword 
& amask
) >> ashift
); 
 915                     alpha
[line 
* width 
+ column
] = temp
; 
 917                     if ( temp 
!= wxALPHA_TRANSPARENT 
) 
 918                         hasValidAlpha 
= true; 
 923         while ( (linepos 
< linesize
) && (comp 
!= 1) && (comp 
!= 2) ) 
 925             stream
.Read(&aByte
, 1); 
 932     image
->SetMask(false); 
 934     // check if we had any valid alpha values in this bitmap 
 935     if ( alpha 
&& !hasValidAlpha 
) 
 937         // we didn't, so finally discard the alpha channel completely 
 941     const wxStreamError err 
= stream
.GetLastError(); 
 942     return err 
== wxSTREAM_NO_ERROR 
|| err 
== wxSTREAM_EOF
; 
 945 bool wxBMPHandler::LoadDib(wxImage 
*image
, wxInputStream
& stream
, 
 946                            bool verbose
, bool IsBmp
) 
 954         // read the header off the .BMP format file 
 955         stream
.Read(bbuf
, 2); 
 956         stream
.Read(dbuf
, 16); 
 960         stream
.Read(dbuf
, 4); 
 963         wxInt32 size 
= wxINT32_SWAP_ON_BE(dbuf
[0]); 
 965     wxFileOffset offset 
= wxINT32_SWAP_ON_BE(dbuf
[2]); 
 967     stream
.Read(dbuf
, 4 * 2); 
 968     int width 
= wxINT32_SWAP_ON_BE((int)dbuf
[0]); 
 969     int height 
= wxINT32_SWAP_ON_BE((int)dbuf
[1]); 
 970     if ( !IsBmp
)height 
= height  
/ 2; // for icons divide by 2 
 976             wxLogError( _("DIB Header: Image width > 32767 pixels for file.") ); 
 980     if ( height 
> 32767 ) 
 984             wxLogError( _("DIB Header: Image height > 32767 pixels for file.") ); 
 989     stream
.Read(&aWord
, 2); 
 992             int planes = (int)wxUINT16_SWAP_ON_BE( aWord ); 
 994     stream
.Read(&aWord
, 2); 
 995     int bpp 
= wxUINT16_SWAP_ON_BE((int)aWord
); 
 996     if ( bpp 
!= 1 && bpp 
!= 4 && bpp 
!= 8 && bpp 
!= 16 && bpp 
!= 24 && bpp 
!= 32 ) 
1000             wxLogError( _("DIB Header: Unknown bitdepth in file.") ); 
1005     stream
.Read(dbuf
, 4 * 4); 
1006     int comp 
= wxINT32_SWAP_ON_BE((int)dbuf
[0]); 
1007     if ( comp 
!= BI_RGB 
&& comp 
!= BI_RLE4 
&& comp 
!= BI_RLE8 
&& 
1008          comp 
!= BI_BITFIELDS 
) 
1012             wxLogError( _("DIB Header: Unknown encoding in file.") ); 
1017     stream
.Read(dbuf
, 4 * 2); 
1019     int ncolors 
= wxINT32_SWAP_ON_BE( (int)dbuf
[0] ); 
1022     /* some more sanity checks */ 
1023     if (((comp 
== BI_RLE4
) && (bpp 
!= 4)) || 
1024         ((comp 
== BI_RLE8
) && (bpp 
!= 8)) || 
1025         ((comp 
== BI_BITFIELDS
) && (bpp 
!= 16 && bpp 
!= 32))) 
1029             wxLogError( _("DIB Header: Encoding doesn't match bitdepth.") ); 
1034     //read DIB; this is the BMP image or the XOR part of an icon image 
1035     if ( !DoLoadDib(image
, width
, height
, bpp
, ncolors
, comp
, offset
, stream
, 
1036                     verbose
, IsBmp
, true) ) 
1040             wxLogError( _("Error in reading image DIB.") ); 
1047         //read Icon mask which is monochrome 
1048         //there is no palette, so we will create one 
1050         if ( !DoLoadDib(&mask
, width
, height
, 1, 2, BI_RGB
, offset
, stream
, 
1051                         verbose
, IsBmp
, false) ) 
1055                 wxLogError( _("ICO: Error in reading mask DIB.") ); 
1059         image
->SetMaskFromImage(mask
, 255, 255, 255); 
1063     // the resolution in the bitmap header is in meters, convert to centimeters 
1064     image
->SetOption(wxIMAGE_OPTION_RESOLUTIONUNIT
, wxIMAGE_RESOLUTION_CM
); 
1065     image
->SetOption(wxIMAGE_OPTION_RESOLUTIONX
, dbuf
[2]/100); 
1066     image
->SetOption(wxIMAGE_OPTION_RESOLUTIONY
, dbuf
[3]/100); 
1071 bool wxBMPHandler::LoadFile(wxImage 
*image
, wxInputStream
& stream
, 
1072                             bool verbose
, int WXUNUSED(index
)) 
1074     // Read a single DIB fom the file: 
1075     return LoadDib(image
, stream
, verbose
, true/*isBmp*/); 
1078 bool wxBMPHandler::DoCanRead(wxInputStream
& stream
) 
1080     unsigned char hdr
[2]; 
1082     if ( !stream
.Read(hdr
, WXSIZEOF(hdr
)) )     // it's ok to modify the stream position here 
1085     // do we have the BMP file signature? 
1086     return hdr
[0] == 'B' && hdr
[1] == 'M'; 
1089 #endif // wxUSE_STREAMS 
1093 //----------------------------------------------------------------------------- 
1095 //----------------------------------------------------------------------------- 
1097 IMPLEMENT_DYNAMIC_CLASS(wxICOHandler
, wxBMPHandler
) 
1103     wxUint8         bWidth
;               // Width of the image 
1104     wxUint8         bHeight
;              // Height of the image (times 2) 
1105     wxUint8         bColorCount
;          // Number of colors in image (0 if >=8bpp) 
1106     wxUint8         bReserved
;            // Reserved 
1108     // these two are different in icons and cursors: 
1110     wxUint16        wPlanes
;              // Color Planes   or  XHotSpot 
1111     wxUint16        wBitCount
;            // Bits per pixel or  YHotSpot 
1113     wxUint32        dwBytesInRes
;         // how many bytes in this resource? 
1114     wxUint32        dwImageOffset
;        // where in the file is this image 
1119     wxUint16     idReserved
;   // Reserved 
1120     wxUint16     idType
;       // resource type (1 for icons, 2 for cursors) 
1121     wxUint16     idCount
;      // how many images? 
1125 bool wxICOHandler::SaveFile(wxImage 
*image
, 
1126                             wxOutputStream
& stream
, 
1130     //sanity check; icon must be less than 127 pixels high and 255 wide 
1131     if ( image
->GetHeight () > 127 ) 
1135             wxLogError(_("ICO: Image too tall for an icon.")); 
1139     if ( image
->GetWidth () > 255 ) 
1143             wxLogError(_("ICO: Image too wide for an icon.")); 
1148     const int images 
= 1; // only generate one image 
1150     // VS: This is a hack of sort - since ICO and CUR files are almost 
1151     //     identical, we have all the meat in wxICOHandler and check for 
1152     //     the actual (handler) type when the code has to distinguish between 
1154     int type 
= (this->GetType() == wxBITMAP_TYPE_CUR
) ? 2 : 1; 
1156     // write a header, (ICONDIR) 
1157     // Calculate the header size 
1158     wxUint32 offset 
= 3 * sizeof(wxUint16
); 
1161     IconDir
.idReserved 
= 0; 
1162     IconDir
.idType 
= wxUINT16_SWAP_ON_BE((wxUint16
)type
); 
1163     IconDir
.idCount 
= wxUINT16_SWAP_ON_BE((wxUint16
)images
); 
1164     stream
.Write(&IconDir
.idReserved
, sizeof(IconDir
.idReserved
)); 
1165     stream
.Write(&IconDir
.idType
, sizeof(IconDir
.idType
)); 
1166     stream
.Write(&IconDir
.idCount
, sizeof(IconDir
.idCount
)); 
1167     if ( !stream
.IsOk() ) 
1171             wxLogError(_("ICO: Error writing the image file!")); 
1176     // for each iamage write a description ICONDIRENTRY: 
1177     ICONDIRENTRY icondirentry
; 
1178     for (int img 
= 0; img 
< images
; img
++) 
1182         if ( image
->HasMask() ) 
1184             // make another image with black/white: 
1185             mask 
= image
->ConvertToMono (image
->GetMaskRed(), image
->GetMaskGreen(), image
->GetMaskBlue() ); 
1187             // now we need to change the masked regions to black: 
1188             unsigned char r 
= image
->GetMaskRed(); 
1189             unsigned char g 
= image
->GetMaskGreen(); 
1190             unsigned char b 
= image
->GetMaskBlue(); 
1191             if ( (r 
!= 0) || (g 
!= 0) || (b 
!= 0) ) 
1193                 // Go round and apply black to the masked bits: 
1195                 for (i 
= 0; i 
< mask
.GetWidth(); i
++) 
1197                     for (j 
= 0; j 
< mask
.GetHeight(); j
++) 
1199                         if ((r 
== mask
.GetRed(i
, j
)) && 
1200                             (g 
== mask
.GetGreen(i
, j
))&& 
1201                             (b 
== mask
.GetBlue(i
, j
)) ) 
1202                                 image
->SetRGB(i
, j
, 0, 0, 0 ); 
1209             // just make a black mask all over: 
1210             mask 
= image
->Copy(); 
1212             for (i 
= 0; i 
< mask
.GetWidth(); i
++) 
1213                 for (j 
= 0; j 
< mask
.GetHeight(); j
++) 
1214                     mask
.SetRGB(i
, j
, 0, 0, 0 ); 
1216         // Set the formats for image and mask 
1217         // (Windows never saves with more than 8 colors): 
1218         image
->SetOption(wxIMAGE_OPTION_BMP_FORMAT
, wxBMP_8BPP
); 
1220         // monochome bitmap: 
1221         mask
.SetOption(wxIMAGE_OPTION_BMP_FORMAT
, wxBMP_1BPP_BW
); 
1223         bool IsMask 
= false; 
1225         //calculate size and offset of image and mask 
1226         wxCountingOutputStream cStream
; 
1227         bool bResult 
= SaveDib(image
, cStream
, verbose
, IsBmp
, IsMask
); 
1232                 wxLogError(_("ICO: Error writing the image file!")); 
1238         bResult 
= SaveDib(&mask
, cStream
, verbose
, IsBmp
, IsMask
); 
1243                 wxLogError(_("ICO: Error writing the image file!")); 
1247         wxUint32 Size 
= cStream
.GetSize(); 
1249         // wxCountingOutputStream::IsOk() always returns true for now and this 
1250         // "if" provokes VC++ warnings in optimized build 
1252         if ( !cStream
.Ok() ) 
1256                 wxLogError(_("ICO: Error writing the image file!")); 
1262         offset 
= offset 
+ sizeof(ICONDIRENTRY
); 
1264         icondirentry
.bWidth 
= (wxUint8
)image
->GetWidth(); 
1265         icondirentry
.bHeight 
= (wxUint8
)(2 * image
->GetHeight()); 
1266         icondirentry
.bColorCount 
= 0; 
1267         icondirentry
.bReserved 
= 0; 
1268         icondirentry
.wPlanes 
= wxUINT16_SWAP_ON_BE(1); 
1269         icondirentry
.wBitCount 
= wxUINT16_SWAP_ON_BE(wxBMP_8BPP
); 
1270         if ( type 
== 2 /*CUR*/) 
1272             int hx 
= image
->HasOption(wxIMAGE_OPTION_CUR_HOTSPOT_X
) ? 
1273                          image
->GetOptionInt(wxIMAGE_OPTION_CUR_HOTSPOT_X
) : 
1274                          image
->GetWidth() / 2; 
1275             int hy 
= image
->HasOption(wxIMAGE_OPTION_CUR_HOTSPOT_Y
) ? 
1276                          image
->GetOptionInt(wxIMAGE_OPTION_CUR_HOTSPOT_Y
) : 
1277                          image
->GetHeight() / 2; 
1279             // actually write the values of the hot spot here: 
1280             icondirentry
.wPlanes 
= wxUINT16_SWAP_ON_BE((wxUint16
)hx
); 
1281             icondirentry
.wBitCount 
= wxUINT16_SWAP_ON_BE((wxUint16
)hy
); 
1283         icondirentry
.dwBytesInRes 
= wxUINT32_SWAP_ON_BE(Size
); 
1284         icondirentry
.dwImageOffset 
= wxUINT32_SWAP_ON_BE(offset
); 
1286         // increase size to allow for the data written: 
1290         stream
.Write(&icondirentry
.bWidth
, sizeof(icondirentry
.bWidth
)); 
1291         stream
.Write(&icondirentry
.bHeight
, sizeof(icondirentry
.bHeight
)); 
1292         stream
.Write(&icondirentry
.bColorCount
, sizeof(icondirentry
.bColorCount
)); 
1293         stream
.Write(&icondirentry
.bReserved
, sizeof(icondirentry
.bReserved
)); 
1294         stream
.Write(&icondirentry
.wPlanes
, sizeof(icondirentry
.wPlanes
)); 
1295         stream
.Write(&icondirentry
.wBitCount
, sizeof(icondirentry
.wBitCount
)); 
1296         stream
.Write(&icondirentry
.dwBytesInRes
, sizeof(icondirentry
.dwBytesInRes
)); 
1297         stream
.Write(&icondirentry
.dwImageOffset
, sizeof(icondirentry
.dwImageOffset
)); 
1298         if ( !stream
.IsOk() ) 
1302                 wxLogError(_("ICO: Error writing the image file!")); 
1307         // actually save it: 
1309         bResult 
= SaveDib(image
, stream
, verbose
, IsBmp
, IsMask
); 
1314                 wxLogError(_("ICO: Error writing the image file!")); 
1320         bResult 
= SaveDib(&mask
, stream
, verbose
, IsBmp
, IsMask
); 
1325                 wxLogError(_("ICO: Error writing the image file!")); 
1330     } // end of for loop 
1335 bool wxICOHandler::LoadFile(wxImage 
*image
, wxInputStream
& stream
, 
1336                             bool verbose
, int index
) 
1338     return DoLoadFile(image
, stream
, verbose
, index
); 
1341 bool wxICOHandler::DoLoadFile(wxImage 
*image
, wxInputStream
& stream
, 
1342                             bool WXUNUSED(verbose
), int index
) 
1344     bool bResult 
wxDUMMY_INITIALIZE(false); 
1349     stream
.Read(&IconDir
, sizeof(IconDir
)); 
1350     wxUint16 nIcons 
= wxUINT16_SWAP_ON_BE(IconDir
.idCount
); 
1352     // nType is 1 for Icons, 2 for Cursors: 
1353     wxUint16 nType 
= wxUINT16_SWAP_ON_BE(IconDir
.idType
); 
1355     // loop round the icons and choose the best one: 
1356     ICONDIRENTRY 
*pIconDirEntry 
= new ICONDIRENTRY
[nIcons
]; 
1357     ICONDIRENTRY 
*pCurrentEntry 
= pIconDirEntry
; 
1360     int iSel 
= wxNOT_FOUND
; 
1362     // remember how many bytes we read from the stream: 
1363     wxFileOffset alreadySeeked 
= sizeof(IconDir
); 
1365     for (unsigned int i 
= 0; i 
< nIcons
; i
++ ) 
1367         alreadySeeked 
+= stream
.Read(pCurrentEntry
, sizeof(ICONDIRENTRY
)).LastRead(); 
1369         // bHeight and bColorCount are wxUint8 
1370         if ( pCurrentEntry
->bWidth 
>= wMax 
) 
1372             // see if we have more colors, ==0 indicates > 8bpp: 
1373             if ( pCurrentEntry
->bColorCount 
== 0 ) 
1374                 pCurrentEntry
->bColorCount 
= 255; 
1375             if ( pCurrentEntry
->bColorCount 
>= colmax 
) 
1378                 wMax 
= pCurrentEntry
->bWidth
; 
1379                 colmax 
= pCurrentEntry
->bColorCount
; 
1388         // VS: Note that we *have* to run the loop above even if index != -1, because 
1389         //     it reads ICONDIRENTRies. 
1393     if ( iSel 
== wxNOT_FOUND 
|| iSel 
< 0 || iSel 
>= nIcons 
) 
1395         wxLogError(_("ICO: Invalid icon index.")); 
1400         // seek to selected icon: 
1401         pCurrentEntry 
= pIconDirEntry 
+ iSel
; 
1403         // NOTE: seeking a positive amount in wxFromCurrent mode allows us to 
1404         //       load even non-seekable streams (see wxInputStream::SeekI docs)! 
1405         wxFileOffset offset 
= wxUINT32_SWAP_ON_BE(pCurrentEntry
->dwImageOffset
) - alreadySeeked
; 
1406         if (offset 
!= 0 && stream
.SeekI(offset
, wxFromCurrent
) == wxInvalidOffset
) 
1409         bResult 
= LoadDib(image
, stream
, true, IsBmp
); 
1410         bool bIsCursorType 
= (this->GetType() == wxBITMAP_TYPE_CUR
) || (this->GetType() == wxBITMAP_TYPE_ANI
); 
1411         if ( bResult 
&& bIsCursorType 
&& nType 
== 2 ) 
1413             // it is a cursor, so let's set the hotspot: 
1414             image
->SetOption(wxIMAGE_OPTION_CUR_HOTSPOT_X
, wxUINT16_SWAP_ON_BE(pCurrentEntry
->wPlanes
)); 
1415             image
->SetOption(wxIMAGE_OPTION_CUR_HOTSPOT_Y
, wxUINT16_SWAP_ON_BE(pCurrentEntry
->wBitCount
)); 
1419     delete [] pIconDirEntry
; 
1424 int wxICOHandler::DoGetImageCount(wxInputStream
& stream
) 
1428     if (stream
.Read(&IconDir
, sizeof(IconDir
)).LastRead() != sizeof(IconDir
)) 
1429              // it's ok to modify the stream position here 
1432     return (int)wxUINT16_SWAP_ON_BE(IconDir
.idCount
); 
1435 bool wxICOHandler::DoCanRead(wxInputStream
& stream
) 
1437     return CanReadICOOrCUR(&stream
, 1 /*for identifying an icon*/); 
1441 #endif // wxUSE_STREAMS 
1444 //----------------------------------------------------------------------------- 
1446 //----------------------------------------------------------------------------- 
1448 IMPLEMENT_DYNAMIC_CLASS(wxCURHandler
, wxICOHandler
) 
1452 bool wxCURHandler::DoCanRead(wxInputStream
& stream
) 
1454     return CanReadICOOrCUR(&stream
, 2 /*for identifying a cursor*/); 
1457 #endif // wxUSE_STREAMS 
1459 //----------------------------------------------------------------------------- 
1461 //----------------------------------------------------------------------------- 
1463 IMPLEMENT_DYNAMIC_CLASS(wxANIHandler
, wxCURHandler
) 
1467 bool wxANIHandler::LoadFile(wxImage 
*image
, wxInputStream
& stream
, 
1468                             bool WXUNUSED(verbose
), int index
) 
1470     wxANIDecoder decoder
; 
1471     if (!decoder
.Load(stream
)) 
1474     return decoder
.ConvertToImage(index 
!= -1 ? (size_t)index 
: 0, image
); 
1477 bool wxANIHandler::DoCanRead(wxInputStream
& stream
) 
1480     return decod
.CanRead(stream
); 
1481              // it's ok to modify the stream position here 
1484 int wxANIHandler::DoGetImageCount(wxInputStream
& stream
) 
1486     wxANIDecoder decoder
; 
1487     if (!decoder
.Load(stream
))  // it's ok to modify the stream position here 
1490     return decoder
.GetFrameCount(); 
1493 static bool CanReadICOOrCUR(wxInputStream 
*stream
, wxUint16 resourceType
) 
1496     if ( !stream
->Read(&iconDir
, sizeof(iconDir
)) )     // it's ok to modify the stream position here 
1501     return !iconDir
.idReserved 
// reserved, must be 0 
1502         && wxUINT16_SWAP_ON_BE(iconDir
.idType
) == resourceType 
// either 1 or 2 
1503         && iconDir
.idCount
; // must contain at least one image 
1506 #endif // wxUSE_STREAMS 
1508 #endif // wxUSE_ICO_CUR 
1510 #endif // wxUSE_IMAGE