1 /******************************************************************************* 
   5  *  DESCRIPTION : Routines for dealing with Device Independent Bitmaps.        * 
   9  *                wxReadDIB()           - Reads a DIB                            * 
  11  *                  WriteDIB()              - Writes a global handle in CF_DIB format* 
  14  *                  wxPaletteSize()       - Calculates the palette size in bytes   * 
  17  *                  DibNumColors()      - Determines the number of colors in DIB * 
  19  *                  DibFromBitmap()     - Creates a DIB repr. the DDB passed in. * 
  22  *                  lread()              - Private routine to read more than 64k  * 
  24  *                  lwrite()              - Private routine to write more than 64k * 
  26  *******************************************************************************/ 
  28 // For compilers that support precompilation, includes "wx.h". 
  29 #include "wx/wxprec.h" 
  31 #if defined(__BORLANDC__) 
  36 #include "wx/bitmap.h" 
  45 #if !defined(__MWERKS__) && !defined(__SALFORDC__) 
  49 #include "wx/msw/dib.h" 
  52 #ifndef wxUSE_NORLANDER_HEADERS 
  54 #include "wx/msw/gnuwin32/extra.h" 
  60 /* flags for _lseek */ 
  66 #define MAXREAD  32768                                 /* Number of bytes to be read during */ 
  67                                                  /* each read operation.                         */ 
  69 /* Header signatutes for various resources */ 
  70 #define BFT_ICON         0x4349         /* 'IC' */ 
  71 #define BFT_BITMAP 0x4d42         /* 'BM' */ 
  72 #define BFT_CURSOR 0x5450         /* 'PT(' */ 
  74 /* macro to determine if resource is a DIB */ 
  75 #define ISDIB(bft) ((bft) == BFT_BITMAP) 
  77 /* Macro to align given value to the closest DWORD (unsigned long ) */ 
  78 #define ALIGNULONG(i)        ((i+3)/4*4) 
  80 /* Macro to determine to round off the given value to the closest byte */ 
  81 #define WIDTHBYTES(i)        ((i+31)/32*4) 
  83 #define PALVERSION                0x300 
  84 #define MAXPALETTE        256          /* max. # supported palette entries */ 
  86 static DWORD PASCAL 
lread(int fh
, VOID FAR 
*pv
, DWORD ul
); 
  87 static DWORD PASCAL 
lwrite(int fh
, VOID FAR 
*pv
, DWORD ul
); 
  89 static BOOL        
WriteDIB (LPTSTR szFile
,HANDLE hdib
); 
  90 WORD        
wxPaletteSize (VOID FAR 
* pv
); // This is non-static as some apps use it externally 
  91 static WORD        
DibNumColors (VOID FAR 
* pv
); 
  92 // HANDLE        DibFromBitmap (HBITMAP hbm, DWORD biStyle, WORD biBits, HPALETTE hpal); 
  93 static BOOL PASCAL 
MakeBitmapAndPalette(HDC
,HANDLE
,HPALETTE 
*,HBITMAP 
*); 
  95 /**************************************************************************** 
  97  *  FUNCTION   : WriteDIB(LPSTR szFile,HANDLE hdib)                            * 
  99  *  PURPOSE    : Write a global handle in CF_DIB format to a file.            * 
 101  *  RETURNS    : TRUE  - if successful.                                     * 
 102  *                 FALSE - otherwise                                            * 
 104  ****************************************************************************/ 
 106 static BOOL 
WriteDIB(LPTSTR szFile
, HANDLE hdib
) 
 108         BITMAPFILEHEADER        hdr
; 
 109         LPBITMAPINFOHEADER  lpbi
; 
 116         fh 
= OpenFile(wxFNCONV(szFile
), &of
, OF_CREATE 
| OF_READWRITE
); 
 120 #ifdef __WINDOWS_386__ 
 121         lpbi 
= (LPBITMAPINFOHEADER
) MK_FP32(GlobalLock(hdib
)); 
 123         lpbi 
