]> git.saurik.com Git - wxWidgets.git/blob - src/msw/dib.cpp
update from Janos Vegh
[wxWidgets.git] / src / msw / dib.cpp
1 ///////////////////////////////////////////////////////////////////////////////
2 // Name: src/msw/dib.cpp
3 // Purpose: implements wxDIB class
4 // Author: Vadim Zeitlin
5 // Modified by:
6 // Created: 03.03.03 (replaces the old file with the same name)
7 // RCS-ID: $Id$
8 // Copyright: (c) 2003 Vadim Zeitlin <vadim@wxwindows.org>
9 // License: wxWindows licence
10 ///////////////////////////////////////////////////////////////////////////////
11
12 // ============================================================================
13 // declarations
14 // ============================================================================
15
16 // ----------------------------------------------------------------------------
17 // headers
18 // ----------------------------------------------------------------------------
19
20 // For compilers that support precompilation, includes "wx.h".
21 #include "wx/wxprec.h"
22
23 #ifdef __BORLANDC__
24 #pragma hdrstop
25 #endif
26
27 #ifndef WX_PRECOMP
28 #include "wx/string.h"
29 #include "wx/log.h"
30 #endif //WX_PRECOMP
31
32 #include "wx/bitmap.h"
33 #include "wx/intl.h"
34
35 #include <stdio.h>
36 #include <stdlib.h>
37
38 #if !defined(__MWERKS__) && !defined(__SALFORDC__)
39 #include <memory.h>
40 #endif
41
42 #ifdef __GNUWIN32_OLD__
43 #include "wx/msw/gnuwin32/extra.h"
44 #endif
45
46 #include "wx/image.h"
47 #include "wx/msw/dib.h"
48
49 // ============================================================================
50 // implementation
51 // ============================================================================
52
53 // ----------------------------------------------------------------------------
54 // wxDIB creation
55 // ----------------------------------------------------------------------------
56
57 bool wxDIB::Create(int width, int height, int depth)
58 {
59 // we don't handle the palette yet
60 wxASSERT_MSG( depth == 24 || depth == 32,
61 _T("unsupported image depth in wxDIB::Create()") );
62
63 static const int infosize = sizeof(BITMAPINFOHEADER);
64
65 BITMAPINFO *info = (BITMAPINFO *)malloc(infosize);
66 wxCHECK_MSG( info, NULL, _T("malloc(BITMAPINFO) failed") );
67
68 memset(info, 0, infosize);
69
70 info->bmiHeader.biSize = infosize;
71 info->bmiHeader.biWidth = width;
72
73 // we use positive height here which corresponds to a DIB with normal, i.e.
74 // bottom to top, order -- normally using negative height (which means
75 // reversed for MS and hence natural for all the normal people top to
76 // bottom line scan order) could be used to avoid the need for the image
77 // reversal in Create(image) but this doesn't work under NT, only Win9x!
78 info->bmiHeader.biHeight = height;
79
80 info->bmiHeader.biPlanes = 1;
81 info->bmiHeader.biBitCount = depth;
82 info->bmiHeader.biCompression = BI_RGB;
83 info->bmiHeader.biSizeImage = GetLineSize(width, depth)*height;
84
85 // No need to report an error here. If it fails, we just won't use a
86 // file mapping and CreateDIBSection will just allocate memory for us.
87 m_handle = ::CreateDIBSection
88 (
89 0, // hdc (unused with DIB_RGB_COLORS)
90 info, // bitmap description
91 DIB_RGB_COLORS, // use RGB, not palette
92 &m_data, // [out] DIB bits
93 NULL, // don't use file mapping
94 0 // file mapping offset (not used here)
95 );
96
97 free(info);
98
99 if ( !m_handle )
100 {
101 wxLogLastError(wxT("CreateDIBSection"));
102
103 return false;
104 }
105
106 m_width = width;
107 m_height = height;
108 m_depth = depth;
109
110 return true;
111 }
112
113 // ----------------------------------------------------------------------------
114 // wxDIB accessors
115 // ----------------------------------------------------------------------------
116
117 void wxDIB::DoGetObject() const
118 {
119 // only do something if we have a valid DIB but we don't [yet] have valid
120 // data
121 if ( m_handle && !m_data )
122 {
123 // although all the info we need is in BITMAP and so we don't really
124 // need DIBSECTION we still ask for it as modifying the bit values only
125 // works for the real DIBs and not for the bitmaps and it's better to
126 // check for this now rather than trying to find out why it doesn't
127 // work later
128 DIBSECTION ds;
129 if ( !::GetObject(m_handle, sizeof(ds), &ds) )
130 {
131 wxLogLastError(_T("GetObject(hDIB)"));
132
133 return;
134 }
135
136 wxDIB *self = wxConstCast(this, wxDIB);
137
138 self->m_width = ds.dsBm.bmWidth;
139 self->m_height = ds.dsBm.bmHeight;
140 self->m_depth = ds.dsBm.bmBitsPixel;
141 self->m_data = ds.dsBm.bmBits;
142 }
143 }
144
145 // ----------------------------------------------------------------------------
146 // wxImage support
147 // ----------------------------------------------------------------------------
148
149 #if wxUSE_IMAGE
150
151 bool wxDIB::Create(const wxImage& image)
152 {
153 wxCHECK_MSG( image.Ok(), false, _T("invalid wxImage in wxDIB ctor") );
154
155 const int h = image.GetHeight();
156 const int w = image.GetWidth();
157
158 // if we have alpha channel, we need to create a 32bpp RGBA DIB, otherwise
159 // a 24bpp RGB is sufficient
160 const bool hasAlpha = image.HasAlpha();
161 const int bpp = hasAlpha ? 32 : 24;
162
163 if ( !Create(w, h, bpp) )
164 return false;
165
166 // DIBs are stored in bottom to top order (see also the comment above in
167 // Create()) so we need to copy bits line by line and starting from the end
168 const int srcBytesPerLine = w * 3;
169 const int dstBytesPerLine = GetLineSize(w, bpp);
170 const unsigned char *src = image.GetData() + ((h - 1) * srcBytesPerLine);
171 const unsigned char *alpha = hasAlpha ? image.GetAlpha() + (h - 1)*w : NULL;
172 unsigned char *dstLineStart = (unsigned char *)m_data;
173 for ( int y = 0; y < h; y++ )
174 {
175 // copy one DIB line
176 unsigned char *dst = dstLineStart;
177 for ( int x = 0; x < w; x++ )
178 {
179 // also, the order of RGB is inversed for DIBs
180 *dst++ = src[2];
181 *dst++ = src[1];
182 *dst++ = src[0];
183
184 src += 3;
185
186 if ( alpha )
187 *dst++ = *alpha++;
188 }
189
190 // pass to the previous line in the image
191 src -= 2*srcBytesPerLine;
192 if ( alpha )
193 alpha -= 2*w;
194
195 // and to the next one in the DIB
196 dstLineStart += dstBytesPerLine;
197 }
198
199 return true;
200 }
201
202 #endif // wxUSE_IMAGE
203
204 // ============================================================================
205 // old DIB code, to be integrated in wxDIB class
206 // ============================================================================
207
208 /*
209 * Routines for dealing with Device Independent Bitmaps.
210 *
211 * wxReadDIB() - Reads a DIB
212 * wxWriteDIB() - Writes a global handle in CF_DIB format
213 * to a file.
214 * wxPaletteSize() - Calculates the palette size in bytes
215 * of given DIB
216 * wxDibNumColors() - Determines the number of colors in DIB
217 * wxDibFromBitmap() - Creates a DIB repr. the DDB passed in.
218 * lread() - Private routine to read more than 64k
219 * lwrite() - Private routine to write more than 64k
220 */
221
222 #ifndef SEEK_CUR
223 /* flags for _lseek */
224 #define SEEK_CUR 1
225 #define SEEK_END 2
226 #define SEEK_SET 0
227 #endif
228
229 #define MAXREAD 32768 /* Number of bytes to be read during */
230 /* each read operation. */
231
232 /* Header signatutes for various resources */
233 #define BFT_ICON 0x4349 /* 'IC' */
234 #define BFT_BITMAP 0x4d42 /* 'BM' */
235 #define BFT_CURSOR 0x5450 /* 'PT(' */
236
237 /* macro to determine if resource is a DIB */
238 #define ISDIB(bft) ((bft) == BFT_BITMAP)
239
240 /* Macro to align given value to the closest DWORD (unsigned long ) */
241 #define ALIGNULONG(i) ((i+3)/4*4)
242
243 /* Macro to determine to round off the given value to the closest byte */
244 #define WIDTHBYTES(i) ((i+31)/32*4)
245
246 #define PALVERSION 0x300
247 #define MAXPALETTE 256 /* max. # supported palette entries */
248
249 static DWORD lread(int fh, VOID FAR *pv, DWORD ul);
250 static DWORD lwrite(int fh, VOID FAR *pv, DWORD ul);
251
252 static bool wxWriteDIB (LPTSTR szFile,HANDLE hdib);
253 WORD wxPaletteSize (VOID FAR * pv); // This is non-static as some apps use it externally
254 static WORD wxDibNumColors (VOID FAR * pv);
255 static bool wxMakeBitmapAndPalette(HDC,HANDLE,HPALETTE *,HBITMAP *);
256
257 /*
258 * FUNCTION : wxWriteDIB(LPSTR szFile,HANDLE hdib)
259 * PURPOSE : Write a global handle in CF_DIB format to a file.
260 * RETURNS : TRUE - if successful.
261 * FALSE - otherwise
262 */
263
264 static bool wxWriteDIB(LPTSTR szFile, HANDLE hdib)
265 {
266 BITMAPFILEHEADER hdr;
267 LPBITMAPINFOHEADER lpbi;
268 int fh;
269 OFSTRUCT of;
270
271 if (!hdib)
272 return FALSE;
273
274 fh = OpenFile(wxConvertWX2MB(szFile), &of, OF_CREATE | OF_READWRITE);
275 if (fh == -1)
276 return FALSE;
277
278 lpbi = (LPBITMAPINFOHEADER) GlobalLock(hdib);
279
280 /* Fill in the fields of the file header */
281 hdr.bfType = BFT_BITMAP;
282 hdr.bfSize = GlobalSize(hdib) + sizeof(BITMAPFILEHEADER);
283 hdr.bfReserved1 = 0;
284 hdr.bfReserved2 = 0;
285 hdr.bfOffBits = (DWORD) sizeof(BITMAPFILEHEADER) + lpbi->biSize +
286 wxPaletteSize(lpbi);
287
288 /* Write the file header */
289 _lwrite(fh, (LPSTR) &hdr, sizeof(BITMAPFILEHEADER));
290
291 /* Write the DIB header and the bits */
292 lwrite(fh, (LPSTR) lpbi, GlobalSize(hdib));
293
294 GlobalUnlock(hdib);
295 _lclose(fh);
296 return TRUE;
297 }
298
299 /*
300 * FUNCTION : wxPaletteSize(VOID FAR * pv)
301 * PURPOSE : Calculates the palette size in bytes. If the info. block
302 * is of the BITMAPCOREHEADER type, the number of colors is
303 * multiplied by 3 to give the palette size, otherwise the
304 * number of colors is multiplied by 4.
305 * RETURNS : Palette size in number of bytes.
306 */
307
308 WORD wxPaletteSize(VOID FAR * pv)
309 {
310 LPBITMAPINFOHEADER lpbi;
311 WORD NumColors;
312
313 lpbi = (LPBITMAPINFOHEADER) pv;
314 NumColors = wxDibNumColors(lpbi);
315
316 if (lpbi->biSize == sizeof(BITMAPCOREHEADER))
317 return (WORD)(NumColors * sizeof(RGBTRIPLE));
318 else
319 return (WORD)(NumColors * sizeof(RGBQUAD));
320 }
321
322 /*
323 * FUNCTION : wxDibNumColors(VOID FAR * pv)
324 * PURPOSE : Determines the number of colors in the DIB by looking at
325 * the BitCount filed in the info block.
326 * RETURNS : The number of colors in the DIB. *
327 */
328
329 static WORD wxDibNumColors(VOID FAR *pv)
330 {
331 int bits;
332 BITMAPINFOHEADER *lpbi;
333 BITMAPCOREHEADER *lpbc;
334
335 lpbi = ((BITMAPINFOHEADER*) pv);
336 lpbc = ((BITMAPCOREHEADER*) pv);
337
338 /* With the BITMAPINFO format headers, the size of the palette
339 * is in biClrUsed, whereas in the BITMAPCORE - style headers, it
340 * is dependent on the bits per pixel ( = 2 raised to the power of
341 * bits/pixel).
342 */
343 if (lpbi->biSize != sizeof(BITMAPCOREHEADER)) {
344 if (lpbi->biClrUsed != 0)
345 return (WORD) lpbi->biClrUsed;
346 bits = lpbi->biBitCount;
347 }
348 else
349 bits = lpbc->bcBitCount;
350
351 switch (bits) {
352 case 1:
353 return 2;
354 case 4:
355 return 16;
356 case 8:
357 return 256;
358 default:
359 /* A 24 bitcount DIB has no color table */
360 return 0;
361 }
362 }
363
364 /*
365 * FUNCTION : lread(int fh, VOID FAR *pv, DWORD ul)
366 * PURPOSE : Reads data in steps of 32k till all the data has been read.
367 * RETURNS : 0 - If read did not proceed correctly.
368 * number of bytes read otherwise.
369 */
370
371 static DWORD lread(int fh, void far *pv, DWORD ul)
372 {
373 DWORD ulT = ul;
374 #if defined(WINNT) || defined(__WIN32__) || defined(__WIN32__)
375 BYTE *hp = (BYTE *) pv;
376 #else
377 BYTE huge *hp = (BYTE huge *) pv;
378 #endif
379 while (ul > (DWORD) MAXREAD) {
380 if (_lread(fh, (LPSTR) hp, (WORD) MAXREAD) != MAXREAD)
381 return 0;
382 ul -= MAXREAD;
383 hp += MAXREAD;
384 }
385 if (_lread(fh, (LPSTR) hp, (WXUINT) ul) != (WXUINT) ul)
386 return 0;
387 return ulT;
388 }
389
390 /*
391 * FUNCTION : lwrite(int fh, VOID FAR *pv, DWORD ul)
392 * PURPOSE : Writes data in steps of 32k till all the data is written.
393 * RETURNS : 0 - If write did not proceed correctly.
394 * number of bytes written otherwise.
395 */
396
397 static DWORD lwrite(int fh, VOID FAR *pv, DWORD ul)
398 {
399 DWORD ulT = ul;
400 #if defined(WINNT) || defined(__WIN32__) || defined(__WIN32__)
401 BYTE *hp = (BYTE *) pv;
402 #else
403 BYTE huge *hp = (BYTE huge *) pv;
404 #endif
405 while (ul > MAXREAD) {
406 if (_lwrite(fh, (LPSTR) hp, (WORD) MAXREAD) != MAXREAD)
407 return 0;
408 ul -= MAXREAD;
409 hp += MAXREAD;
410 }
411 if (_lwrite(fh, (LPSTR) hp, (WXUINT) ul) != (WXUINT) ul)
412 return 0;
413 return ulT;
414 }
415
416 /*
417 * FUNCTION : wxReadDIB(hWnd)
418 * PURPOSE : Reads a DIB from a file, obtains a handle to its
419 * BITMAPINFO struct. and loads the DIB. Once the DIB
420 * is loaded, the function also creates a bitmap and
421 * palette out of the DIB for a device-dependent form.
422 * RETURNS : TRUE - DIB loaded and bitmap/palette created
423 * The DIBINIT structure pointed to by pInfo is
424 * filled with the appropriate handles.
425 * FALSE - otherwise
426 */
427
428 bool wxReadDIB(LPTSTR lpFileName, HBITMAP *bitmap, HPALETTE *palette)
429 {
430 int fh;
431 LPBITMAPINFOHEADER lpbi;
432 OFSTRUCT of;
433 BITMAPFILEHEADER bf;
434 WORD nNumColors;
435 bool result = FALSE;
436 WORD offBits;
437 HDC hDC;
438 bool bCoreHead = FALSE;
439 HANDLE hDIB = 0;
440
441 /* Open the file and get a handle to it's BITMAPINFO */
442
443 fh = OpenFile (wxConvertWX2MB(lpFileName), &of, OF_READ);
444 if (fh == -1) {
445 wxLogError(_("Can't open file '%s'"), lpFileName);
446 return (0);
447 }
448
449 hDIB = GlobalAlloc(GHND, (DWORD)(sizeof(BITMAPINFOHEADER) +
450 256 * sizeof(RGBQUAD)));
451 if (!hDIB)
452 return(0);
453
454 lpbi = (LPBITMAPINFOHEADER)GlobalLock(hDIB);
455
456 /* read the BITMAPFILEHEADER */
457 if (sizeof (bf) != _lread (fh, (LPSTR)&bf, sizeof (bf)))
458 goto ErrExit;
459
460 if (bf.bfType != 0x4d42) /* 'BM' */
461 goto ErrExit;
462
463 if (sizeof(BITMAPCOREHEADER) != _lread (fh, (LPSTR)lpbi, sizeof(BITMAPCOREHEADER)))
464 goto ErrExit;
465
466 if (lpbi->biSize == sizeof(BITMAPCOREHEADER))
467 {
468 lpbi->biSize = sizeof(BITMAPINFOHEADER);
469 lpbi->biBitCount = ((LPBITMAPCOREHEADER)lpbi)->bcBitCount;
470 lpbi->biPlanes = ((LPBITMAPCOREHEADER)lpbi)->bcPlanes;
471 lpbi->biHeight = ((LPBITMAPCOREHEADER)lpbi)->bcHeight;
472 lpbi->biWidth = ((LPBITMAPCOREHEADER)lpbi)->bcWidth;
473 bCoreHead = TRUE;
474 }
475 else
476 {
477 // get to the start of the header and read INFOHEADER
478 _llseek(fh,sizeof(BITMAPFILEHEADER),SEEK_SET);
479 if (sizeof(BITMAPINFOHEADER) != _lread (fh, (LPSTR)lpbi, sizeof(BITMAPINFOHEADER)))
480 goto ErrExit;
481 }
482
483 nNumColors = (WORD)lpbi->biClrUsed;
484 if ( nNumColors == 0 )
485 {
486 /* no color table for 24-bit, default size otherwise */
487 if (lpbi->biBitCount != 24)
488 nNumColors = 1 << lpbi->biBitCount; /* standard size table */
489 }
490
491 /* fill in some default values if they are zero */
492 if (lpbi->biClrUsed == 0)
493 lpbi->biClrUsed = nNumColors;
494
495 if (lpbi->biSizeImage == 0)
496 {
497 lpbi->biSizeImage = ((((lpbi->biWidth * (DWORD)lpbi->biBitCount) + 31) & ~31) >> 3)
498 * lpbi->biHeight;
499 }
500
501 /* get a proper-sized buffer for header, color table and bits */
502 GlobalUnlock(hDIB);
503 hDIB = GlobalReAlloc(hDIB, lpbi->biSize +
504 nNumColors * sizeof(RGBQUAD) +
505 lpbi->biSizeImage, 0);
506 if (!hDIB) /* can't resize buffer for loading */
507 goto ErrExit2;
508
509 lpbi = (LPBITMAPINFOHEADER)GlobalLock(hDIB);
510
511 /* read the color table */
512 if (!bCoreHead)
513 _lread(fh, (LPSTR)(lpbi) + lpbi->biSize, nNumColors * sizeof(RGBQUAD));
514 else
515 {
516 signed int i;
517 RGBQUAD FAR *pQuad;
518 RGBTRIPLE FAR *pTriple;
519
520 _lread(fh, (LPSTR)(lpbi) + lpbi->biSize, nNumColors * sizeof(RGBTRIPLE));
521
522 pQuad = (RGBQUAD FAR *)((LPSTR)lpbi + lpbi->biSize);
523 pTriple = (RGBTRIPLE FAR *) pQuad;
524 for (i = nNumColors - 1; i >= 0; i--)
525 {
526 pQuad[i].rgbRed = pTriple[i].rgbtRed;
527 pQuad[i].rgbBlue = pTriple[i].rgbtBlue;
528 pQuad[i].rgbGreen = pTriple[i].rgbtGreen;
529 pQuad[i].rgbReserved = 0;
530 }
531 }
532
533 /* offset to the bits from start of DIB header */
534 offBits = (WORD)(lpbi->biSize + nNumColors * sizeof(RGBQUAD));
535
536 if (bf.bfOffBits != 0L)
537 {
538 _llseek(fh,bf.bfOffBits,SEEK_SET);
539 }
540
541 if (lpbi->biSizeImage == lread(fh, (LPSTR)lpbi + offBits, lpbi->biSizeImage))
542 {
543 GlobalUnlock(hDIB);
544
545 hDC = GetDC(NULL);
546 if (!wxMakeBitmapAndPalette(hDC, hDIB, palette,
547 bitmap))
548 {
549 ReleaseDC(NULL,hDC);
550 goto ErrExit2;
551 }
552 else
553 {
554 ReleaseDC(NULL,hDC);
555 GlobalFree(hDIB);
556 result = TRUE;
557 }
558 }
559 else
560 {
561 ErrExit:
562 GlobalUnlock(hDIB);
563 ErrExit2:
564 GlobalFree(hDIB);
565 }
566
567 _lclose(fh);
568 return(result);
569 }
570
571 /*
572 * FUNCTION : wxMakeBitmapAndPalette
573 * PURPOSE : Given a DIB, creates a bitmap and corresponding palette
574 * to be used for a device-dependent representation of
575 * of the image.
576 * RETURNS : TRUE --> success. phPal and phBitmap are filled with
577 * appropriate handles. Caller is responsible
578 * for freeing objects.
579 * FALSE --> unable to create objects. both pointer are
580 * not valid
581 */
582
583 static bool wxMakeBitmapAndPalette(HDC hDC, HANDLE hDIB,
584 HPALETTE * phPal, HBITMAP * phBitmap)
585 {
586 LPBITMAPINFOHEADER lpInfo;
587 bool result = FALSE;
588 HBITMAP hBitmap;
589 HPALETTE hPalette, hOldPal;
590 LPSTR lpBits;
591
592 lpInfo = (LPBITMAPINFOHEADER) GlobalLock(hDIB);
593
594 hPalette = wxMakeDIBPalette(lpInfo);
595 if ( hPalette )
596 {
597 // Need to realize palette for converting DIB to bitmap.
598 hOldPal = SelectPalette(hDC, hPalette, TRUE);
599 RealizePalette(hDC);
600
601 lpBits = (LPSTR)lpInfo + (WORD)lpInfo->biSize +
602 (WORD)lpInfo->biClrUsed * sizeof(RGBQUAD);
603 hBitmap = CreateDIBitmap(hDC, lpInfo, CBM_INIT, lpBits,
604 (LPBITMAPINFO)lpInfo, DIB_RGB_COLORS);
605
606 SelectPalette(hDC, hOldPal, TRUE);
607 RealizePalette(hDC);
608
609 if (!hBitmap)
610 DeleteObject(hPalette);
611 else
612 {
613 *phBitmap = hBitmap;
614 *phPal = hPalette;
615 result = TRUE;
616 }
617 }
618
619 GlobalUnlock (hDIB); // glt
620
621 return(result);
622 }
623
624 /*
625 * FUNCTION : wxMakeDIBPalette(lpInfo)
626 * PURPOSE : Given a BITMAPINFOHEADER, create a palette based on
627 * the color table.
628 * RETURNS : non-zero - handle of a corresponding palette
629 * zero - unable to create palette
630 */
631
632 HPALETTE wxMakeDIBPalette(LPBITMAPINFOHEADER lpInfo)
633 {
634 LPLOGPALETTE npPal;
635 RGBQUAD far *lpRGB;
636 HPALETTE hLogPal;
637 WORD i;
638
639 /* since biClrUsed field was filled during the loading of the DIB,
640 * we know it contains the number of colors in the color table.
641 */
642 if (lpInfo->biClrUsed)
643 {
644 npPal = (LPLOGPALETTE)malloc(sizeof(LOGPALETTE) +
645 (WORD)lpInfo->biClrUsed * sizeof(PALETTEENTRY));
646 if (!npPal)
647 return NULL;
648
649 npPal->palVersion = 0x300;
650 npPal->palNumEntries = (WORD)lpInfo->biClrUsed;
651
652 /* get pointer to the color table */
653 lpRGB = (RGBQUAD FAR *)((LPSTR)lpInfo + lpInfo->biSize);
654
655 /* copy colors from the color table to the LogPalette structure */
656 for (i = 0; (DWORD)i < lpInfo->biClrUsed; i++, lpRGB++)
657 {
658 npPal->palPalEntry[i].peRed = lpRGB->rgbRed;
659 npPal->palPalEntry[i].peGreen = lpRGB->rgbGreen;
660 npPal->palPalEntry[i].peBlue = lpRGB->rgbBlue;
661 npPal->palPalEntry[i].peFlags = 0;
662 }
663
664 hLogPal = CreatePalette((LPLOGPALETTE)npPal);
665 free(npPal);
666
667 return(hLogPal);
668 }
669
670 /* 24-bit DIB with no color table. Return default palette. Another
671 * option would be to create a 256 color "rainbow" palette to provide
672 * some good color choices.
673 */
674 else
675 return((HPALETTE) GetStockObject(DEFAULT_PALETTE));
676 }
677
678 bool wxLoadIntoBitmap(wxChar *filename, wxBitmap *bitmap, wxPalette **pal)
679 {
680 HBITMAP hBitmap = NULL;
681 HPALETTE hPalette = NULL;
682
683 bool success = (wxReadDIB(filename, &hBitmap, &hPalette) != 0);
684
685 if (!success)
686 {
687 if (hPalette)
688 DeleteObject(hPalette);
689 return FALSE;
690 }
691
692 if (hPalette)
693 {
694 #if wxUSE_PALETTE
695 if (pal)
696 {
697 *pal = new wxPalette;
698 (*pal)->SetHPALETTE((WXHPALETTE) hPalette);
699 }
700 else
701 #endif // wxUSE_PALETTE
702 DeleteObject(hPalette);
703 }
704 else if (pal)
705 *pal = NULL;
706
707 if (hBitmap)
708 {
709 BITMAP bm;
710 GetObject(hBitmap, sizeof(bm), (LPSTR)&bm);
711
712 bitmap->SetHBITMAP((WXHBITMAP) hBitmap);
713 bitmap->SetWidth(bm.bmWidth);
714 bitmap->SetHeight(bm.bmHeight);
715 bitmap->SetDepth(bm.bmPlanes * bm.bmBitsPixel);
716 #if WXWIN_COMPATIBILITY_2
717 bitmap->SetOk(TRUE);
718 #endif // WXWIN_COMPATIBILITY_2
719 return TRUE;
720 }
721 else return FALSE;
722 }
723
724 wxBitmap *wxLoadBitmap(wxChar *filename, wxPalette **pal)
725 {
726 wxBitmap *bitmap = new wxBitmap;
727 if (wxLoadIntoBitmap(filename, bitmap, pal))
728 return bitmap;
729 else
730 {
731 delete bitmap;
732 return NULL;
733 }
734 }
735
736 /*
737 *
738 * Function: InitBitmapInfoHeader
739 *
740 * Purpose: Does a "standard" initialization of a BITMAPINFOHEADER,
741 * given the Width, Height, and Bits per Pixel for the
742 * DIB.
743 *
744 * By standard, I mean that all the relevant fields are set
745 * to the specified values. biSizeImage is computed, the
746 * biCompression field is set to "no compression," and all
747 * other fields are 0.
748 *
749 * Note that DIBs only allow BitsPixel values of 1, 4, 8, or
750 * 24. This routine makes sure that one of these values is
751 * used (whichever is most appropriate for the specified
752 * nBPP).
753 *
754 * Parms: lpBmInfoHdr == Far pointer to a BITMAPINFOHEADER structure
755 * to be filled in.
756 * dwWidth == Width of DIB (not in Win 3.0 & 3.1, high
757 * word MUST be 0).
758 * dwHeight == Height of DIB (not in Win 3.0 & 3.1, high
759 * word MUST be 0).
760 * nBPP == Bits per Pixel for the DIB.
761 *
762 */
763
764 static void InitBitmapInfoHeader (LPBITMAPINFOHEADER lpBmInfoHdr,
765 DWORD dwWidth,
766 DWORD dwHeight,
767 int nBPP)
768 {
769 // _fmemset (lpBmInfoHdr, 0, sizeof (BITMAPINFOHEADER));
770 memset (lpBmInfoHdr, 0, sizeof (BITMAPINFOHEADER));
771
772 lpBmInfoHdr->biSize = sizeof (BITMAPINFOHEADER);
773 lpBmInfoHdr->biWidth = dwWidth;
774 lpBmInfoHdr->biHeight = dwHeight;
775 lpBmInfoHdr->biPlanes = 1;
776
777 if (nBPP <= 1)
778 nBPP = 1;
779 else if (nBPP <= 4)
780 nBPP = 4;
781 else if (nBPP <= 8)
782 nBPP = 8;
783 /* Doesn't work
784 else if (nBPP <= 16)
785 nBPP = 16;
786 */
787 else
788 nBPP = 24;
789
790 lpBmInfoHdr->biBitCount = nBPP;
791 lpBmInfoHdr->biSizeImage = WIDTHBYTES (dwWidth * nBPP) * dwHeight;
792 }
793
794 LPSTR wxFindDIBBits (LPSTR lpbi)
795 {
796 return (lpbi + *(LPDWORD)lpbi + wxPaletteSize (lpbi));
797 }
798
799 /*
800 * Function: BitmapToDIB
801 *
802 * Purpose: Given a device dependent bitmap and a palette, returns
803 * a handle to global memory with a DIB spec in it. The
804 * DIB is rendered using the colors of the palette passed in.
805 *
806 * Parms: hBitmap == Handle to device dependent bitmap compatible
807 * with default screen display device.
808 * hPal == Palette to render the DDB with. If it's NULL,
809 * use the default palette.
810 */
811
812 HANDLE wxBitmapToDIB (HBITMAP hBitmap, HPALETTE hPal)
813 {
814 BITMAP Bitmap;
815 BITMAPINFOHEADER bmInfoHdr;
816 LPBITMAPINFOHEADER lpbmInfoHdr;
817 LPSTR lpBits;
818 HDC hMemDC;
819 HANDLE hDIB;
820 HPALETTE hOldPal = NULL;
821
822 // Do some setup -- make sure the Bitmap passed in is valid,
823 // get info on the bitmap (like its height, width, etc.),
824 // then setup a BITMAPINFOHEADER.
825
826 if (!hBitmap)
827 return NULL;
828
829 if (!GetObject (hBitmap, sizeof (Bitmap), (LPSTR) &Bitmap))
830 return NULL;
831
832 InitBitmapInfoHeader (&bmInfoHdr,
833 Bitmap.bmWidth,
834 Bitmap.bmHeight,
835 Bitmap.bmPlanes * Bitmap.bmBitsPixel);
836
837 // Now allocate memory for the DIB. Then, set the BITMAPINFOHEADER
838 // into this memory, and find out where the bitmap bits go.
839
840 hDIB = GlobalAlloc (GHND, sizeof (BITMAPINFOHEADER) +
841 wxPaletteSize ((LPSTR) &bmInfoHdr) + bmInfoHdr.biSizeImage);
842
843 if (!hDIB)
844 return NULL;
845
846 lpbmInfoHdr = (LPBITMAPINFOHEADER) GlobalLock (hDIB);
847
848 *lpbmInfoHdr = bmInfoHdr;
849 lpBits = wxFindDIBBits ((LPSTR) lpbmInfoHdr);
850
851
852 // Now, we need a DC to hold our bitmap. If the app passed us
853 // a palette, it should be selected into the DC.
854
855 hMemDC = GetDC (NULL);
856
857 if (hPal)
858 {
859 hOldPal = SelectPalette (hMemDC, hPal, FALSE);
860 RealizePalette (hMemDC);
861 }
862
863 // We're finally ready to get the DIB. Call the driver and let
864 // it party on our bitmap. It will fill in the color table,
865 // and bitmap bits of our global memory block.
866
867 if (!GetDIBits (hMemDC, hBitmap, 0, Bitmap.bmHeight, lpBits,
868 (LPBITMAPINFO) lpbmInfoHdr, DIB_RGB_COLORS))
869 {
870 GlobalUnlock (hDIB);
871 GlobalFree (hDIB);
872 hDIB = NULL;
873 }
874 else
875 GlobalUnlock (hDIB);
876
877 // Finally, clean up and return.
878
879 if (hOldPal)
880 SelectPalette (hMemDC, hOldPal, FALSE);
881
882 ReleaseDC (NULL, hMemDC);
883
884 return hDIB;
885 }
886
887 bool wxSaveBitmap(wxChar *filename, wxBitmap *bitmap, wxPalette *palette)
888 {
889 HPALETTE hPalette = 0;
890 #if wxUSE_PALETTE
891 if (palette)
892 hPalette = (HPALETTE) palette->GetHPALETTE();
893 #endif // wxUSE_PALETTE
894
895 HANDLE dibHandle = wxBitmapToDIB((HBITMAP) bitmap->GetHBITMAP(), hPalette);
896 if (dibHandle)
897 {
898 bool success = (wxWriteDIB(filename, dibHandle) != 0);
899 GlobalFree(dibHandle);
900 return success;
901 }
902 else return FALSE;
903 }
904