1 |
william |
31 |
///////////////////////////////////////////////////////////////////////////// |
2 |
|
|
// Name: src/common/imagepng.cpp |
3 |
|
|
// Purpose: wxImage PNG handler |
4 |
|
|
// Author: Robert Roebling |
5 |
|
|
// RCS-ID: $Id: imagpng.cpp 53479 2008-05-07 16:23:55Z PC $ |
6 |
|
|
// Copyright: (c) Robert Roebling |
7 |
|
|
// Licence: wxWindows licence |
8 |
|
|
///////////////////////////////////////////////////////////////////////////// |
9 |
|
|
|
10 |
|
|
// ============================================================================ |
11 |
|
|
// declarations |
12 |
|
|
// ============================================================================ |
13 |
|
|
|
14 |
|
|
// ---------------------------------------------------------------------------- |
15 |
|
|
// headers |
16 |
|
|
// ---------------------------------------------------------------------------- |
17 |
|
|
|
18 |
|
|
// For compilers that support precompilation, includes "wx.h". |
19 |
|
|
#include "wx/wxprec.h" |
20 |
|
|
|
21 |
|
|
#ifdef __BORLANDC__ |
22 |
|
|
#pragma hdrstop |
23 |
|
|
#endif |
24 |
|
|
|
25 |
|
|
#if wxUSE_IMAGE && wxUSE_LIBPNG |
26 |
|
|
|
27 |
|
|
#include "wx/imagpng.h" |
28 |
|
|
|
29 |
|
|
#ifndef WX_PRECOMP |
30 |
|
|
#include "wx/log.h" |
31 |
|
|
#include "wx/app.h" |
32 |
|
|
#include "wx/bitmap.h" |
33 |
|
|
#include "wx/module.h" |
34 |
|
|
#endif |
35 |
|
|
|
36 |
|
|
#include "png.h" |
37 |
|
|
#include "wx/filefn.h" |
38 |
|
|
#include "wx/wfstream.h" |
39 |
|
|
#include "wx/intl.h" |
40 |
|
|
#include "wx/palette.h" |
41 |
|
|
|
42 |
|
|
// For memcpy |
43 |
|
|
#include <string.h> |
44 |
|
|
|
45 |
|
|
#ifdef __SALFORDC__ |
46 |
|
|
#ifdef FAR |
47 |
|
|
#undef FAR |
48 |
|
|
#endif |
49 |
|
|
#endif |
50 |
|
|
|
51 |
|
|
// ---------------------------------------------------------------------------- |
52 |
|
|
// constants |
53 |
|
|
// ---------------------------------------------------------------------------- |
54 |
|
|
|
55 |
|
|
// image can not have any transparent pixels at all, have only 100% opaque |
56 |
|
|
// and/or 100% transparent pixels in which case a simple mask is enough to |
57 |
|
|
// store this information in wxImage or have a real alpha channel in which case |
58 |
|
|
// we need to have it in wxImage as well |
59 |
|
|
enum Transparency |
60 |
|
|
{ |
61 |
|
|
Transparency_None, |
62 |
|
|
Transparency_Mask, |
63 |
|
|
Transparency_Alpha |
64 |
|
|
}; |
65 |
|
|
|
66 |
|
|
// ---------------------------------------------------------------------------- |
67 |
|
|
// local functions |
68 |
|
|
// ---------------------------------------------------------------------------- |
69 |
|
|
|
70 |
|
|
// return the kind of transparency needed for this image assuming that it does |
71 |
|
|
// have transparent pixels, i.e. either Transparency_Alpha or Transparency_Mask |
72 |
|
|
static Transparency |
73 |
|
|
CheckTransparency(unsigned char **lines, |
74 |
|
|
png_uint_32 x, png_uint_32 y, png_uint_32 w, png_uint_32 h, |
75 |
|
|
size_t numColBytes); |
76 |
|
|
|
77 |
|
|
// init the alpha channel for the image and fill it with 1s up to (x, y) |
78 |
|
|
static unsigned char *InitAlpha(wxImage *image, png_uint_32 x, png_uint_32 y); |
79 |
|
|
|
80 |
|
|
// find a free colour for the mask in the PNG data array |
81 |
|
|
static void |
82 |
|
|
FindMaskColour(unsigned char **lines, png_uint_32 width, png_uint_32 height, |
83 |
|
|
unsigned char& rMask, unsigned char& gMask, unsigned char& bMask); |
84 |
|
|
|
85 |
|
|
// is the pixel with this value of alpha a fully opaque one? |
86 |
|
|
static inline |
87 |
|
|
bool IsOpaque(unsigned char a) |
88 |
|
|
{ |
89 |
|
|
return a == 0xff; |
90 |
|
|
} |
91 |
|
|
|
92 |
|
|
// is the pixel with this value of alpha a fully transparent one? |
93 |
|
|
static inline |
94 |
|
|
bool IsTransparent(unsigned char a) |
95 |
|
|
{ |
96 |
|
|
return !a; |
97 |
|
|
} |
98 |
|
|
|
99 |
|
|
// ============================================================================ |
100 |
|
|
// wxPNGHandler implementation |
101 |
|
|
// ============================================================================ |
102 |
|
|
|
103 |
|
|
IMPLEMENT_DYNAMIC_CLASS(wxPNGHandler,wxImageHandler) |
104 |
|
|
|
105 |
|
|
#if wxUSE_STREAMS |
106 |
|
|
|
107 |
|
|
#ifndef PNGLINKAGEMODE |
108 |
|
|
#ifdef PNGAPI |
109 |
|
|
#define PNGLINKAGEMODE PNGAPI |
110 |
|
|
#elif defined(__WATCOMC__) |
111 |
|
|
// we need an explicit cdecl for Watcom, at least according to |
112 |
|
|
// |
113 |
|
|
// http://sf.net/tracker/index.php?func=detail&aid=651492&group_id=9863&atid=109863 |
114 |
|
|
// |
115 |
|
|
// more testing is needed for this however, please remove this comment |
116 |
|
|
// if you can confirm that my fix works with Watcom 11 |
117 |
|
|
#define PNGLINKAGEMODE cdecl |
118 |
|
|
#else |
119 |
|
|
#define PNGLINKAGEMODE LINKAGEMODE |
120 |
|
|
#endif |
121 |
|
|
#endif |
122 |
|
|
|
123 |
|
|
|
124 |
|
|
// VS: wxPNGInfoStruct declared below is a hack that needs some explanation. |
125 |
|
|
// First, let me describe what's the problem: libpng uses jmp_buf in |
126 |
|
|
// its png_struct structure. Unfortunately, this structure is |
127 |
|
|
// compiler-specific and may vary in size, so if you use libpng compiled |
128 |
|
|
// as DLL with another compiler than the main executable, it may not work |
129 |
|
|
// (this is for example the case with wxMGL port and SciTech MGL library |
130 |
|
|
// that provides custom runtime-loadable libpng implementation with jmpbuf |
131 |
|
|
// disabled altogether). Luckily, it is still possible to use setjmp() & |
132 |
|
|
// longjmp() as long as the structure is not part of png_struct. |
133 |
|
|
// |
134 |
|
|
// Sadly, there's no clean way to attach user-defined data to png_struct. |
135 |
|
|
// There is only one customizable place, png_struct.io_ptr, which is meant |
136 |
|
|
// only for I/O routines and is set with png_set_read_fn or |
137 |
|
|
// png_set_write_fn. The hacky part is that we use io_ptr to store |
138 |
|
|
// a pointer to wxPNGInfoStruct that holds I/O structures _and_ jmp_buf. |
139 |
|
|
|
140 |
|
|
struct wxPNGInfoStruct |
141 |
|
|
{ |
142 |
|
|
jmp_buf jmpbuf; |
143 |
|
|
bool verbose; |
144 |
|
|
|
145 |
|
|
union |
146 |
|
|
{ |
147 |
|
|
wxInputStream *in; |
148 |
|
|
wxOutputStream *out; |
149 |
|
|
} stream; |
150 |
|
|
}; |
151 |
|
|
|
152 |
|
|
#define WX_PNG_INFO(png_ptr) ((wxPNGInfoStruct*)png_get_io_ptr(png_ptr)) |
153 |
|
|
|
154 |
|
|
// ---------------------------------------------------------------------------- |
155 |
|
|
// helper functions |
156 |
|
|
// ---------------------------------------------------------------------------- |
157 |
|
|
|
158 |
|
|
extern "C" |
159 |
|
|
{ |
160 |
|
|
|
161 |
|
|
void PNGLINKAGEMODE wx_PNG_stream_reader( png_structp png_ptr, png_bytep data, |
162 |
|
|
png_size_t length ) |
163 |
|
|
{ |
164 |
|
|
WX_PNG_INFO(png_ptr)->stream.in->Read(data, length); |
165 |
|
|
} |
166 |
|
|
|
167 |
|
|
void PNGLINKAGEMODE wx_PNG_stream_writer( png_structp png_ptr, png_bytep data, |
168 |
|
|
png_size_t length ) |
169 |
|
|
{ |
170 |
|
|
WX_PNG_INFO(png_ptr)->stream.out->Write(data, length); |
171 |
|
|
} |
172 |
|
|
|
173 |
|
|
void |
174 |
|
|
PNGLINKAGEMODE wx_png_warning(png_structp png_ptr, png_const_charp message) |
175 |
|
|
{ |
176 |
|
|
wxPNGInfoStruct *info = png_ptr ? WX_PNG_INFO(png_ptr) : NULL; |
177 |
|
|
if ( !info || info->verbose ) |
178 |
|
|
wxLogWarning( wxString::FromAscii(message) ); |
179 |
|
|
} |
180 |
|
|
|
181 |
|
|
// from pngerror.c |
182 |
|
|
// so that the libpng doesn't send anything on stderr |
183 |
|
|
void |
184 |
|
|
PNGLINKAGEMODE wx_png_error(png_structp png_ptr, png_const_charp message) |
185 |
|
|
{ |
186 |
|
|
wx_png_warning(NULL, message); |
187 |
|
|
|
188 |
|
|
// we're not using libpng built-in jump buffer (see comment before |
189 |
|
|
// wxPNGInfoStruct above) so we have to return ourselves, otherwise libpng |
190 |
|
|
// would just abort |
191 |
|
|
longjmp(WX_PNG_INFO(png_ptr)->jmpbuf, 1); |
192 |
|
|
} |
193 |
|
|
|
194 |
|
|
} // extern "C" |
195 |
|
|
|
196 |
|
|
// ---------------------------------------------------------------------------- |
197 |
|
|
// LoadFile() helpers |
198 |
|
|
// ---------------------------------------------------------------------------- |
199 |
|
|
|
200 |
|
|
// determine the kind of transparency we need for this image: if the only alpha |
201 |
|
|
// values it has are 0 (transparent) and 0xff (opaque) then we can simply |
202 |
|
|
// create a mask for it, we should be ok with a simple mask but otherwise we |
203 |
|
|
// need a full blown alpha channel in wxImage |
204 |
|
|
// |
205 |
|
|
// parameters: |
206 |
|
|
// lines raw PNG data |
207 |
|
|
// x, y starting position |
208 |
|
|
// w, h size of the image |
209 |
|
|
// numColBytes number of colour bytes (1 for grey scale, 3 for RGB) |
210 |
|
|
// (NB: alpha always follows the colour bytes) |
211 |
|
|
Transparency |
212 |
|
|
CheckTransparency(unsigned char **lines, |
213 |
|
|
png_uint_32 x, png_uint_32 y, png_uint_32 w, png_uint_32 h, |
214 |
|
|
size_t numColBytes) |
215 |
|
|
{ |
216 |
|
|
// suppose that a mask will suffice and check all the remaining alpha |
217 |
|
|
// values to see if it does |
218 |
|
|
for ( ; y < h; y++ ) |
219 |
|
|
{ |
220 |
|
|
// each pixel is numColBytes+1 bytes, offset into the current line by |
221 |
|
|
// the current x position |
222 |
|
|
unsigned const char *ptr = lines[y] + (x * (numColBytes + 1)); |
223 |
|
|
|
224 |
|
|
for ( png_uint_32 x2 = x; x2 < w; x2++ ) |
225 |
|
|
{ |
226 |
|
|
// skip the grey or colour byte(s) |
227 |
|
|
ptr += numColBytes; |
228 |
|
|
|
229 |
|
|
unsigned char a2 = *ptr++; |
230 |
|
|
|
231 |
|
|
if ( !IsTransparent(a2) && !IsOpaque(a2) ) |
232 |
|
|
{ |
233 |
|
|
// not fully opaque nor fully transparent, hence need alpha |
234 |
|
|
return Transparency_Alpha; |
235 |
|
|
} |
236 |
|
|
} |
237 |
|
|
|
238 |
|
|
// during the next loop iteration check all the pixels in the row |
239 |
|
|
x = 0; |
240 |
|
|
} |
241 |
|
|
|
242 |
|
|
// mask will be enough |
243 |
|
|
return Transparency_Mask; |
244 |
|
|
} |
245 |
|
|
|
246 |
|
|
unsigned char *InitAlpha(wxImage *image, png_uint_32 x, png_uint_32 y) |
247 |
|
|
{ |
248 |
|
|
// create alpha channel |
249 |
|
|
image->SetAlpha(); |
250 |
|
|
|
251 |
|
|
unsigned char *alpha = image->GetAlpha(); |
252 |
|
|
|
253 |
|
|
// set alpha for the pixels we had so far |
254 |
|
|
png_uint_32 end = y * image->GetWidth() + x; |
255 |
|
|
for ( png_uint_32 i = 0; i < end; i++ ) |
256 |
|
|
{ |
257 |
|
|
// all the previous pixels were opaque |
258 |
|
|
*alpha++ = 0xff; |
259 |
|
|
} |
260 |
|
|
|
261 |
|
|
return alpha; |
262 |
|
|
} |
263 |
|
|
|
264 |
|
|
void |
265 |
|
|
FindMaskColour(unsigned char **lines, png_uint_32 width, png_uint_32 height, |
266 |
|
|
unsigned char& rMask, unsigned char& gMask, unsigned char& bMask) |
267 |
|
|
{ |
268 |
|
|
// choosing the colour for the mask is more |
269 |
|
|
// difficult: we need to iterate over the entire |
270 |
|
|
// image for this in order to choose an unused |
271 |
|
|
// colour (this is not very efficient but what else |
272 |
|
|
// can we do?) |
273 |
|
|
wxImageHistogram h; |
274 |
|
|
unsigned nentries = 0; |
275 |
|
|
unsigned char r2, g2, b2; |
276 |
|
|
for ( png_uint_32 y2 = 0; y2 < height; y2++ ) |
277 |
|
|
{ |
278 |
|
|
const unsigned char *p = lines[y2]; |
279 |
|
|
for ( png_uint_32 x2 = 0; x2 < width; x2++ ) |
280 |
|
|
{ |
281 |
|
|
r2 = *p++; |
282 |
|
|
g2 = *p++; |
283 |
|
|
b2 = *p++; |
284 |
|
|
++p; // jump over alpha |
285 |
|
|
|
286 |
|
|
wxImageHistogramEntry& |
287 |
|
|
entry = h[wxImageHistogram:: MakeKey(r2, g2, b2)]; |
288 |
|
|
|
289 |
|
|
if ( entry.value++ == 0 ) |
290 |
|
|
entry.index = nentries++; |
291 |
|
|
} |
292 |
|
|
} |
293 |
|
|
|
294 |
|
|
if ( !h.FindFirstUnusedColour(&rMask, &gMask, &bMask) ) |
295 |
|
|
{ |
296 |
|
|
wxLogWarning(_("Too many colours in PNG, the image may be slightly blurred.")); |
297 |
|
|
|
298 |
|
|
// use a fixed mask colour and we'll fudge |
299 |
|
|
// the real pixels with this colour (see |
300 |
|
|
// below) |
301 |
|
|
rMask = 0xfe; |
302 |
|
|
gMask = 0; |
303 |
|
|
bMask = 0xff; |
304 |
|
|
} |
305 |
|
|
} |
306 |
|
|
|
307 |
|
|
// ---------------------------------------------------------------------------- |
308 |
|
|
// reading PNGs |
309 |
|
|
// ---------------------------------------------------------------------------- |
310 |
|
|
|
311 |
|
|
bool wxPNGHandler::DoCanRead( wxInputStream& stream ) |
312 |
|
|
{ |
313 |
|
|
unsigned char hdr[4]; |
314 |
|
|
|
315 |
|
|
if ( !stream.Read(hdr, WXSIZEOF(hdr)) ) |
316 |
|
|
return false; |
317 |
|
|
|
318 |
|
|
return memcmp(hdr, "\211PNG", WXSIZEOF(hdr)) == 0; |
319 |
|
|
} |
320 |
|
|
|
321 |
|
|
// convert data from RGB to wxImage format |
322 |
|
|
static |
323 |
|
|
void CopyDataFromPNG(wxImage *image, |
324 |
|
|
unsigned char **lines, |
325 |
|
|
png_uint_32 width, |
326 |
|
|
png_uint_32 height, |
327 |
|
|
int color_type) |
328 |
|
|
{ |
329 |
|
|
Transparency transparency = Transparency_None; |
330 |
|
|
|
331 |
|
|
// only non NULL if transparency == Transparency_Alpha |
332 |
|
|
unsigned char *alpha = NULL; |
333 |
|
|
|
334 |
|
|
// RGB of the mask colour if transparency == Transparency_Mask |
335 |
|
|
// (but init them anyhow to avoid compiler warnings) |
336 |
|
|
unsigned char rMask = 0, |
337 |
|
|
gMask = 0, |
338 |
|
|
bMask = 0; |
339 |
|
|
|
340 |
|
|
unsigned char *ptrDst = image->GetData(); |
341 |
|
|
if ( !(color_type & PNG_COLOR_MASK_COLOR) ) |
342 |
|
|
{ |
343 |
|
|
// grey image: GAGAGA... where G == grey component and A == alpha |
344 |
|
|
for ( png_uint_32 y = 0; y < height; y++ ) |
345 |
|
|
{ |
346 |
|
|
const unsigned char *ptrSrc = lines[y]; |
347 |
|
|
for ( png_uint_32 x = 0; x < width; x++ ) |
348 |
|
|
{ |
349 |
|
|
unsigned char g = *ptrSrc++; |
350 |
|
|
unsigned char a = *ptrSrc++; |
351 |
|
|
|
352 |
|
|
// the first time we encounter a transparent pixel we must |
353 |
|
|
// decide about what to do about them |
354 |
|
|
if ( !IsOpaque(a) && transparency == Transparency_None ) |
355 |
|
|
{ |
356 |
|
|
// we'll need at least the mask for this image and |
357 |
|
|
// maybe even full alpha channel info: the former is |
358 |
|
|
// only enough if we have alpha values of 0 and 0xff |
359 |
|
|
// only, otherwisewe need the latter |
360 |
|
|
transparency = CheckTransparency |
361 |
|
|
( |
362 |
|
|
lines, |
363 |
|
|
x, y, |
364 |
|
|
width, height, |
365 |
|
|
1 |
366 |
|
|
); |
367 |
|
|
|
368 |
|
|
if ( transparency == Transparency_Mask ) |
369 |
|
|
{ |
370 |
|
|
// let's choose this colour for the mask: this is |
371 |
|
|
// not a problem here as all the other pixels are |
372 |
|
|
// grey, i.e. R == G == B which is not the case for |
373 |
|
|
// this one so no confusion is possible |
374 |
|
|
rMask = 0xff; |
375 |
|
|
gMask = 0; |
376 |
|
|
bMask = 0xff; |
377 |
|
|
} |
378 |
|
|
else // transparency == Transparency_Alpha |
379 |
|
|
{ |
380 |
|
|
alpha = InitAlpha(image, x, y); |
381 |
|
|
} |
382 |
|
|
} |
383 |
|
|
|
384 |
|
|
switch ( transparency ) |
385 |
|
|
{ |
386 |
|
|
case Transparency_Mask: |
387 |
|
|
if ( IsTransparent(a) ) |
388 |
|
|
{ |
389 |
|
|
*ptrDst++ = rMask; |
390 |
|
|
*ptrDst++ = gMask; |
391 |
|
|
*ptrDst++ = bMask; |
392 |
|
|
break; |
393 |
|
|
} |
394 |
|
|
// else: !transparent |
395 |
|
|
|
396 |
|
|
// must be opaque then as otherwise we shouldn't be |
397 |
|
|
// using the mask at all |
398 |
|
|
wxASSERT_MSG( IsOpaque(a), _T("logic error") ); |
399 |
|
|
|
400 |
|
|
// fall through |
401 |
|
|
|
402 |
|
|
case Transparency_Alpha: |
403 |
|
|
if ( alpha ) |
404 |
|
|
*alpha++ = a; |
405 |
|
|
// fall through |
406 |
|
|
|
407 |
|
|
case Transparency_None: |
408 |
|
|
*ptrDst++ = g; |
409 |
|
|
*ptrDst++ = g; |
410 |
|
|
*ptrDst++ = g; |
411 |
|
|
break; |
412 |
|
|
} |
413 |
|
|
} |
414 |
|
|
} |
415 |
|
|
} |
416 |
|
|
else // colour image: RGBRGB... |
417 |
|
|
{ |
418 |
|
|
for ( png_uint_32 y = 0; y < height; y++ ) |
419 |
|
|
{ |
420 |
|
|
const unsigned char *ptrSrc = lines[y]; |
421 |
|
|
for ( png_uint_32 x = 0; x < width; x++ ) |
422 |
|
|
{ |
423 |
|
|
unsigned char r = *ptrSrc++; |
424 |
|
|
unsigned char g = *ptrSrc++; |
425 |
|
|
unsigned char b = *ptrSrc++; |
426 |
|
|
unsigned char a = *ptrSrc++; |
427 |
|
|
|
428 |
|
|
// the logic here is the same as for the grey case except |
429 |
|
|
// where noted |
430 |
|
|
if ( !IsOpaque(a) && transparency == Transparency_None ) |
431 |
|
|
{ |
432 |
|
|
transparency = CheckTransparency |
433 |
|
|
( |
434 |
|
|
lines, |
435 |
|
|
x, y, |
436 |
|
|
width, height, |
437 |
|
|
3 |
438 |
|
|
); |
439 |
|
|
|
440 |
|
|
if ( transparency == Transparency_Mask ) |
441 |
|
|
{ |
442 |
|
|
FindMaskColour(lines, width, height, |
443 |
|
|
rMask, gMask, bMask); |
444 |
|
|
} |
445 |
|
|
else // transparency == Transparency_Alpha |
446 |
|
|
{ |
447 |
|
|
alpha = InitAlpha(image, x, y); |
448 |
|
|
} |
449 |
|
|
|
450 |
|
|
} |
451 |
|
|
|
452 |
|
|
switch ( transparency ) |
453 |
|
|
{ |
454 |
|
|
case Transparency_Mask: |
455 |
|
|
if ( IsTransparent(a) ) |
456 |
|
|
{ |
457 |
|
|
*ptrDst++ = rMask; |
458 |
|
|
*ptrDst++ = gMask; |
459 |
|
|
*ptrDst++ = bMask; |
460 |
|
|
break; |
461 |
|
|
} |
462 |
|
|
else // !transparent |
463 |
|
|
{ |
464 |
|
|
// must be opaque then as otherwise we shouldn't be |
465 |
|
|
// using the mask at all |
466 |
|
|
wxASSERT_MSG( IsOpaque(a), _T("logic error") ); |
467 |
|
|
|
468 |
|
|
// if we couldn't find a unique colour for the |
469 |
|
|
// mask, we can have real pixels with the same |
470 |
|
|
// value as the mask and it's better to slightly |
471 |
|
|
// change their colour than to make them |
472 |
|
|
// transparent |
473 |
|
|
if ( r == rMask && g == gMask && b == bMask ) |
474 |
|
|
{ |
475 |
|
|
r++; |
476 |
|
|
} |
477 |
|
|
} |
478 |
|
|
|
479 |
|
|
// fall through |
480 |
|
|
|
481 |
|
|
case Transparency_Alpha: |
482 |
|
|
if ( alpha ) |
483 |
|
|
*alpha++ = a; |
484 |
|
|
// fall through |
485 |
|
|
|
486 |
|
|
case Transparency_None: |
487 |
|
|
*ptrDst++ = r; |
488 |
|
|
*ptrDst++ = g; |
489 |
|
|
*ptrDst++ = b; |
490 |
|
|
break; |
491 |
|
|
} |
492 |
|
|
} |
493 |
|
|
} |
494 |
|
|
} |
495 |
|
|
|
496 |
|
|
if ( transparency == Transparency_Mask ) |
497 |
|
|
{ |
498 |
|
|
image->SetMaskColour(rMask, gMask, bMask); |
499 |
|
|
} |
500 |
|
|
} |
501 |
|
|
|
502 |
|
|
// temporarily disable the warning C4611 (interaction between '_setjmp' and |
503 |
|
|
// C++ object destruction is non-portable) - I don't see any dtors here |
504 |
|
|
#ifdef __VISUALC__ |
505 |
|
|
#pragma warning(disable:4611) |
506 |
|
|
#endif /* VC++ */ |
507 |
|
|
|
508 |
|
|
bool |
509 |
|
|
wxPNGHandler::LoadFile(wxImage *image, |
510 |
|
|
wxInputStream& stream, |
511 |
|
|
bool verbose, |
512 |
|
|
int WXUNUSED(index)) |
513 |
|
|
{ |
514 |
|
|
// VZ: as this function uses setjmp() the only fool-proof error handling |
515 |
|
|
// method is to use goto (setjmp is not really C++ dtors friendly...) |
516 |
|
|
|
517 |
|
|
unsigned char **lines = NULL; |
518 |
|
|
png_infop info_ptr = (png_infop) NULL; |
519 |
|
|
wxPNGInfoStruct wxinfo; |
520 |
|
|
|
521 |
|
|
png_uint_32 i, width, height = 0; |
522 |
|
|
int bit_depth, color_type, interlace_type; |
523 |
|
|
|
524 |
|
|
wxinfo.verbose = verbose; |
525 |
|
|
wxinfo.stream.in = &stream; |
526 |
|
|
|
527 |
|
|
image->Destroy(); |
528 |
|
|
|
529 |
|
|
png_structp png_ptr = png_create_read_struct |
530 |
|
|
( |
531 |
|
|
PNG_LIBPNG_VER_STRING, |
532 |
|
|
(voidp) NULL, |
533 |
|
|
wx_png_error, |
534 |
|
|
wx_png_warning |
535 |
|
|
); |
536 |
|
|
if (!png_ptr) |
537 |
|
|
goto error; |
538 |
|
|
|
539 |
|
|
// NB: please see the comment near wxPNGInfoStruct declaration for |
540 |
|
|
// explanation why this line is mandatory |
541 |
|
|
png_set_read_fn( png_ptr, &wxinfo, wx_PNG_stream_reader); |
542 |
|
|
|
543 |
|
|
info_ptr = png_create_info_struct( png_ptr ); |
544 |
|
|
if (!info_ptr) |
545 |
|
|
goto error; |
546 |
|
|
|
547 |
|
|
if (setjmp(wxinfo.jmpbuf)) |
548 |
|
|
goto error; |
549 |
|
|
|
550 |
|
|
png_read_info( png_ptr, info_ptr ); |
551 |
|
|
png_get_IHDR( png_ptr, info_ptr, &width, &height, &bit_depth, &color_type, &interlace_type, (int*) NULL, (int*) NULL ); |
552 |
|
|
|
553 |
|
|
if (color_type == PNG_COLOR_TYPE_PALETTE) |
554 |
|
|
png_set_expand( png_ptr ); |
555 |
|
|
|
556 |
|
|
// Fix for Bug [ 439207 ] Monochrome PNG images come up black |
557 |
|
|
if (bit_depth < 8) |
558 |
|
|
png_set_expand( png_ptr ); |
559 |
|
|
|
560 |
|
|
png_set_strip_16( png_ptr ); |
561 |
|
|
png_set_packing( png_ptr ); |
562 |
|
|
if (png_get_valid( png_ptr, info_ptr, PNG_INFO_tRNS)) |
563 |
|
|
png_set_expand( png_ptr ); |
564 |
|
|
png_set_filler( png_ptr, 0xff, PNG_FILLER_AFTER ); |
565 |
|
|
|
566 |
|
|
image->Create((int)width, (int)height, (bool) false /* no need to init pixels */); |
567 |
|
|
|
568 |
|
|
if (!image->Ok()) |
569 |
|
|
goto error; |
570 |
|
|
|
571 |
|
|
lines = (unsigned char **)malloc( (size_t)(height * sizeof(unsigned char *)) ); |
572 |
|
|
if ( !lines ) |
573 |
|
|
goto error; |
574 |
|
|
|
575 |
|
|
for (i = 0; i < height; i++) |
576 |
|
|
{ |
577 |
|
|
if ((lines[i] = (unsigned char *)malloc( (size_t)(width * (sizeof(unsigned char) * 4)))) == NULL) |
578 |
|
|
{ |
579 |
|
|
for ( unsigned int n = 0; n < i; n++ ) |
580 |
|
|
free( lines[n] ); |
581 |
|
|
goto error; |
582 |
|
|
} |
583 |
|
|
} |
584 |
|
|
|
585 |
|
|
png_read_image( png_ptr, lines ); |
586 |
|
|
png_read_end( png_ptr, info_ptr ); |
587 |
|
|
|
588 |
|
|
#if wxUSE_PALETTE |
589 |
|
|
if (color_type == PNG_COLOR_TYPE_PALETTE) |
590 |
|
|
{ |
591 |
|
|
const size_t ncolors = info_ptr->num_palette; |
592 |
|
|
unsigned char* r = new unsigned char[ncolors]; |
593 |
|
|
unsigned char* g = new unsigned char[ncolors]; |
594 |
|
|
unsigned char* b = new unsigned char[ncolors]; |
595 |
|
|
|
596 |
|
|
for (size_t j = 0; j < ncolors; j++) |
597 |
|
|
{ |
598 |
|
|
r[j] = info_ptr->palette[j].red; |
599 |
|
|
g[j] = info_ptr->palette[j].green; |
600 |
|
|
b[j] = info_ptr->palette[j].blue; |
601 |
|
|
} |
602 |
|
|
|
603 |
|
|
image->SetPalette(wxPalette(ncolors, r, g, b)); |
604 |
|
|
delete[] r; |
605 |
|
|
delete[] g; |
606 |
|
|
delete[] b; |
607 |
|
|
} |
608 |
|
|
#endif // wxUSE_PALETTE |
609 |
|
|
|
610 |
|
|
png_destroy_read_struct( &png_ptr, &info_ptr, (png_infopp) NULL ); |
611 |
|
|
|
612 |
|
|
// loaded successfully, now init wxImage with this data |
613 |
|
|
CopyDataFromPNG(image, lines, width, height, color_type); |
614 |
|
|
|
615 |
|
|
for ( i = 0; i < height; i++ ) |
616 |
|
|
free( lines[i] ); |
617 |
|
|
free( lines ); |
618 |
|
|
|
619 |
|
|
return true; |
620 |
|
|
|
621 |
|
|
error: |
622 |
|
|
if (verbose) |
623 |
|
|
wxLogError(_("Couldn't load a PNG image - file is corrupted or not enough memory.")); |
624 |
|
|
|
625 |
|
|
if ( image->Ok() ) |
626 |
|
|
{ |
627 |
|
|
image->Destroy(); |
628 |
|
|
} |
629 |
|
|
|
630 |
|
|
if ( lines ) |
631 |
|
|
{ |
632 |
|
|
for ( unsigned int n = 0; n < height; n++ ) |
633 |
|
|
free( lines[n] ); |
634 |
|
|
|
635 |
|
|
free( lines ); |
636 |
|
|
} |
637 |
|
|
|
638 |
|
|
if ( png_ptr ) |
639 |
|
|
{ |
640 |
|
|
if ( info_ptr ) |
641 |
|
|
{ |
642 |
|
|
png_destroy_read_struct( &png_ptr, &info_ptr, (png_infopp) NULL ); |
643 |
|
|
free(info_ptr); |
644 |
|
|
} |
645 |
|
|
else |
646 |
|
|
png_destroy_read_struct( &png_ptr, (png_infopp) NULL, (png_infopp) NULL ); |
647 |
|
|
} |
648 |
|
|
return false; |
649 |
|
|
} |
650 |
|
|
|
651 |
|
|
// ---------------------------------------------------------------------------- |
652 |
|
|
// writing PNGs |
653 |
|
|
// ---------------------------------------------------------------------------- |
654 |
|
|
|
655 |
|
|
bool wxPNGHandler::SaveFile( wxImage *image, wxOutputStream& stream, bool verbose ) |
656 |
|
|
{ |
657 |
|
|
wxPNGInfoStruct wxinfo; |
658 |
|
|
|
659 |
|
|
wxinfo.verbose = verbose; |
660 |
|
|
wxinfo.stream.out = &stream; |
661 |
|
|
|
662 |
|
|
png_structp png_ptr = png_create_write_struct |
663 |
|
|
( |
664 |
|
|
PNG_LIBPNG_VER_STRING, |
665 |
|
|
NULL, |
666 |
|
|
wx_png_error, |
667 |
|
|
wx_png_warning |
668 |
|
|
); |
669 |
|
|
if (!png_ptr) |
670 |
|
|
{ |
671 |
|
|
if (verbose) |
672 |
|
|
wxLogError(_("Couldn't save PNG image.")); |
673 |
|
|
return false; |
674 |
|
|
} |
675 |
|
|
|
676 |
|
|
png_infop info_ptr = png_create_info_struct(png_ptr); |
677 |
|
|
if (info_ptr == NULL) |
678 |
|
|
{ |
679 |
|
|
png_destroy_write_struct( &png_ptr, (png_infopp)NULL ); |
680 |
|
|
if (verbose) |
681 |
|
|
wxLogError(_("Couldn't save PNG image.")); |
682 |
|
|
return false; |
683 |
|
|
} |
684 |
|
|
|
685 |
|
|
if (setjmp(wxinfo.jmpbuf)) |
686 |
|
|
{ |
687 |
|
|
png_destroy_write_struct( &png_ptr, (png_infopp)NULL ); |
688 |
|
|
if (verbose) |
689 |
|
|
wxLogError(_("Couldn't save PNG image.")); |
690 |
|
|
return false; |
691 |
|
|
} |
692 |
|
|
|
693 |
|
|
// NB: please see the comment near wxPNGInfoStruct declaration for |
694 |
|
|
// explanation why this line is mandatory |
695 |
|
|
png_set_write_fn( png_ptr, &wxinfo, wx_PNG_stream_writer, NULL); |
696 |
|
|
|
697 |
|
|
const int iColorType = image->HasOption(wxIMAGE_OPTION_PNG_FORMAT) |
698 |
|
|
? image->GetOptionInt(wxIMAGE_OPTION_PNG_FORMAT) |
699 |
|
|
: wxPNG_TYPE_COLOUR; |
700 |
|
|
const int iBitDepth = image->HasOption(wxIMAGE_OPTION_PNG_BITDEPTH) |
701 |
|
|
? image->GetOptionInt(wxIMAGE_OPTION_PNG_BITDEPTH) |
702 |
|
|
: 8; |
703 |
|
|
|
704 |
|
|
wxASSERT_MSG( iBitDepth == 8 || iBitDepth == 16, |
705 |
|
|
_T("PNG bit depth must be 8 or 16") ); |
706 |
|
|
|
707 |
|
|
bool bHasAlpha = image->HasAlpha(); |
708 |
|
|
bool bHasMask = image->HasMask(); |
709 |
|
|
bool bUseAlpha = bHasAlpha || bHasMask; |
710 |
|
|
|
711 |
|
|
int iPngColorType; |
712 |
|
|
if ( iColorType==wxPNG_TYPE_COLOUR ) |
713 |
|
|
{ |
714 |
|
|
iPngColorType = bUseAlpha ? PNG_COLOR_TYPE_RGB_ALPHA |
715 |
|
|
: PNG_COLOR_TYPE_RGB; |
716 |
|
|
} |
717 |
|
|
else |
718 |
|
|
{ |
719 |
|
|
iPngColorType = bUseAlpha ? PNG_COLOR_TYPE_GRAY_ALPHA |
720 |
|
|
: PNG_COLOR_TYPE_GRAY; |
721 |
|
|
} |
722 |
|
|
|
723 |
|
|
png_set_IHDR( png_ptr, info_ptr, image->GetWidth(), image->GetHeight(), |
724 |
|
|
iBitDepth, iPngColorType, |
725 |
|
|
PNG_INTERLACE_NONE, PNG_COMPRESSION_TYPE_BASE, |
726 |
|
|
PNG_FILTER_TYPE_BASE); |
727 |
|
|
|
728 |
|
|
int iElements; |
729 |
|
|
png_color_8 sig_bit; |
730 |
|
|
|
731 |
|
|
if ( iPngColorType & PNG_COLOR_MASK_COLOR ) |
732 |
|
|
{ |
733 |
|
|
sig_bit.red = |
734 |
|
|
sig_bit.green = |
735 |
|
|
sig_bit.blue = (png_byte)iBitDepth; |
736 |
|
|
iElements = 3; |
737 |
|
|
} |
738 |
|
|
else // grey |
739 |
|
|
{ |
740 |
|
|
sig_bit.gray = (png_byte)iBitDepth; |
741 |
|
|
iElements = 1; |
742 |
|
|
} |
743 |
|
|
|
744 |
|
|
if ( iPngColorType & PNG_COLOR_MASK_ALPHA ) |
745 |
|
|
{ |
746 |
|
|
sig_bit.alpha = (png_byte)iBitDepth; |
747 |
|
|
iElements++; |
748 |
|
|
} |
749 |
|
|
|
750 |
|
|
if ( iBitDepth == 16 ) |
751 |
|
|
iElements *= 2; |
752 |
|
|
|
753 |
|
|
png_set_sBIT( png_ptr, info_ptr, &sig_bit ); |
754 |
|
|
png_write_info( png_ptr, info_ptr ); |
755 |
|
|
png_set_shift( png_ptr, &sig_bit ); |
756 |
|
|
png_set_packing( png_ptr ); |
757 |
|
|
|
758 |
|
|
unsigned char * |
759 |
|
|
data = (unsigned char *)malloc( image->GetWidth() * iElements ); |
760 |
|
|
if ( !data ) |
761 |
|
|
{ |
762 |
|
|
png_destroy_write_struct( &png_ptr, (png_infopp)NULL ); |
763 |
|
|
return false; |
764 |
|
|
} |
765 |
|
|
|
766 |
|
|
unsigned char * |
767 |
|
|
pAlpha = (unsigned char *)(bHasAlpha ? image->GetAlpha() : NULL); |
768 |
|
|
int iHeight = image->GetHeight(); |
769 |
|
|
int iWidth = image->GetWidth(); |
770 |
|
|
|
771 |
|
|
unsigned char uchMaskRed = 0, uchMaskGreen = 0, uchMaskBlue = 0; |
772 |
|
|
|
773 |
|
|
if ( bHasMask ) |
774 |
|
|
{ |
775 |
|
|
uchMaskRed = image->GetMaskRed(); |
776 |
|
|
uchMaskGreen = image->GetMaskGreen(); |
777 |
|
|
uchMaskBlue = image->GetMaskBlue(); |
778 |
|
|
} |
779 |
|
|
|
780 |
|
|
unsigned char *pColors = image->GetData(); |
781 |
|
|
|
782 |
|
|
for (int y = 0; y != iHeight; ++y) |
783 |
|
|
{ |
784 |
|
|
unsigned char *pData = data; |
785 |
|
|
for (int x = 0; x != iWidth; x++) |
786 |
|
|
{ |
787 |
|
|
unsigned char uchRed = *pColors++; |
788 |
|
|
unsigned char uchGreen = *pColors++; |
789 |
|
|
unsigned char uchBlue = *pColors++; |
790 |
|
|
|
791 |
|
|
switch ( iColorType ) |
792 |
|
|
{ |
793 |
|
|
default: |
794 |
|
|
wxFAIL_MSG( _T("unknown wxPNG_TYPE_XXX") ); |
795 |
|
|
// fall through |
796 |
|
|
|
797 |
|
|
case wxPNG_TYPE_COLOUR: |
798 |
|
|
*pData++ = uchRed; |
799 |
|
|
if ( iBitDepth == 16 ) |
800 |
|
|
*pData++ = 0; |
801 |
|
|
*pData++ = uchGreen; |
802 |
|
|
if ( iBitDepth == 16 ) |
803 |
|
|
*pData++ = 0; |
804 |
|
|
*pData++ = uchBlue; |
805 |
|
|
if ( iBitDepth == 16 ) |
806 |
|
|
*pData++ = 0; |
807 |
|
|
break; |
808 |
|
|
|
809 |
|
|
case wxPNG_TYPE_GREY: |
810 |
|
|
{ |
811 |
|
|
// where do these coefficients come from? maybe we |
812 |
|
|
// should have image options for them as well? |
813 |
|
|
unsigned uiColor = |
814 |
|
|
(unsigned) (76.544*(unsigned)uchRed + |
815 |
|
|
150.272*(unsigned)uchGreen + |
816 |
|
|
36.864*(unsigned)uchBlue); |
817 |
|
|
|
818 |
|
|
*pData++ = (unsigned char)((uiColor >> 8) & 0xFF); |
819 |
|
|
if ( iBitDepth == 16 ) |
820 |
|
|
*pData++ = (unsigned char)(uiColor & 0xFF); |
821 |
|
|
} |
822 |
|
|
break; |
823 |
|
|
|
824 |
|
|
case wxPNG_TYPE_GREY_RED: |
825 |
|
|
*pData++ = uchRed; |
826 |
|
|
if ( iBitDepth == 16 ) |
827 |
|
|
*pData++ = 0; |
828 |
|
|
break; |
829 |
|
|
} |
830 |
|
|
|
831 |
|
|
if ( bUseAlpha ) |
832 |
|
|
{ |
833 |
|
|
unsigned char uchAlpha = 255; |
834 |
|
|
if ( bHasAlpha ) |
835 |
|
|
uchAlpha = *pAlpha++; |
836 |
|
|
|
837 |
|
|
if ( bHasMask ) |
838 |
|
|
{ |
839 |
|
|
if ( (uchRed == uchMaskRed) |
840 |
|
|
&& (uchGreen == uchMaskGreen) |
841 |
|
|
&& (uchBlue == uchMaskBlue) ) |
842 |
|
|
uchAlpha = 0; |
843 |
|
|
} |
844 |
|
|
|
845 |
|
|
*pData++ = uchAlpha; |
846 |
|
|
if ( iBitDepth == 16 ) |
847 |
|
|
*pData++ = 0; |
848 |
|
|
} |
849 |
|
|
} |
850 |
|
|
|
851 |
|
|
png_bytep row_ptr = data; |
852 |
|
|
png_write_rows( png_ptr, &row_ptr, 1 ); |
853 |
|
|
} |
854 |
|
|
|
855 |
|
|
free(data); |
856 |
|
|
png_write_end( png_ptr, info_ptr ); |
857 |
|
|
png_destroy_write_struct( &png_ptr, (png_infopp)&info_ptr ); |
858 |
|
|
|
859 |
|
|
return true; |
860 |
|
|
} |
861 |
|
|
|
862 |
|
|
#ifdef __VISUALC__ |
863 |
|
|
#pragma warning(default:4611) |
864 |
|
|
#endif /* VC++ */ |
865 |
|
|
|
866 |
|
|
#endif // wxUSE_STREAMS |
867 |
|
|
|
868 |
|
|
#endif // wxUSE_LIBPNG |