= (LPBITMAPINFOHEADER
) GlobalLock(hdib
); 
 125     /* Fill in the fields of the file header */ 
 126         hdr
.bfType 
= BFT_BITMAP
; 
 127         hdr
.bfSize 
= GlobalSize(hdib
) + sizeof(BITMAPFILEHEADER
); 
 130         hdr
.bfOffBits 
= (DWORD
) sizeof(BITMAPFILEHEADER
) + lpbi
->biSize 
+ 
 133     /* Write the file header */ 
 134         _lwrite(fh
, (LPSTR
) &hdr
, sizeof(BITMAPFILEHEADER
)); 
 136     /* Write the DIB header and the bits */ 
 137         lwrite(fh
, (LPSTR
) lpbi
, GlobalSize(hdib
)); 
 144 /**************************************************************************** 
 146  *  FUNCTION   :  wxPaletteSize(VOID FAR * pv)                                    * 
 148  *  PURPOSE    :  Calculates the palette size in bytes. If the info. block  * 
 149  *                  is of the BITMAPCOREHEADER type, the number of colors is  * 
 150  *                  multiplied by 3 to give the palette size, otherwise the   * 
 151  *                  number of colors is multiplied by 4.                                                                * 
 153  *  RETURNS    :  Palette size in number of bytes.                            * 
 155  ****************************************************************************/ 
 157 WORD 
wxPaletteSize(VOID FAR 
* pv
) 
 159         LPBITMAPINFOHEADER lpbi
; 
 162         lpbi 
= (LPBITMAPINFOHEADER
) pv
; 
 163         NumColors 
= DibNumColors(lpbi
); 
 165         if (lpbi
->biSize 
== sizeof(BITMAPCOREHEADER
)) 
 166                 return NumColors 
* sizeof(RGBTRIPLE
); 
 168                 return NumColors 
* sizeof(RGBQUAD
); 
 171 /**************************************************************************** 
 173  *  FUNCTION   : DibNumColors(VOID FAR * pv)                                    * 
 175  *  PURPOSE    : Determines the number of colors in the DIB by looking at   * 
 176  *                 the BitCount filed in the info block.                            * 
 178  *  RETURNS    : The number of colors in the DIB.                            * 
 180  ****************************************************************************/ 
 182 static WORD 
DibNumColors(VOID FAR 
*pv
) 
 185         BITMAPINFOHEADER        
*lpbi
; 
 186         BITMAPCOREHEADER        
*lpbc
; 
 188         lpbi 
= ((BITMAPINFOHEADER
*) pv
); 
 189         lpbc 
= ((BITMAPCOREHEADER
*) pv
); 
 191     /*        With the BITMAPINFO format headers, the size of the palette 
 192      *        is in biClrUsed, whereas in the BITMAPCORE - style headers, it 
 193      *        is dependent on the bits per pixel ( = 2 raised to the power of 
 196         if (lpbi
->biSize 
!= sizeof(BITMAPCOREHEADER
)) { 
 197                 if (lpbi
->biClrUsed 
!= 0) 
 198                         return (WORD
) lpbi
->biClrUsed
; 
 199                 bits 
= lpbi
->biBitCount
; 
 202                 bits 
= lpbc
->bcBitCount
; 
 212                 /* A 24 bitcount DIB has no color table */ 
 217 /**************************************************************************** 
 219  *  FUNCTION   : DibFromBitmap()                                            * 
 221  *  PURPOSE    : Will create a global memory block in DIB format that            * 
 222  *                 represents the Device-dependent bitmap (DDB) passed in.    * 
 224  *  RETURNS    : A handle to the DIB                                            * 
 226  ****************************************************************************/ 
 229 static HANDLE 
DibFromBitmap(HBITMAP hbm
, DWORD biStyle
, WORD biBits
, HPALETTE hpal
) 
 233         BITMAPINFOHEADER FAR 
*lpbi
; 
 243                 hpal 
= GetStockObject(DEFAULT_PALETTE
); 
 245         GetObject(hbm
, sizeof (bm
), (LPSTR
) &bm
); 
 248                 biBits 
