1 //* Written by Microsoft Product Support Services, Windows Developer Support. * 
   2 //***************************************************************************** 
   3 //   (C) Copyright Microsoft Corp. 1993.  All rights reserved. 
   4 //   You have a royalty-free right to use, modify, reproduce and 
   5 //   distribute the Sample Files (and/or any modified version) in 
   6 //   any way you find useful, provided that you agree that 
   7 //   Microsoft has no warranty obligations or liability for any 
   8 //   Sample Application Files which are modified. 
  10 // Modified by Petr Smilauer, March 1994 for wxWin library purposes! 
  12 // For compilers that support precompilation, includes "wx.h". 
  13 #include "wx/wxprec.h" 
  22 #if defined(__MWERKS__) 
  27 #ifdef __GNUWIN32_OLD__ 
  28     #include "wx/msw/gnuwin32/extra.h" 
  31 #include "wx/wxchar.h" 
  32 #include "wx/msw/curicop.h" 
  33 #include "wx/msw/curico.h" 
  34 #include "wx/string.h" 
  36 //***************************************************************************** 
  37 //* Function : ReadIconFile()                                                 * 
  38 //* Purpose  : Reads an icon resource file and creates an icon based on that  * 
  40 //* Parameters : char *szFileName - The icon resource file.                   * 
  41 //* Returns :  A handle to an icon. The handle will be NULL if an icon cannot * 
  42 //*            be created for any reason.                                     * 
  43 //***************************************************************************** 
  45 HICON 
ReadIconFile( wxChar 
*szFileName
, HINSTANCE hInst
, int *W
, int *H
) 
  49   if( (hDIB 
= ReadIcon(szFileName
, W
, H
)) == (HANDLE
) NULL
) 
  50                                           // read the icon DIB from file 
  52   hIcon 
= MakeIcon( hDIB
, hInst
);         // create an icon from DIB 
  57 //***************************************************************************** 
  58 //* Function : CursorToIcon()                                                 * 
  59 //* Purpose  : Reads a cursor resource file and creates an icon based on that * 
  61 //* Parameters : char *szFileName - The cursor resource file.                 * 
  62 //* Returns :  A handle to an icon. The handle will be NULL if an icon cannot * 
  63 //*            be created for any reason.                                     * 
  64 //* Comments : A cursor is monochrome. So, the resulting icon will also be    * 
  66 //***************************************************************************** 
  68 HICON 
CursorToIcon( wxChar 
*szFileName
, HINSTANCE hInst
, int *W
, int *H
) 
  69 { HANDLE  hDIB
;     // Handle to DIB memory 
  70   HICON   hIcon
;    // Handle to Icon 
  72   if( (hDIB 
= ReadCur( szFileName
, NULL
, W
, H
)) == (HANDLE
) NULL
) 
  75   hIcon 
= MakeIcon( hDIB
, hInst
);      // make icon from cursor DIB 
  80 //***************************************************************************** 
  81 //* Function : ReadIcon()                                                     * 
  82 //* Purpose  : Reads an icon resource file and extracts the DIB information.  * 
  83 //* Parameters : char *szFileName - The icon resource file.                   * 
  84 //* Returns :  A handle to a DIB. The handle will be NULL if the resource file* 
  85 //*            is corrupt or if memory cannot be allocated for the DIB info.  * 
  86 //***************************************************************************** 
  88 HANDLE 
ReadIcon( wxChar 
*szFileName
, int *W
, int *H
) 
  89 { ICONFILEHEADER iconFileHead
;   // ICON file header structure 
  90   ICONFILERES    iconFileRes
;    // ICON file resource 
  93                  cbBits
;         // Used for reading in file 
  94   int            hFile
;          // File handle 
  95   LPBITMAPINFO   lpDIB
;          // Pointer to DIB memory 
  97   int            nWidth      
= GetSystemMetrics( SM_CXICON
), 
  98                  nHeight     
= GetSystemMetrics( SM_CYICON
), 
 101    // Open and read the .ICO file header and the first ICONFILERES 
 102   hFile  
= _lopen( wxFNCONV(szFileName
), OF_READ
); 
 103   cbHead 
= _lread( hFile
, (LPSTR
)&iconFileHead
, sizeof(ICONFILEHEADER
)); 
 104   cbRes  
= _lread( hFile
, (LPSTR
)&iconFileRes
, sizeof(ICONFILERES
)); 
 107   if((cbHead 
!= sizeof( ICONFILEHEADER
)) || (cbRes 
!= sizeof( ICONFILERES
))) 
 108     return (HANDLE
) NULL
; 
 109   // Verify that it's an .ICON file 
 110   if( iconFileHead
.wResourceType 
!= 1) 
 111     return (HANDLE
) NULL
; 
 114   while( ((unsigned)nDirEntries 
< iconFileHead
.wResourceCount
) && 
 115          ((iconFileRes
.bWidth 
!= nWidth
) || (iconFileRes
.bHeight 
!= nHeight
))) 
 117     cbRes 
= _lread( hFile
, (LPSTR 
)&iconFileRes
, sizeof( ICONFILERES
)); 
 118     if(cbRes 
!= sizeof( ICONFILERES
)) 
 119       return (HANDLE
) NULL
; 
 125     *W 
= iconFileRes
.bWidth
; 
 127     *H 
= iconFileRes
.bHeight
; 
 129   // Allocate and lock memory to read in the DIB 
 130   hDIB 
= GlobalAlloc(GHND
, iconFileRes
.dwDIBSize
); 
 131   if(hDIB 
== (HANDLE
) NULL
) 
 132     return (HANDLE
) NULL
; 
 134 #ifdef __WINDOWS_386__ 
 135   lpDIB 
= (LPBITMAPINFO
)MK_FP32(GlobalLock(hDIB
)); 
 137   lpDIB 
= (LPBITMAPINFO
)GlobalLock(hDIB
); 
 140   // Now read the DIB portion of the file, which follows the 
 141   // end of icon resource table 
 142   _llseek( hFile
, iconFileRes
.dwDIBOffset
, 0); 
 143   cbBits 
= _lread( hFile
, (LPSTR 
)lpDIB
, (WORD 
)iconFileRes
.dwDIBSize
); 
 150   if( (DWORD 
)cbBits 
!= iconFileRes
.dwDIBSize
) 
 153     return (HANDLE
) NULL
; 
 158 //***************************************************************************** 
 159 //* Function : MakeIcon()                                                     * 
 160 //* Purpose  : Creates an icon based on the DIB info. returned by ReadIcon.   * 
 161 //* Parameters : HANDLE hDIB - A handle to the icon's DIB information.        * 
 162 //* Returns :  A handle to an Icon. NULL is returned if an icon cannot be     * 
 163 //*            successfully created.                                          * 
 164 //* Comments : The steps involved in making an icon from a DIB are very       * 
 165 //*            similar to those involved in making a cursor from a DIB.       * 
 166 //* Steps : 1) Obtain a pointer to the Icon's DIB bits.                       * 
 167 //*         2) Divide the DIB'd height with 2 to account for the fact that the* 
 168 //*            DIB stores both the XOR and the AND masks, one after the other.* 
 169 //*         3) Determine the offset to the XOR bits.                          * 
 170 //*         4) Determine the offset to the AND bits.                          * 
 171 //*         5) Create a device dependent bitmap with the XOR bits.            * 
 172 //*         6) Obtain the device dependent XOR bitmask and save in memory.    * 
 173 //*            The AND bitmask is monochrome. Monochrome bits are identical   * 
 174 //*            in both the device dependent bitmaps and device independent    * 
 175 //*            bitmaps. So, no need to convert the AND bitmask.               * 
 176 //*         7) Since a DIB is stored upside down, flip the monochrome AND bits* 
 178 //*         8) Use the XOR and AND bits and create an icon with CreateIcon.   *  
 179 //***************************************************************************** 
 181 HICON 
MakeIcon( HANDLE hDIB
, HINSTANCE hInst
) 
 183                 lpANDbits
;    // Pointer to XOR and AND bits 
 184   HBITMAP       hbmXor
;       // handle to XOR bitmap 
 185   BITMAP        bmpXor
;       // Used to manipulate XOR bitmap 
 186   DWORD         dwBmpSize
;    // Size of XOR bitmap 
 196   // 1) Obtain a pointer to the Icon's DIB bits. 
 197 #ifdef __WINDOWS_386__ 
 198   lpDIB 
= (LPBITMAPINFO 
)MK_FP32(GlobalLock( hDIB
)); 
 200   lpDIB 
= (LPBITMAPINFO 
)GlobalLock( hDIB
); 
 203   // 2) Divide the DIB'd height with 2 to account for the fact that the 
 204   //    DIB stores both the XOR and the AND masks, one after the other. 
 205   lpDIB
->bmiHeader
.biHeight 
/= 2; 
 207   // 3) Determine the offset to the XOR bits. 
 208   // To obtain this value, we have to skip the header, and color table 
 209   lpXORbits 
= (LPSTR 
)lpDIB 
+ (int )lpDIB
->bmiHeader
.biSize 
+ 
 210               (DIBNumColors( (LPSTR 
)lpDIB
) * sizeof( RGBQUAD
)); 
 212   // 4) Determine the offset to the AND bits. 
 213   //    To obtain this value, skip the XOR bits 
 214   lpANDbits 
= lpXORbits 
+ (int )(lpDIB
->bmiHeader
.biHeight 
* 
 215               (WIDTHBYTES ( lpDIB
->bmiHeader
.biWidth 
* 
 216                             lpDIB
->bmiHeader
.biBitCount
))); 
 218   // Get a hDC so we can create a bitmap compatible with it 
 219   hDC 
= CreateDC( wxT("DISPLAY"), NULL
, NULL
, NULL
); 
 221   // 5) Create a device dependent bitmap with the XOR bits. 
 222   hbmXor 
= CreateDIBitmap( hDC
, (LPBITMAPINFOHEADER
)&(lpDIB
->bmiHeader
), 
 223                            CBM_INIT
, lpXORbits
, lpDIB
, DIB_RGB_COLORS
); 
 225   GetObject( hbmXor
, sizeof(BITMAP
), (LPSTR
)&bmpXor
); 
 227   dwBmpSize 
= (DWORD 
)(bmpXor
.bmWidthBytes 
* bmpXor
.bmHeight 
* bmpXor
.bmPlanes
); 
 228   hXorDDB 
= GlobalAlloc( GHND
, dwBmpSize
); 
 229   if(hXorDDB 
== (HANDLE
) NULL
) 
 231     // clean up before quitting 
 232     DeleteObject( hbmXor
); 
 238 #ifdef __WINDOWS_386__ 
 239   lpXorDDB 
= (LPSTR
)MK_FP32(GlobalLock( hXorDDB
)); 
 241   lpXorDDB 
= (LPSTR
)GlobalLock( hXorDDB
); 
 244   // 6) Obtain the device dependent XOR bitmask and save in memory. 
 245   // The AND bitmask is monochrome. Monochrome bits are identical 
 246   // in both the device dependent bitmaps and device independent 
 247   // bitmaps. So, no need to convert the AND bitmask. 
 248   GetBitmapBits( hbmXor
, dwBmpSize
, lpXorDDB
); 
 250   // 7) Since a DIB is stored upside down, flip the monochrome AND bits by scanlines. 
 251   k 
= (int )lpDIB
->bmiHeader
.biHeight
; 
 252   for( j 
= 0 ; j 
< k 
; j
++, lpANDbits 
+= sizeof(DWORD
)) 
 253     szFlip
[(k 
- 1) - j
] = *(DWORD FAR 
*)lpANDbits
; 
 255   // 8) Use the XOR and AND bits and create an icon with CreateIcon. 
 256   hIcon 
= CreateIcon( hInst
, bmpXor
.bmWidth
, bmpXor
.bmHeight
, bmpXor
.bmPlanes
, 
 257                       bmpXor
.bmBitsPixel
, (const BYTE 
*)szFlip
, (const BYTE 
*)lpXorDDB
); 
 259   // Clean up before exiting. 
 260   DeleteObject( hbmXor
); 
 261   GlobalUnlock( hXorDDB
); 
 262   GlobalFree( hXorDDB
); 
 269 // ************************************************************************** 
 271 //***************************************************************************** 
 272 //* Function : ReadCursorFile()                                               * 
 273 //* Purpose  : Reads a cursor resource file and creates a cursor based on that* 
 275 //* Parameters : char *szFileName - The cursor resource file.                 * 
 276 //* Returns :  A handle to a cursor. The handle will be NULL if a cursor can't* 
 277 //*            be created for any reason.                                     * 
 278 //***************************************************************************** 
 280 HCURSOR 
ReadCursorFile( wxChar 
*szFileName
, HINSTANCE hInst
, int *W
, int *H
, 
 281                         int *XHot
, int *YHot
) 
 282 { HANDLE    hDIB
;    // Handle to DIB memory 
 286   // read cur DIB from file 
 287   if( (hDIB 
= ReadCur( szFileName
, (LPPOINT 
)&ptHotSpot
, W
, H
)) == (HANDLE
) NULL
) 
 288     return (HCURSOR
) NULL
; 
 289   hCursor 
= MakeCursor( hDIB
, (LPPOINT 
)&ptHotSpot
, hInst
);//create cur from DIB 
 298 //***************************************************************************** 
 299 //* Function : IconToCursor()                                                 * 
 300 //* Purpose  : Reads an icon resource file and creates a cursor based on that * 
 302 //* Parameters : char *szFileName - The icon resource file.                   * 
 303 //* Returns :  A handle to a cursor. The handle will be NULL if a cursor can't* 
 304 //*            be created for any reason.                                     * 
 305 //* Comments : An icon may be in color. So, the DIB has to be forced to be    * 
 307 //***************************************************************************** 
 309 HCURSOR 
IconToCursor( wxChar 
*szFileName
, HINSTANCE hInst
, int XHot
, int YHot
, 
 315   if( (hDIB 
= ReadIcon( szFileName
, W
, H
)) == (HANDLE
) NULL
) 
 316                                              //read icon file to get icon DIB 
 317     return (HCURSOR
) NULL
; 
 318   // Set the hot spot of the cursor 
 321   hCursor     
= MakeCursor( hDIB
, (LPPOINT 
)&ptHotSpot
, hInst
); 
 322                               //create cursor from DIB 
 327 //***************************************************************************** 
 328 //* Function : ReadCur()                                                      * 
 329 //* Purpose  : Reads a cursor resource file and extracts the DIB information. * 
 330 //* Parameters : LPSTR szFileName - The cursor resource file.                 * 
 331 //* Returns :  A handle to a DIB. The handle will be NULL if the resource file* 
 332 //*            is corrupt or if memory cannot be allocated for the DIB info.  * 
 333 //***************************************************************************** 
 335 HANDLE 
ReadCur( wxChar 
*szFileName
, LPPOINT lpptHotSpot
, int *W
, int *H
) 
 336 { CURFILEHEADER   curFileHead
;  // CURSOR file header structure 
 337   CURFILERES      curFileRes
;   // CURSOR file resource 
 340                   cbBits
;       // Used for reading in file 
 341   LPBITMAPINFO    lpDIB
;        // Pointer to DIB memory 
 342   int             hFile
;        // Handle to File 
 344   int             nWidth      
= GetSystemMetrics( SM_CXCURSOR
), 
 345                   nHeight     
= GetSystemMetrics( SM_CYCURSOR
), 
 348   // Open and read the .ICO file header and the first ICONFILERES 
 349   hFile  
= _lopen( wxFNCONV(szFileName
), OF_READ
); 
 350   cbHead 
= _lread( hFile
,  (LPSTR 
)&curFileHead
, sizeof( CURFILEHEADER
)); 
 351   cbRes  
= _lread( hFile
,  (LPSTR 
)&curFileRes
,  sizeof( CURFILERES
)); 
 354   if((cbHead 
!= sizeof( CURFILEHEADER
)) || (cbRes 
!= sizeof( CURFILERES
))) 
 355     return (HANDLE
) NULL
; 
 357    // Verify that it's an .CUR file 
 358   if ((curFileRes
.bReserved1 
!= 0) || (curFileHead
.wResourceType 
!= 2)) 
 359     return (HANDLE
) NULL
; 
 361   // following added by P.S. 
 362   while( ((unsigned)nDirEntries 
< curFileHead
.wResourceCount
) && 
 363          ((curFileRes
.bWidth 
!= nWidth
) || (curFileRes
.bHeight 
!= nHeight
))) 
 365     cbRes 
= _lread( hFile
, (LPSTR 
)&curFileRes
, sizeof( CURFILERES
)); 
 366     if(cbRes 
!= sizeof( CURFILERES
)) 
 367       return (HANDLE
) NULL
; 
 372     *W 
= curFileRes
.bWidth
; 
 374     *H 
= curFileRes
.bHeight
; 
 377   // Allocate & lock memory to read in the DIB 
 378   hDIB 
= GlobalAlloc(GHND
, curFileRes
.dwDIBSize
); 
 379   if(hDIB 
== (HANDLE
) NULL
) 
 380     return (HANDLE
) NULL
; 
 382 #ifdef __WINDOWS_386__ 
 383   lpDIB 
= (LPBITMAPINFO 
)MK_FP32(GlobalLock(hDIB
)); 
 385   lpDIB 
= (LPBITMAPINFO 
)GlobalLock(hDIB
); 
 388   // Now read the DIB portion of the file, which follows the 
 389   // end of icon resource table 
 390   _llseek( hFile
, curFileRes
.dwDIBOffset
, 0); 
 391   cbBits 
= _lread( hFile
, (LPSTR 
)lpDIB
, (WORD 
)curFileRes
.dwDIBSize
); 
 396   if((DWORD
)cbBits 
!= curFileRes
.dwDIBSize
) 
 400     return (HANDLE
) NULL
; 
 402   if(lpptHotSpot 
!= (LPPOINT
) NULL
)  // If it is necessary to know the hot spot 
 404     lpptHotSpot
->x 
= (int )curFileRes
.wXHotspot
; 
 405     lpptHotSpot
->y 
= (int )curFileRes
.wYHotspot
; 
 411 //***************************************************************************** 
 412 //* Function : ColorDDBToMonoDDB()                                            * 
 413 //* Purpose  : Converts a color bitmap to a monochrome bitmap.                * 
 414 //* Parameters : HBITMAP hbm - The color bitmap.                              * 
 415 //* Returns :  A handle to a monochrome bitmap.                               * 
 416 //***************************************************************************** 
 417 HBITMAP 
ColorDDBToMonoDDB ( HBITMAP hbm
) 
 420   LPBITMAPINFOHEADER  lpbi
; 
 427   GetObject( hbm
, sizeof( bm
), (LPSTR 
)&bm
); 
 429   bi
.biSize           
= sizeof( BITMAPINFOHEADER
);    // size of this structure 
 430   bi
.biWidth          
= bm
.bmWidth
;                   // bitmap width in pixels 
 431   bi
.biHeight         
= bm
.bmHeight
;                  // bitmap height in pixels 
 432   bi
.biPlanes         
= 1;                            // # of planes always 1 for DIBs 
 433   bi
.biBitCount       
= bm
.bmPlanes 
* bm
.bmBitsPixel
; // color bits per pixel 
 434   bi
.biCompression    
= BI_RGB
;                       // no compression 
 435   bi
.biSizeImage      
= 0;                            // 0 means default size 
 436   bi
.biXPelsPerMeter  
= 0;                            // not used 
 437   bi
.biYPelsPerMeter  
= 0;                            // not used 
 438   bi
.biClrUsed        
= 0;                            // 0 means default colors 
 439   bi
.biClrImportant   
= 0;                            // 0 means defaults 
 441   dwLen 
= bi
.biSize 
+ PaletteSize((LPSTR
)&bi
); 
 443   hdc 
= GetDC( (HWND
) NULL
); 
 445   hdib 
= GlobalAlloc( GHND
, dwLen
); 
 446   if (hdib 
== (HANDLE
) NULL
) 
 448     ReleaseDC( (HWND
) NULL
, hdc
); 
 449     return (HBITMAP
) NULL
; 
 452 #ifdef __WINDOWS_386__ 
 453   lpbi 
= (LPBITMAPINFOHEADER 
)MK_FP32(GlobalLock( hdib
)); 
 455   lpbi 
= (LPBITMAPINFOHEADER 
)GlobalLock( hdib
); 
 460   // Call GetDIBits with a NULL lpBits parameter; it will calculate 
 461   // the biSizeImage field. 
 462   GetDIBits( hdc
, hbm
, 0, (WORD
)bi
.biHeight
, 
 463              NULL
, (LPBITMAPINFO
)lpbi
, DIB_RGB_COLORS
); 
 468   // If the driver did not fill in the biSizeImage field, make one up. 
 469   if(bi
.biSizeImage 
== 0) 
 470     bi
.biSizeImage 
= WIDTHBYTES( (DWORD 
)bm
.bmWidth 
* bi
.biBitCount
) * bm
.bmHeight
; 
 472   // Reallocate the buffer big enough to hold all the bits. 
 473   dwLen 
= bi
.biSize 
+ PaletteSize((LPSTR
)&bi
) + bi
.biSizeImage
; 
 474   if( (h 
= GlobalReAlloc( hdib
, dwLen
, 0)) != 0) 
 479     ReleaseDC( (HWND
) NULL
, hdc
); 
 480     return (HBITMAP
) NULL
; 
 483   // Call GetDIBits with a NON-NULL lpBits parameter, to actually 
 484   // get the bits this time. 
 486 #ifdef __WINDOWS_386__ 
 487   lpbi 
= (LPBITMAPINFOHEADER 
)MK_FP32(GlobalLock( hdib
)); 
 489   lpbi 
= (LPBITMAPINFOHEADER 
)GlobalLock( hdib
); 
 492   if( GetDIBits( hdc
, hbm
, 0, (WORD
)bi
.biHeight
, 
 493                  (LPSTR
)lpbi 
+ (WORD
)lpbi
->biSize 
+ PaletteSize((LPSTR
)lpbi
), 
 494                  (LPBITMAPINFO
)lpbi
, DIB_RGB_COLORS
) == 0) 
 497     hdib 
= (HANDLE
) NULL
; 
 498     ReleaseDC( (HWND
) NULL
, hdc
); 
 499     return (HBITMAP
) NULL
; 
 502   // Finally, create a monochrome DDB, and put the DIB into 
 503   // it. SetDIBits does smart color conversion. 
 504   hbmMono 
= CreateBitmap((WORD
)lpbi
->biWidth
, (WORD
)lpbi
->biHeight
, 1, 1, NULL
); 
 505   SetDIBits( hdc
, hbmMono
, (WORD
)0, (WORD
)lpbi
->biHeight
, 
 506              (LPSTR
)lpbi 
+ (int )lpbi
->biSize 
+ PaletteSize((LPSTR
)lpbi
), 
 507              (LPBITMAPINFO
)lpbi
, DIB_RGB_COLORS
); 
 513   ReleaseDC((HWND
) NULL
, hdc
); 
 517 //***************************************************************************** 
 518 //* Function : MakeCursor()                                                   * 
 519 //* Purpose  : Creates a cursor based on the DIB info. returned by ReadCursor.* 
 520 //* Parameters : HANDLE hDIB - A handle to the cursor's DIB information.      * 
 521 //*              LPPOINT lppt - A pointer to a point struct. indicating the   * 
 522 //*                             location of the Cursor's hot spot.            * 
 523 //* Returns :  A handle to a cursor. NULL is returned if a cursor cannot be   * 
 524 //*            successfully created.                                          * 
 525 //* Comments : The steps involved in making a cursor from a DIB are very      * 
 526 //*            similar to those involved in making an icon from a DIB.        * 
 527 //* Steps : 1) Obtain a pointer to the Cursor's DIB bits.                     * 
 528 //*         2) Divide the DIB's height with 2 to account for the fact that the* 
 529 //*            DIB stores both the XOR and the AND masks, one after the other.* 
 530 //*         3) Determine the offset to the XOR bits.                          * 
 531 //*         4) Determine the offset to the AND bits.                          * 
 532 //*         5) Create a device dependent bitmap with the XOR bits.            * 
 533 //*         6) Obtain the device dependent XOR bitmask and save in memory.    * 
 534 //*            The AND bitmask is monochrome. Monochrome bits are identical   * 
 535 //*            in both the device dependent bitmaps and device independent    * 
 536 //*            bitmaps. So, no need to convert the AND bitmask.               * 
 537 //*         7) Since a DIB is stored upside down, flip the monochrome AND bits* 
 539 //*         8) Use the XOR and AND bits and create a cursor with CreateCursor.* 
 540 //***************************************************************************** 
 542 HCURSOR 
MakeCursor( HANDLE hDIB
, LPPOINT lpptHotSpot
, HINSTANCE hInst
) 
 544                 lpANDbits
;    // Pointer to XOR and AND bits 
 545   HBITMAP       hbmXor
;       // handle to XOR bitmap 
 546   BITMAP        bmpXor
;       // Used to manipulate XOR bitmap 
 547   DWORD         dwBmpSize
;    // Size of XOR bitmap 
 557   // 1) Obtain a pointer to the Cursor's DIB bits. 
 558 #ifdef __WINDOWS_386__ 
 559   lpDIB 
= (LPBITMAPINFO 
)MK_FP32(GlobalLock( hDIB
)); 
 561   lpDIB 
= (LPBITMAPINFO 
)GlobalLock( hDIB
); 
 564   // 2) Divide the DIB's height with 2 to account for the fact that the 
 565   //    DIB stores both the XOR and the AND masks, one after the other. 
 566   lpDIB
->bmiHeader
.biHeight 
/= 2; 
 568   // 3) Determine the offset to the XOR bits. 
 569   //    To obtain this value, we have to skip the header, and color table 
 570   lpXORbits 
= (LPSTR 
)lpDIB 
+ (int )lpDIB
->bmiHeader
.biSize 
+ 
 571               (DIBNumColors((LPSTR
)lpDIB
) * sizeof(RGBQUAD
)); 
 573   // 4) Determine the offset to the AND bits 
 574   //    To obtain this value, skip the XOR bits 
 575   lpANDbits 
= lpXORbits 
+ (int )( lpDIB
->bmiHeader
.biHeight 
* 
 576               (WIDTHBYTES( lpDIB
->bmiHeader
.biWidth 
* 
 577                            lpDIB
->bmiHeader
.biBitCount
))); 
 579   // Get a hDC so we can create a bitmap compatible with it 
 580   hDC 
= CreateDC( wxT("DISPLAY"), NULL
, NULL
, NULL
); 
 582   // 5) Create a device dependent bitmap with the XOR bits. 
 583   hbmXor 
= CreateBitmap( (int )lpDIB
->bmiHeader
.biWidth
, 
 584                          (int )lpDIB
->bmiHeader
.biHeight
, 1, 1, NULL
); 
 585   SetDIBits( hDC
, hbmXor
, 0, (WORD
)lpDIB
->bmiHeader
.biHeight
, lpXORbits
, 
 586              lpDIB
, DIB_RGB_COLORS
); 
 587   GetObject( hbmXor
, sizeof( BITMAP
), (LPSTR 
)&bmpXor
); 
 589   dwBmpSize 
= (DWORD 
)(bmpXor
.bmWidthBytes 
* bmpXor
.bmHeight 
* bmpXor
.bmPlanes
); 
 590   hXorDDB 
= GlobalAlloc( GHND
, dwBmpSize
); 
 591   if(hXorDDB 
== (HANDLE
) NULL
) 
 592   {  // clean up before quitting 
 593     DeleteObject( hbmXor
); 
 596     return (HCURSOR
) NULL
; 
 598 #ifdef __WINDOWS_386__ 
 599   lpXorDDB 
= (LPSTR
)MK_FP32(GlobalLock( hXorDDB
)); 
 601   lpXorDDB 
= (LPSTR
)GlobalLock( hXorDDB
); 
 604   // 6) Obtain the device dependent XOR bitmask and save in memory. 
 605   //    The AND bitmask is monochrome. Monochrome bits are identical 
 606   //    in both the device dependent bitmaps and device independent 
 607   //    bitmaps. So, no need to convert the AND bitmask. 
 608   GetBitmapBits( hbmXor
, dwBmpSize
, lpXorDDB
); 
 610   // 7) Since a DIB is stored upside down, flip the monochrome AND bits by scanlines. 
 611   k 
= (int)lpDIB
->bmiHeader
.biHeight
; 
 612   for( j 
= 0 ; j 
< k
; j
++, lpANDbits 
+= sizeof( DWORD
)) 
 613     szFlip
[(k 
- 1) - j
] = *(DWORD FAR 
*)lpANDbits
; 
 615   // 8) Use the XOR and AND bits and create a cursor with CreateCursor. 
 616   hCursor 
= CreateCursor( hInst
, lpptHotSpot
->x
, lpptHotSpot
->y
, 
 617                           bmpXor
.bmWidth
, bmpXor
.bmHeight
, (LPSTR
)szFlip
, lpXorDDB
); 
 619   // Clean up before exiting. 
 620   DeleteObject( hbmXor
); 
 621   GlobalUnlock( hXorDDB
); 
 622   GlobalFree( hXorDDB
); 
 629 //***************************************************************************** 
 630 //* Function : PaletteSize()                                                  * 
 631 //* Purpose  : Calculates the palette size in bytes. If the info. block is of * 
 632 //*            the BITMAPCOREHEADER type, the number of colors is multiplied  * 
 633 //*            by sizeof(RGBTRIPLE) to give the palette size, otherwise the   * 
 634 //*            number of colors is multiplied by sizeof(RGBQUAD).             * 
 635 //* Parameters : LPSTR pv - pointer to the BITMAPINFOHEADER                   * 
 636 //* Returns :  The size of the palette.                                       * 
 637 //***************************************************************************** 
 639 WORD 
PaletteSize( LPSTR pv
) 
 640 { LPBITMAPINFOHEADER  lpbi
; 
 643   lpbi      
= (LPBITMAPINFOHEADER 
)pv
; 
 644   NumColors 
= DIBNumColors((LPSTR 
)lpbi
); 
 646   if(lpbi
->biSize 
== sizeof( BITMAPCOREHEADER
))  // OS/2 style DIBs 
 647     return NumColors 
* sizeof( RGBTRIPLE
); 
 649     return NumColors 
* sizeof( RGBQUAD
); 
 652 //***************************************************************************** 
 653 //* Function : DIBNumColors()                                                 * 
 654 //* Purpose  : This function calculates the number of colors in the DIB's     * 
 655 //*            color table by finding the bits per pixel for the DIB (whether * 
 656 //*            Win3.0 or OS/2-style DIB). If bits per pixel is 1: colors=2,   * 
 657 //*            if 4: colors=16, if 8: colors=256, if 24, no colors in color   * 
 659 //* Parameters : LPSTR lpbi - pointer to packed-DIB memory block.             * 
 660 //* Returns :  The number of colors in the color table.                       * 
 661 //***************************************************************************** 
 663 WORD 
DIBNumColors ( LPSTR pv
) 
 665   BITMAPINFOHEADER  
*lpbi
; 
 666   BITMAPCOREHEADER  
*lpbc
; 
 668   lpbi 
= ((BITMAPINFOHEADER
* )pv
);     // assume win 3.0 style DIBs 
 669   lpbc 
= ((BITMAPCOREHEADER
* )pv
);     // assume OS/2 style DIBs 
 671   // With the BITMAPINFO format headers, the size of the palette 
 672   // is in biClrUsed, whereas in the BITMAPCORE - style headers, it 
 673   // is dependent on the bits per pixel ( = 2 raised to the power of 
 676   if(lpbi
->biSize 
!= sizeof( BITMAPCOREHEADER
)) 
 678     if(lpbi
->biClrUsed 
!= 0) 
 679       return (WORD
)lpbi
->biClrUsed
; 
 680     bits 
= lpbi
->biBitCount
; 
 683     bits 
= lpbc
->bcBitCount
; 
 694       // A 24 bitcount DIB has no color table 
 700 // ****************************************************************** 
 701 BOOL 
fGetXPixmap( BOOL fIsIcon
, wxChar 
*szFileName
, HINSTANCE hInst
, 
 702                   char cData
[], int &width
, int &height
) 
 715   HCURSOR   hIconOrCursor 
= fIsIcon 
? 
 716                       IconToCursor( szFileName
, hInst
, 0, 0, &w
, &h
) 
 717                     : ReadCursorFile( szFileName
, hInst
, &w
, &h
, 0, 0); 
 720   if(hIconOrCursor 
== 0) 
 723   hdc       
= GetDC( GetDesktopWindow()); 
 724   hdcMemory 
= CreateCompatibleDC( hdc
); 
 725   hbmp      
= CreateCompatibleBitmap( hdc
, w
, h
); 
 726   holdbmp   
= SelectObject( hdcMemory
, hbmp
); 
 727   PatBlt( hdcMemory
, 0, 0, w
, h
, BLACKNESS
); // or use WHITENESS?? 
 728   DrawIcon( hdcMemory
, 0, 0, hIconOrCursor
); //using HCURSOR with DrawIcon is OK 
 730   // the data retrieval follows: 
 733   for( j 
= 0, s 
= (BYTE 
*)cData 
; j 
< h 
; ++j
) 
 734     for( i 
= 0 ; i 
< w 
; ++i
, cMask 
>>= 1) 
 741       rgb  
= GetPixel( hdcMemory
, i
, j
); 
 742       sum  
= (int )(rgb 
& 0xFFL
); 
 743       sum 
+= (int )((rgb 
& 0xFF00L
) >> 8); 
 744       sum 
+= (int )((rgb 
& 0xFF0000L
) >> 16); 
 746         cByte 
= cByte 
| cMask
; 
 753   SelectObject( hdcMemory
, holdbmp
); 
 754   DeleteDC( hdcMemory
); 
 755   ReleaseDC( GetDesktopWindow(), hdc
); 
 756   DestroyCursor( hIconOrCursor
); 
 762 // Added from scavenged internet code, JACS 23/6/95 
 763 HCURSOR 
MakeCursorFromBitmap(HINSTANCE hInst
, HBITMAP hBitmap
, POINT 
*pPoint
) 
 765   HDC hDCColor
, hDCMono
; 
 776   hDC 
= GetDC((HWND
) NULL
); 
 777   hDCColor 
= CreateCompatibleDC(hDC
); 
 778   hDCMono 
= CreateCompatibleDC(hDC
); 
 779   hAndBmp 
= CreateCompatibleBitmap(hDCMono
, 32, 32); 
 780   hXorBmp 
= CreateCompatibleBitmap(hDCMono
, 32, 32); 
 782   hBmpOld 
= (HBITMAP
) SelectObject(hDCColor
, hBitmap
); 
 783   SelectObject(hDCMono
, hAndBmp
); 
 784   SetBkColor(hDCColor
, RGB(191, 191, 191)); 
 786   BitBlt(hDCMono
, 0, 0, 32, 32, hDCColor
, 0, 0, SRCCOPY
); 
 788   // Now we have the AND Mask 
 790   GetObject(hAndBmp
, sizeof(BITMAP
), (LPSTR
) &bm
); 
 791   dwBytes 
= (bm
.bmWidthBytes 
* bm
.bmHeight
); 
 792   andBits 
= (NPSTR
) LocalAlloc(LPTR
, dwBytes
); 
 793   GetBitmapBits(hAndBmp
, dwBytes
, andBits
); 
 795   SelectObject(hDCMono
, hXorBmp
); 
 796   SetBkColor(hDCColor
, RGB(0, 0, 0)); 
 798   BitBlt(hDCMono
, 0, 0, 32, 32, hDCColor
, 0, 0, SRCCOPY
); 
 800   // Now we have the XOR Mask 
 802   GetObject(hXorBmp
, sizeof(BITMAP
), (LPSTR
) &bm
); 
 803   dwBytes 
= (bm
.bmWidthBytes 
* bm
.bmHeight
); 
 804   xorBits 
= (NPSTR
) LocalAlloc(LPTR
, dwBytes
); 
 805   GetBitmapBits(hXorBmp
, dwBytes
, xorBits
); 
 812   hNewCursor 
= CreateCursor(hInst
, 
 813                             pPoint
->x
, pPoint
->y
, 32, 32, andBits
, xorBits
); 
 815   SelectObject(hDCColor
, hBmpOld
); 
 816   SelectObject(hDCMono
, hBmpOld
); 
 819   DeleteObject(hAndBmp
); 
 820   DeleteObject(hXorBmp
); 
 821   ReleaseDC(NULL
, hDC
); 
 824   LocalUnlock(LocalHandle((void NEAR
*) andBits
)); 
 825   LocalUnlock(LocalHandle((void NEAR
*) xorBits
)); 
 826   LocalFree(LocalHandle((void NEAR
*) andBits
)); 
 827   LocalFree(LocalHandle((void NEAR
*) xorBits
)); 
 829   LocalUnlock(LocalHandle((WORD
) andBits
)); 
 830   LocalUnlock(LocalHandle((WORD
) xorBits
)); 
 831   LocalFree(LocalHandle((WORD
) andBits
)); 
 832   LocalFree(LocalHandle((WORD
) xorBits
)); 
 835   LocalUnlock(LocalHandle((LPCVOID
) andBits
)); 
 836   LocalUnlock(LocalHandle((LPCVOID
) xorBits
)); 
 837   LocalFree(LocalHandle((LPCVOID
) andBits
)); 
 838   LocalFree(LocalHandle((LPCVOID
) xorBits
)); 
 844  * This doesn't work: just gives us a grey square. Ideas, anyone? 
 847 HICON 
MakeIconFromBitmap(HINSTANCE hInst
, HBITMAP hBitmap
) 
 849   HDC hDCColor
, hDCMono
; 
 860   hDC 
= GetDC((HWND
) NULL
); 
 861   hDCColor 
= CreateCompatibleDC(hDC
); 
 862   hDCMono 
= CreateCompatibleDC(hDC
); 
 863   hAndBmp 
= CreateCompatibleBitmap(hDCMono
, 32, 32); 
 864   hXorBmp 
= CreateCompatibleBitmap(hDCMono
, 32, 32); 
 866   hBmpOld 
= (HBITMAP
) SelectObject(hDCColor
, hBitmap
); 
 867   SelectObject(hDCMono
, hAndBmp
); 
 868   SetBkColor(hDCColor
, RGB(191, 191, 191)); 
 870   BitBlt(hDCMono
, 0, 0, 32, 32, hDCColor
, 0, 0, SRCCOPY
); 
 872   // Now we have the AND Mask 
 874   GetObject(hAndBmp
, sizeof(BITMAP
), (LPSTR
) &bm
); 
 875   dwBytes 
= (bm
.bmWidthBytes 
* bm
.bmHeight
); 
 876   andBits 
= (NPSTR
) LocalAlloc(LPTR
, dwBytes
); 
 877   GetBitmapBits(hAndBmp
, dwBytes
, andBits
); 
 879   SelectObject(hDCMono
, hXorBmp
); 
 880   SetBkColor(hDCColor
, RGB(0, 0, 0)); 
 882   BitBlt(hDCMono
, 0, 0, 32, 32, hDCColor
, 0, 0, SRCCOPY
); 
 884   // Now we have the XOR Mask 
 886   GetObject(hXorBmp
, sizeof(BITMAP
), (LPSTR
) &bm
); 
 887   dwBytes 
= (bm
.bmWidthBytes 
* bm
.bmHeight
); 
 888   xorBits 
= (NPSTR
) LocalAlloc(LPTR
, dwBytes
); 
 889   GetBitmapBits(hXorBmp
, dwBytes
, xorBits
); 
 891   hNewIcon 
= CreateIcon(hInst
, 1, 4, 32, 32, (unsigned char *)andBits
, (unsigned char *)xorBits
); 
 893   SelectObject(hDCColor
, hBmpOld
); 
 894   SelectObject(hDCMono
, hBmpOld
); 
 897   DeleteObject(hAndBmp
); 
 898   DeleteObject(hXorBmp
); 
 899   ReleaseDC((HWND
) NULL
, hDC
); 
 902   LocalUnlock(LocalHandle((void NEAR
*) andBits
)); 
 903   LocalUnlock(LocalHandle((void NEAR
*) xorBits
)); 
 904   LocalFree(LocalHandle((void NEAR
*) andBits
)); 
 905   LocalFree(LocalHandle((void NEAR
*) xorBits
)); 
 907   LocalUnlock(LocalHandle((WORD
) andBits
)); 
 908   LocalUnlock(LocalHandle((WORD
) xorBits
)); 
 909   LocalFree(LocalHandle((WORD
) andBits
)); 
 910   LocalFree(LocalHandle((WORD
) xorBits
)); 
 913   LocalUnlock(LocalHandle((LPCVOID
) andBits
)); 
 914   LocalUnlock(LocalHandle((LPCVOID
) xorBits
)); 
 915   LocalFree(LocalHandle((LPCVOID
) andBits
)); 
 916   LocalFree(LocalHandle((LPCVOID
) xorBits
));