= bm
.bmPlanes 
* bm
.bmBitsPixel
; 
 250         bi
.biSize 
= sizeof(BITMAPINFOHEADER
); 
 251         bi
.biWidth 
= bm
.bmWidth
; 
 252         bi
.biHeight 
= bm
.bmHeight
; 
 254         bi
.biBitCount 
= biBits
; 
 255         bi
.biCompression 
= biStyle
; 
 257         bi
.biXPelsPerMeter 
= 0; 
 258         bi
.biYPelsPerMeter 
= 0; 
 260         bi
.biClrImportant 
= 0; 
 262         dwLen 
= bi
.biSize 
+ wxPaletteSize(&bi
); 
 264         hdc 
= GetDC((HWND
) NULL
); 
 265         hpal 
= SelectPalette(hdc
, hpal
, FALSE
); 
 268         hdib 
= GlobalAlloc(GHND
, dwLen
); 
 271                 SelectPalette(hdc
, hpal
, FALSE
); 
 272                 ReleaseDC(NULL
, hdc
); 
 276 #ifdef __WINDOWS_386__ 
 277         lpbi 
= (BITMAPINFOHEADER FAR 
*) MK_FP32(GlobalLock(hdib
)); 
 279         lpbi 
= (BITMAPINFOHEADER FAR 
*) GlobalLock(hdib
); 
 284     /*        call GetDIBits with a NULL lpBits param, so it will calculate the 
 285      *  biSizeImage field for us 
 287         GetDIBits(hdc
, hbm
, 0, (WORD
) bi
.biHeight
, 
 288                  NULL
, (LPBITMAPINFO
) lpbi
, DIB_RGB_COLORS
); 
 293     /* If the driver did not fill in the biSizeImage field, make one up */ 
 294         if (bi
.biSizeImage 
== 0) { 
 295                 bi
.biSizeImage 
= WIDTHBYTES((DWORD
)bm
.bmWidth 
* biBits
) * bm
.bmHeight
; 
 297                 if (biStyle 
!= BI_RGB
) 
 298                         bi
.biSizeImage 
= (bi
.biSizeImage 
* 3) / 2; 
 301     /*        realloc the buffer big enough to hold all the bits */ 
 302         dwLen 
= bi
.biSize 
+ wxPaletteSize(&bi
) + bi
.biSizeImage
; 
 303         if (h 
= GlobalReAlloc(hdib
, dwLen
, 0)) 
 309                 SelectPalette(hdc
, hpal
, FALSE
); 
 310                 ReleaseDC(NULL
, hdc
); 
 314     /*        call GetDIBits with a NON-NULL lpBits param, and actualy get the 
 317 #ifdef __WINDOWS_386__ 
 318         lpbi 
= (BITMAPINFOHEADER FAR 
*) MK_FP32(GlobalLock(hdib
)); 
 320         lpbi 
= (BITMAPINFOHEADER FAR 
*) GlobalLock(hdib
); 
 327                  (LPSTR
) lpbi 
+ (WORD
) lpbi
->biSize 
+ wxPaletteSize(lpbi
), 
 328                  (LPBITMAPINFO
) lpbi
, DIB_RGB_COLORS
) == 0) { 
 331                 SelectPalette(hdc
, hpal
, FALSE
); 
 332                 ReleaseDC((HWND
) NULL
, hdc
); 
 339         SelectPalette(hdc
, hpal
, FALSE
); 
 340         ReleaseDC(NULL
, hdc
); 
 345  /************* PRIVATE ROUTINES TO READ/WRITE MORE THAN 64K ***************/ 
 346 /**************************************************************************** 
 348  *  FUNCTION   : lread(int fh, VOID FAR *pv, DWORD ul)                            * 
 350  *  PURPOSE    : Reads data in steps of 32k till all the data has been read.* 
 352  *  RETURNS    : 0 - If read did not proceed correctly.                     * 
 353  *                 number of bytes read otherwise.                            * 
 355  ****************************************************************************/ 
 357 static DWORD PASCAL 
lread(int fh
, void far 
*pv
, DWORD ul
) 
 360 #if defined(WINNT) || defined(__WIN32__) || defined(__WIN32__) || defined(__WXWINE__) 
 361         BYTE 
*hp 
= (BYTE 
*) pv
; 
 363         BYTE huge 
*hp 
= (BYTE huge 
*) pv
; 
 365         while (ul 
> (DWORD
) MAXREAD
) { 
 366                 if (_lread(fh
, (LPSTR
) hp
, (WORD
) MAXREAD
) != MAXREAD
) 
 371         if (_lread(fh
, (LPSTR
) hp
, (WORD
) ul
) != (WORD
) ul
) 
 376 /**************************************************************************** 
 378  *  FUNCTION   : lwrite(int fh, VOID FAR *pv, DWORD ul)                     * 
 380  *  PURPOSE    : Writes data in steps of 32k till all the data is written.  * 
 382  *  RETURNS    : 0 - If write did not proceed correctly.                    * 
 383  *                 number of bytes written otherwise.                            * 
 385  ****************************************************************************/ 
 387 static DWORD PASCAL 
lwrite(int fh
, VOID FAR 
*pv
, DWORD ul
) 
 390 #if defined(WINNT) || defined(__WIN32__) || defined(__WIN32__) || defined(__WXWINE__) 
 391         BYTE 
*hp 
= (BYTE 
*) pv
; 
 393         BYTE huge 
*hp 
= (BYTE huge 
*) pv
; 
 395         while (ul 
> MAXREAD
) { 
 396                 if (_lwrite(fh
, (LPSTR
) hp
, (WORD
) MAXREAD
) != MAXREAD
) 
 401         if (_lwrite(fh
, (LPSTR
) hp
, (WORD
) ul
) != (WORD
) ul
) 
 406 /**************************************************************************** 
 408  *  FUNCTION   : ReadDIB(hWnd) 
 410  *  PURPOSE    : Reads a DIB from a file, obtains a handle to its 
 411  *                 BITMAPINFO struct. and loads the DIB.  Once the DIB 
 412  *               is loaded, the function also creates a bitmap and 
 413  *               palette out of the DIB for a device-dependent form. 
 415  *  RETURNS    : TRUE  - DIB loaded and bitmap/palette created 
 416  *                       The DIBINIT structure pointed to by pInfo is 
 417  *                       filled with the appropriate handles. 
 420  ****************************************************************************/ 
 421 BOOL 
wxReadDIB(LPTSTR lpFileName
, HBITMAP 
*bitmap
, HPALETTE 
*palette
) 
 424     LPBITMAPINFOHEADER lpbi
; 
 431     BOOL bCoreHead 
= FALSE
; 
 434     /* Open the file and get a handle to it's BITMAPINFO */ 
 436     fh 
= OpenFile (wxFNCONV(lpFileName
), &of
, OF_READ
); 
 438         wxLogError(_("Can't open file '%s'"), lpFileName
); 
 442     hDIB 
= GlobalAlloc(GHND
, (DWORD
)(sizeof(BITMAPINFOHEADER
) + 
 443                                             256 * sizeof(RGBQUAD
))); 
 447 #ifdef __WINDOWS_386__ 
 448     lpbi 
= (LPBITMAPINFOHEADER
)MK_FP32(GlobalLock(hDIB
)); 
 450     lpbi 
= (LPBITMAPINFOHEADER
)GlobalLock(hDIB
); 
 453     /* read the BITMAPFILEHEADER */ 
 454     if (sizeof (bf
) != _lread (fh
, (LPSTR
)&bf
, sizeof (bf
))) 
 457     if (bf
.bfType 
!= 0x4d42)        /* 'BM' */ 
 460     if (sizeof(BITMAPCOREHEADER
) != _lread (fh
, (LPSTR
)lpbi
, sizeof(BITMAPCOREHEADER
))) 
 463     if (lpbi
->biSize 
== sizeof(BITMAPCOREHEADER
)) 
 465         lpbi
->biSize 
= sizeof(BITMAPINFOHEADER
); 
 466         lpbi
->biBitCount 
= ((LPBITMAPCOREHEADER
)lpbi
)->bcBitCount
; 
 467         lpbi
->biPlanes 
= ((LPBITMAPCOREHEADER
)lpbi
)->bcPlanes
; 
 468         lpbi
->biHeight 
= ((LPBITMAPCOREHEADER
)lpbi
)->bcHeight
; 
 469         lpbi
->biWidth 
= ((LPBITMAPCOREHEADER
)lpbi
)->bcWidth
; 
 474             // get to the start of the header and read INFOHEADER 
 475         _llseek(fh
,sizeof(BITMAPFILEHEADER
),SEEK_SET
); 
 476         if (sizeof(BITMAPINFOHEADER
) != _lread (fh
, (LPSTR
)lpbi
, sizeof(BITMAPINFOHEADER
))) 
 480     nNumColors 
= (WORD
)lpbi
->biClrUsed
; 
 481     if ( nNumColors 
== 0 ) 
 483         /* no color table for 24-bit, default size otherwise */ 
 484         if (lpbi
->biBitCount 
!= 24) 
 485             nNumColors 
= 1 << lpbi
->biBitCount
;        /* standard size table */ 
 488     /*        fill in some default values if they are zero */ 
 489     if (lpbi
->biClrUsed 
== 0) 
 490         lpbi
->biClrUsed 
= nNumColors
; 
 492     if (lpbi
->biSizeImage 
== 0) 
 494         lpbi
->biSizeImage 
= ((((lpbi
->biWidth 
* (DWORD
)lpbi
->biBitCount
) + 31) & ~31) >> 3) 
 498     /* get a proper-sized buffer for header, color table and bits */ 
 500     hDIB 
= GlobalReAlloc(hDIB
, lpbi
->biSize 
+ 
 501                                             nNumColors 
* sizeof(RGBQUAD
) + 
 502                                         lpbi
->biSizeImage
, 0); 
 503     if (!hDIB
)        /* can't resize buffer for loading */ 
 506 #ifdef __WINDOWS_386__ 
 507     lpbi 
= (LPBITMAPINFOHEADER
)MK_FP32(GlobalLock(hDIB
)); 
 509     lpbi 
= (LPBITMAPINFOHEADER
)GlobalLock(hDIB
); 
 512     /* read the color table */ 
 514         _lread(fh
, (LPSTR
)(lpbi
) + lpbi
->biSize
, nNumColors 
* sizeof(RGBQUAD
)); 
 519         RGBTRIPLE FAR 
*pTriple
; 
 521         _lread(fh
, (LPSTR
)(lpbi
) + lpbi
->biSize
, nNumColors 
* sizeof(RGBTRIPLE
)); 
 523         pQuad 
= (RGBQUAD FAR 
*)((LPSTR
)lpbi 
+ lpbi
->biSize
); 
 524         pTriple 
= (RGBTRIPLE FAR 
*) pQuad
; 
 525         for (i 
= nNumColors 
- 1; i 
>= 0; i
--) 
 527             pQuad
[i
].rgbRed 
= pTriple
[i
].rgbtRed
; 
 528             pQuad
[i
].rgbBlue 
= pTriple
[i
].rgbtBlue
; 
 529             pQuad
[i
].rgbGreen 
= pTriple
[i
].rgbtGreen
; 
 530             pQuad
[i
].rgbReserved 
= 0; 
 534     /* offset to the bits from start of DIB header */ 
 535     offBits 
= (WORD
)lpbi
->biSize 
+ nNumColors 
* sizeof(RGBQUAD
); 
 537     if (bf
.bfOffBits 
!= 0L) 
 539         _llseek(fh
,bf
.bfOffBits
,SEEK_SET
); 
 542     if (lpbi
->biSizeImage 
== lread(fh
, (LPSTR
)lpbi 
+ offBits
, lpbi
->biSizeImage
)) 
 547         if (!MakeBitmapAndPalette(hDC
, hDIB
, palette
, 
 572 /**************************************************************************** 
 574  *  FUNCTION   : MakeBitmapAndPalette 
 576  *  PURPOSE    : Given a DIB, creates a bitmap and corresponding palette 
 577  *               to be used for a device-dependent representation of 
 580  *  RETURNS    : TRUE  --> success. phPal and phBitmap are filled with 
 581  *                         appropriate handles.  Caller is responsible 
 582  *                         for freeing objects. 
 583  *               FALSE --> unable to create objects.  both pointer are 
 586  ****************************************************************************/ 
 587 static BOOL PASCAL 
MakeBitmapAndPalette(HDC hDC
, HANDLE hDIB
, 
 588                         HPALETTE 
* phPal
, HBITMAP 
* phBitmap
) 
 590     LPBITMAPINFOHEADER lpInfo
; 
 593     HPALETTE hPalette
, hOldPal
; 
 596 #ifdef __WINDOWS_386__ 
 597     lpInfo 
= (LPBITMAPINFOHEADER
) MK_FP32(GlobalLock(hDIB
)); 
 599     lpInfo 
= (LPBITMAPINFOHEADER
) GlobalLock(hDIB
); 
 602     hPalette 
= wxMakeDIBPalette(lpInfo
); 
 605         // Need to realize palette for converting DIB to bitmap. 
 606         hOldPal 
= SelectPalette(hDC
, hPalette
, TRUE
); 
 609         lpBits 
= (LPSTR
)lpInfo 
+ (WORD
)lpInfo
->biSize 
+ 
 610                 (WORD
)lpInfo
->biClrUsed 
* sizeof(RGBQUAD
); 
 611         hBitmap 
= CreateDIBitmap(hDC
, lpInfo
, CBM_INIT
, lpBits
, 
 612                                 (LPBITMAPINFO
)lpInfo
, DIB_RGB_COLORS
); 
 614         SelectPalette(hDC
, hOldPal
, TRUE
); 
 618             DeleteObject(hPalette
); 
 629 /**************************************************************************** 
 631  *  FUNCTION   : wxMakeDIBPalette(lpInfo)                                            * 
 633  *  PURPOSE    : Given a BITMAPINFOHEADER, create a palette based on 
 637  *  RETURNS    : non-zero - handle of a corresponding palette 
 638  *                 zero - unable to create palette 
 640  ****************************************************************************/ 
 641 HPALETTE 
wxMakeDIBPalette(LPBITMAPINFOHEADER lpInfo
) 
 651     /* since biClrUsed field was filled during the loading of the DIB, 
 652     ** we know it contains the number of colors in the color table. 
 654     if (lpInfo
->biClrUsed
) 
 657         npPal = (NPLOGPALETTE)LocalAlloc(LMEM_FIXED, sizeof(LOGPALETTE) + 
 658                                 (WORD)lpInfo->biClrUsed * sizeof(PALETTEENTRY)); 
 660         npPal 
= (LPLOGPALETTE
)malloc(sizeof(LOGPALETTE
) + 
 661                                 (WORD
)lpInfo
->biClrUsed 
* sizeof(PALETTEENTRY
)); 
 665         npPal
->palVersion 
= 0x300; 
 666         npPal
->palNumEntries 
= (WORD
)lpInfo
->biClrUsed
; 
 668         /* get pointer to the color table */ 
 669         lpRGB 
= (RGBQUAD FAR 
*)((LPSTR
)lpInfo 
+ lpInfo
->biSize
); 
 671         /* copy colors from the color table to the LogPalette structure */ 
 672         for (i 
= 0; i 
< lpInfo
->biClrUsed
; i
++, lpRGB
++) 
 674             npPal
->palPalEntry
[i
].peRed 
= lpRGB
->rgbRed
; 
 675             npPal
->palPalEntry
[i
].peGreen 
= lpRGB
->rgbGreen
; 
 676             npPal
->palPalEntry
[i
].peBlue 
= lpRGB
->rgbBlue
; 
 677             npPal
->palPalEntry
[i
].peFlags 
= 0; 
 680         hLogPal 
= CreatePalette((LPLOGPALETTE
)npPal
); 
 681 //        LocalFree((HANDLE)npPal); 
 687     /* 24-bit DIB with no color table.  return default palette.  Another 
 688     ** option would be to create a 256 color "rainbow" palette to provide 
 689     ** some good color choices. 
 692         return((HPALETTE
) GetStockObject(DEFAULT_PALETTE
)); 
 697 bool wxLoadIntoBitmap(wxChar 
*filename
, wxBitmap 
*bitmap
, wxPalette 
**pal
) 
 702   bool success 
= (wxReadDIB(filename
, &hBitmap
, &hPalette
) != 0); 
 706     DeleteObject(hPalette
); 
 714       *pal 
= new wxPalette
; 
 715       (*pal
)->SetHPALETTE((WXHPALETTE
) hPalette
); 
 718       DeleteObject(hPalette
); 
 726     GetObject(hBitmap
, sizeof(bm
), (LPSTR
)&bm
); 
 728     bitmap
->SetHBITMAP((WXHBITMAP
) hBitmap
); 
 729     bitmap
->SetWidth(bm
.bmWidth
); 
 730     bitmap
->SetHeight(bm
.bmHeight
); 
 731     bitmap
->SetDepth(bm
.bmPlanes 
* bm
.bmBitsPixel
); 
 732 #if WXWIN_COMPATIBILITY_2 
 734 #endif // WXWIN_COMPATIBILITY_2 
 740 wxBitmap 
*wxLoadBitmap(wxChar 
*filename
, wxPalette 
**pal
) 
 742   wxBitmap 
*bitmap 
= new wxBitmap
; 
 743   if (wxLoadIntoBitmap(filename
, bitmap
, pal
)) 
 752 //--------------------------------------------------------------------- 
 754 // Function:   InitBitmapInfoHeader 
 756 // Purpose:    Does a "standard" initialization of a BITMAPINFOHEADER, 
 757 //             given the Width, Height, and Bits per Pixel for the 
 760 //             By standard, I mean that all the relevant fields are set 
 761 //             to the specified values.  biSizeImage is computed, the 
 762 //             biCompression field is set to "no compression," and all 
 763 //             other fields are 0. 
 765 //             Note that DIBs only allow BitsPixel values of 1, 4, 8, or 
 766 //             24.  This routine makes sure that one of these values is 
 767 //             used (whichever is most appropriate for the specified 
 770 // Parms:      lpBmInfoHdr == Far pointer to a BITMAPINFOHEADER structure 
 772 //             dwWidth     == Width of DIB (not in Win 3.0 & 3.1, high 
 774 //             dwHeight    == Height of DIB (not in Win 3.0 & 3.1, high 
 776 //             nBPP        == Bits per Pixel for the DIB. 
 778 // History:   Date      Reason 
 781 //--------------------------------------------------------------------- 
 783 static void InitBitmapInfoHeader (LPBITMAPINFOHEADER lpBmInfoHdr
, 
 788 //   _fmemset (lpBmInfoHdr, 0, sizeof (BITMAPINFOHEADER)); 
 789    memset (lpBmInfoHdr
, 0, sizeof (BITMAPINFOHEADER
)); 
 791    lpBmInfoHdr
->biSize      
= sizeof (BITMAPINFOHEADER
); 
 792    lpBmInfoHdr
->biWidth     
= dwWidth
; 
 793    lpBmInfoHdr
->biHeight    
= dwHeight
; 
 794    lpBmInfoHdr
->biPlanes    
= 1; 
 809    lpBmInfoHdr
->biBitCount  
= nBPP
; 
 810    lpBmInfoHdr
->biSizeImage 
= WIDTHBYTES (dwWidth 
* nBPP
) * dwHeight
; 
 816 LPSTR 
wxFindDIBBits (LPSTR lpbi
) 
 818    return (lpbi 
+ *(LPDWORD
)lpbi 
+ wxPaletteSize (lpbi
)); 
 821 //--------------------------------------------------------------------- 
 823 // Function:   BitmapToDIB 
 825 // Purpose:    Given a device dependent bitmap and a palette, returns 
 826 //             a handle to global memory with a DIB spec in it.  The 
 827 //             DIB is rendered using the colors of the palette passed in. 
 829 //             Stolen almost verbatim from ShowDIB. 
 831 // Parms:      hBitmap == Handle to device dependent bitmap compatible 
 832 //                        with default screen display device. 
 833 //             hPal    == Palette to render the DDB with.  If it's NULL, 
 834 //                        use the default palette. 
 836 // History:   Date      Reason 
 839 //--------------------------------------------------------------------- 
 841 HANDLE 
wxBitmapToDIB (HBITMAP hBitmap
, HPALETTE hPal
) 
 844    BITMAPINFOHEADER   bmInfoHdr
; 
 845    LPBITMAPINFOHEADER lpbmInfoHdr
; 
 849    HPALETTE           hOldPal 
= NULL
; 
 851       // Do some setup -- make sure the Bitmap passed in is valid, 
 852       //  get info on the bitmap (like its height, width, etc.), 
 853       //  then setup a BITMAPINFOHEADER. 
 858    if (!GetObject (hBitmap
, sizeof (Bitmap
), (LPSTR
) &Bitmap
)) 
 861    InitBitmapInfoHeader (&bmInfoHdr
, 
 864                          Bitmap
.bmPlanes 
* Bitmap
.bmBitsPixel
); 
 867       // Now allocate memory for the DIB.  Then, set the BITMAPINFOHEADER 
 868       //  into this memory, and find out where the bitmap bits go. 
 870    hDIB 
= GlobalAlloc (GHND
, sizeof (BITMAPINFOHEADER
) + 
 871              wxPaletteSize ((LPSTR
) &bmInfoHdr
) + bmInfoHdr
.biSizeImage
); 
 876 #ifdef __WINDOWS_386__ 
 877    lpbmInfoHdr  
= (LPBITMAPINFOHEADER
) MK_FP32(GlobalLock (hDIB
)); 
 879    lpbmInfoHdr  
= (LPBITMAPINFOHEADER
) GlobalLock (hDIB
); 
 882    *lpbmInfoHdr 
= bmInfoHdr
; 
 883    lpBits       
= wxFindDIBBits ((LPSTR
) lpbmInfoHdr
); 
 886       // Now, we need a DC to hold our bitmap.  If the app passed us 
 887       //  a palette, it should be selected into the DC. 
 889    hMemDC       
= GetDC (NULL
); 
 893       hOldPal 
= SelectPalette (hMemDC
, hPal
, FALSE
); 
 894       RealizePalette (hMemDC
); 
 899       // We're finally ready to get the DIB.  Call the driver and let 
 900       //  it party on our bitmap.  It will fill in the color table, 
 901       //  and bitmap bits of our global memory block. 
 903    if (!GetDIBits (hMemDC
, 
 908                    (LPBITMAPINFO
) lpbmInfoHdr
, 
 919       // Finally, clean up and return. 
 922       SelectPalette (hMemDC
, hOldPal
, FALSE
); 
 924    ReleaseDC (NULL
, hMemDC
); 
 929 bool wxSaveBitmap(wxChar 
*filename
, wxBitmap 
*bitmap
, wxPalette 
*colourmap
) 
 931   HPALETTE hPalette 
= 0; 
 933     hPalette 
= (HPALETTE
) colourmap
->GetHPALETTE(); 
 935   HANDLE dibHandle 
= wxBitmapToDIB((HBITMAP
) bitmap
->GetHBITMAP(), hPalette
); 
 938     bool success 
= (WriteDIB(filename
, dibHandle
) != 0); 
 939     GlobalFree(dibHandle
);