1 ///////////////////////////////////////////////////////////////////////////// 
   3 // Purpose:     wxImage class 
   4 // Author:      Robert Roebling 
   6 // Copyright:   (c) Robert Roebling 
   7 // Licence:     wxWindows licence 
   8 ///////////////////////////////////////////////////////////////////////////// 
  17 #include "wx/object.h" 
  18 #include "wx/string.h" 
  19 #include "wx/gdicmn.h" 
  20 #include "wx/hashmap.h" 
  23 #  include "wx/stream.h" 
  26 // on some systems (Unixware 7.x) index is defined as a macro in the headers 
  27 // which breaks the compilation below 
  30 #define wxIMAGE_OPTION_QUALITY  wxString(_T("quality")) 
  31 #define wxIMAGE_OPTION_FILENAME wxString(_T("FileName")) 
  33 #define wxIMAGE_OPTION_RESOLUTION            wxString(_T("Resolution")) 
  34 #define wxIMAGE_OPTION_RESOLUTIONX           wxString(_T("ResolutionX")) 
  35 #define wxIMAGE_OPTION_RESOLUTIONY           wxString(_T("ResolutionY")) 
  37 #define wxIMAGE_OPTION_RESOLUTIONUNIT        wxString(_T("ResolutionUnit")) 
  39 // constants used with wxIMAGE_OPTION_RESOLUTIONUNIT 
  41 // NB: don't change these values, they correspond to libjpeg constants 
  42 enum wxImageResolution
 
  44     // Resolution not specified 
  45     wxIMAGE_RESOLUTION_NONE 
= 0, 
  47     // Resolution specified in inches 
  48     wxIMAGE_RESOLUTION_INCHES 
= 1, 
  50     // Resolution specified in centimeters 
  51     wxIMAGE_RESOLUTION_CM 
= 2 
  54 // Constants for wxImage::Scale() for determining the level of quality 
  57     wxIMAGE_QUALITY_NORMAL 
= 0, 
  58     wxIMAGE_QUALITY_HIGH 
= 1 
  61 // alpha channel values: fully transparent, default threshold separating 
  62 // transparent pixels from opaque for a few functions dealing with alpha and 
  64 const unsigned char wxIMAGE_ALPHA_TRANSPARENT 
= 0; 
  65 const unsigned char wxIMAGE_ALPHA_THRESHOLD 
= 0x80; 
  66 const unsigned char wxIMAGE_ALPHA_OPAQUE 
= 0xff; 
  68 //----------------------------------------------------------------------------- 
  70 //----------------------------------------------------------------------------- 
  72 class WXDLLIMPEXP_FWD_CORE wxImageHandler
; 
  73 class WXDLLIMPEXP_FWD_CORE wxImage
; 
  74 class WXDLLIMPEXP_FWD_CORE wxPalette
; 
  76 //----------------------------------------------------------------------------- 
  78 //----------------------------------------------------------------------------- 
  81 #include "wx/variant.h" 
  82 DECLARE_VARIANT_OBJECT_EXPORTED(wxImage
,WXDLLEXPORT
) 
  85 //----------------------------------------------------------------------------- 
  87 //----------------------------------------------------------------------------- 
  89 class WXDLLEXPORT wxImageHandler
: public wxObject
 
  93         : m_name(wxEmptyString
), m_extension(wxEmptyString
), m_mime(), m_type(0) 
  97     virtual bool LoadFile( wxImage 
*image
, wxInputStream
& stream
, bool verbose
=true, int index
=-1 ); 
  98     virtual bool SaveFile( wxImage 
*image
, wxOutputStream
& stream
, bool verbose
=true ); 
 100     virtual int GetImageCount( wxInputStream
& stream 
); 
 102     bool CanRead( wxInputStream
& stream 
) { return CallDoCanRead(stream
); } 
 103     bool CanRead( const wxString
& name 
); 
 104 #endif // wxUSE_STREAMS 
 106     void SetName(const wxString
& name
) { m_name 
= name
; } 
 107     void SetExtension(const wxString
& ext
) { m_extension 
= ext
; } 
 108     void SetType(long type
) { m_type 
= type
; } 
 109     void SetMimeType(const wxString
& type
) { m_mime 
= type
; } 
 110     const wxString
& GetName() const { return m_name
; } 
 111     const wxString
& GetExtension() const { return m_extension
; } 
 112     long GetType() const { return m_type
; } 
 113     const wxString
& GetMimeType() const { return m_mime
; } 
 117     virtual bool DoCanRead( wxInputStream
& stream 
) = 0; 
 119     // save the stream position, call DoCanRead() and restore the position 
 120     bool CallDoCanRead(wxInputStream
& stream
); 
 121 #endif // wxUSE_STREAMS 
 123     // helper for the derived classes SaveFile() implementations: returns the 
 124     // values of x- and y-resolution options specified as the image options if 
 126     static wxImageResolution
 
 127     GetResolutionFromOptions(const wxImage
& image
, int *x
, int *y
); 
 131     wxString  m_extension
; 
 136     DECLARE_CLASS(wxImageHandler
) 
 139 //----------------------------------------------------------------------------- 
 141 //----------------------------------------------------------------------------- 
 143 class WXDLLEXPORT wxImageHistogramEntry
 
 146     wxImageHistogramEntry() { index 
= value 
= 0; } 
 151 WX_DECLARE_EXPORTED_HASH_MAP(unsigned long, wxImageHistogramEntry
, 
 152                              wxIntegerHash
, wxIntegerEqual
, 
 153                              wxImageHistogramBase
); 
 155 class WXDLLEXPORT wxImageHistogram 
: public wxImageHistogramBase
 
 158     wxImageHistogram() : wxImageHistogramBase(256) { } 
 160     // get the key in the histogram for the given RGB values 
 161     static unsigned long MakeKey(unsigned char r
, 
 165         return (r 
<< 16) | (g 
<< 8) | b
; 
 168     // find first colour that is not used in the image and has higher 
 169     // RGB values than RGB(startR, startG, startB) 
 171     // returns true and puts this colour in r, g, b (each of which may be NULL) 
 172     // on success or returns false if there are no more free colours 
 173     bool FindFirstUnusedColour(unsigned char *r
, 
 176                                unsigned char startR 
= 1, 
 177                                unsigned char startG 
= 0, 
 178                                unsigned char startB 
= 0 ) const; 
 181 //----------------------------------------------------------------------------- 
 183 //----------------------------------------------------------------------------- 
 185 class WXDLLEXPORT wxImage
: public wxObject
 
 188     // red, green and blue are 8 bit unsigned integers in the range of 0..255 
 189     // We use the identifier RGBValue instead of RGB, since RGB is #defined 
 193       RGBValue(unsigned char r
=0, unsigned char g
=0, unsigned char b
=0) 
 194         : red(r
), green(g
), blue(b
) {} 
 200     // hue, saturation and value are doubles in the range 0.0..1.0 
 204         HSVValue(double h
=0.0, double s
=0.0, double v
=0.0) 
 205             : hue(h
), saturation(s
), value(v
) {} 
 212     wxImage( int width
, int height
, bool clear 
= true ); 
 213     wxImage( int width
, int height
, unsigned char* data
, bool static_data 
= false ); 
 214     wxImage( int width
, int height
, unsigned char* data
, unsigned char* alpha
, bool static_data 
= false ); 
 215     wxImage( const wxString
& name
, long type 
= wxBITMAP_TYPE_ANY
, int index 
= -1 ); 
 216     wxImage( const wxString
& name
, const wxString
& mimetype
, int index 
= -1 ); 
 217     wxImage( const char* const* xpmData 
); 
 220     wxImage( wxInputStream
& stream
, long type 
= wxBITMAP_TYPE_ANY
, int index 
= -1 ); 
 221     wxImage( wxInputStream
& stream
, const wxString
& mimetype
, int index 
= -1 ); 
 222 #endif // wxUSE_STREAMS 
 224     bool Create( int width
, int height
, bool clear 
= true ); 
 225     bool Create( int width
, int height
, unsigned char* data
, bool static_data 
= false ); 
 226     bool Create( int width
, int height
, unsigned char* data
, unsigned char* alpha
, bool static_data 
= false ); 
 227     bool Create( const char* const* xpmData 
); 
 229     // needed for Borland 5.5 
 230     wxImage( char** xpmData 
) { Create(wx_const_cast(const char* const*, xpmData
)); } 
 231     bool Create( char** xpmData 
) { return Create(wx_const_cast(const char* const*, xpmData
)); } 
 235     // creates an identical copy of the image (the = operator 
 236     // just raises the ref count) 
 237     wxImage 
Copy() const; 
 239     // return the new image with size width*height 
 240     wxImage 
GetSubImage( const wxRect
& rect
) const; 
 242     // Paste the image or part of this image into an image of the given size at the pos 
 243     //  any newly exposed areas will be filled with the rgb colour 
 244     //  by default if r = g = b = -1 then fill with this image's mask colour or find and 
 245     //  set a suitable mask colour 
 246     wxImage 
Size( const wxSize
& size
, const wxPoint
& pos
, 
 247                   int r 
= -1, int g 
= -1, int b 
= -1 ) const; 
 249     // pastes image into this instance and takes care of 
 250     // the mask colour and out of bounds problems 
 251     void Paste( const wxImage 
&image
, int x
, int y 
); 
 253     // return the new image with size width*height 
 254     wxImage 
Scale( int width
, int height
, int quality 
= wxIMAGE_QUALITY_NORMAL 
) const; 
 256     // box averager and bicubic filters for up/down sampling 
 257     wxImage 
ResampleBox(int width
, int height
) const; 
 258     wxImage 
ResampleBicubic(int width
, int height
) const; 
 260     // blur the image according to the specified pixel radius 
 261     wxImage 
Blur(int radius
); 
 262     wxImage 
BlurHorizontal(int radius
); 
 263     wxImage 
BlurVertical(int radius
); 
 265     wxImage 
ShrinkBy( int xFactor 
, int yFactor 
) const ; 
 267     // rescales the image in place 
 268     wxImage
& Rescale( int width
, int height
, int quality 
= wxIMAGE_QUALITY_NORMAL 
) { return *this = Scale(width
, height
, quality
); } 
 270     // resizes the image in place 
 271     wxImage
& Resize( const wxSize
& size
, const wxPoint
& pos
, 
 272                      int r 
= -1, int g 
= -1, int b 
= -1 ) { return *this = Size(size
, pos
, r
, g
, b
); } 
 274     // Rotates the image about the given point, 'angle' radians. 
 275     // Returns the rotated image, leaving this image intact. 
 276     wxImage 
Rotate(double angle
, const wxPoint 
& centre_of_rotation
, 
 277                    bool interpolating 
= true, wxPoint 
* offset_after_rotation 
= (wxPoint
*) NULL
) const; 
 279     wxImage 
Rotate90( bool clockwise 
= true ) const; 
 280     wxImage 
Mirror( bool horizontally 
= true ) const; 
 282     // replace one colour with another 
 283     void Replace( unsigned char r1
, unsigned char g1
, unsigned char b1
, 
 284                   unsigned char r2
, unsigned char g2
, unsigned char b2 
); 
 286     // Convert to greyscale image. Uses the luminance component (Y) of the image. 
 287     // The luma value (YUV) is calculated using (R * lr) + (G * lg) + (B * lb), defaults to ITU-T BT.601 
 288     wxImage 
ConvertToGreyscale( double lr 
= 0.299, double lg 
= 0.587, double lb 
= 0.114 ) const; 
 290     // convert to monochrome image (<r,g,b> will be replaced by white, 
 291     // everything else by black) 
 292     wxImage 
ConvertToMono( unsigned char r
, unsigned char g
, unsigned char b 
) const; 
 294     // these routines are slow but safe 
 295     void SetRGB( int x
, int y
, unsigned char r
, unsigned char g
, unsigned char b 
); 
 296     void SetRGB( const wxRect
& rect
, unsigned char r
, unsigned char g
, unsigned char b 
); 
 297     unsigned char GetRed( int x
, int y 
) const; 
 298     unsigned char GetGreen( int x
, int y 
) const; 
 299     unsigned char GetBlue( int x
, int y 
) const; 
 301     void SetAlpha(int x
, int y
, unsigned char alpha
); 
 302     unsigned char GetAlpha(int x
, int y
) const; 
 304     // find first colour that is not used in the image and has higher 
 305     // RGB values than <startR,startG,startB> 
 306     bool FindFirstUnusedColour( unsigned char *r
, unsigned char *g
, unsigned char *b
, 
 307                                 unsigned char startR 
= 1, unsigned char startG 
= 0, 
 308                                 unsigned char startB 
= 0 ) const; 
 309     // Set image's mask to the area of 'mask' that has <r,g,b> colour 
 310     bool SetMaskFromImage(const wxImage 
& mask
, 
 311                           unsigned char mr
, unsigned char mg
, unsigned char mb
); 
 313     // converts image's alpha channel to mask, if it has any, does nothing 
 315     bool ConvertAlphaToMask(unsigned char threshold 
= wxIMAGE_ALPHA_THRESHOLD
); 
 317     // This method converts an image where the original alpha 
 318     // information is only available as a shades of a colour 
 319     // (actually shades of grey) typically when you draw anti- 
 320     // aliased text into a bitmap. The DC drawinf routines 
 321     // draw grey values on the black background although they 
 322     // actually mean to draw white with differnt alpha values. 
 323     // This method reverses it, assuming a black (!) background 
 324     // and white text (actually only the red channel is read). 
 325     // The method will then fill up the whole image with the 
 327     bool ConvertColourToAlpha( unsigned char r
, unsigned char g
, unsigned char b 
); 
 329     static bool CanRead( const wxString
& name 
); 
 330     static int GetImageCount( const wxString
& name
, long type 
= wxBITMAP_TYPE_ANY 
); 
 331     virtual bool LoadFile( const wxString
& name
, long type 
= wxBITMAP_TYPE_ANY
, int index 
= -1 ); 
 332     virtual bool LoadFile( const wxString
& name
, const wxString
& mimetype
, int index 
= -1 ); 
 335     static bool CanRead( wxInputStream
& stream 
); 
 336     static int GetImageCount( wxInputStream
& stream
, long type 
= wxBITMAP_TYPE_ANY 
); 
 337     virtual bool LoadFile( wxInputStream
& stream
, long type 
= wxBITMAP_TYPE_ANY
, int index 
= -1 ); 
 338     virtual bool LoadFile( wxInputStream
& stream
, const wxString
& mimetype
, int index 
= -1 ); 
 341     virtual bool SaveFile( const wxString
& name 
) const; 
 342     virtual bool SaveFile( const wxString
& name
, int type 
) const; 
 343     virtual bool SaveFile( const wxString
& name
, const wxString
& mimetype 
) const; 
 346     virtual bool SaveFile( wxOutputStream
& stream
, int type 
) const; 
 347     virtual bool SaveFile( wxOutputStream
& stream
, const wxString
& mimetype 
) const; 
 350     bool Ok() const { return IsOk(); } 
 352     int GetWidth() const; 
 353     int GetHeight() const; 
 355     // these functions provide fastest access to wxImage data but should be 
 356     // used carefully as no checks are done 
 357     unsigned char *GetData() const; 
 358     void SetData( unsigned char *data
, bool static_data
=false ); 
 359     void SetData( unsigned char *data
, int new_width
, int new_height
, bool static_data
=false ); 
 361     unsigned char *GetAlpha() const;    // may return NULL! 
 362     bool HasAlpha() const { return GetAlpha() != NULL
; } 
 363     void SetAlpha(unsigned char *alpha 
= NULL
, bool static_data
=false); 
 366     // return true if this pixel is masked or has alpha less than specified 
 368     bool IsTransparent(int x
, int y
, 
 369                        unsigned char threshold 
= wxIMAGE_ALPHA_THRESHOLD
) const; 
 372     void SetMaskColour( unsigned char r
, unsigned char g
, unsigned char b 
); 
 373     // Get the current mask colour or find a suitable colour 
 374     // returns true if using current mask colour 
 375     bool GetOrFindMaskColour( unsigned char *r
, unsigned char *g
, unsigned char *b 
) const; 
 376     unsigned char GetMaskRed() const; 
 377     unsigned char GetMaskGreen() const; 
 378     unsigned char GetMaskBlue() const; 
 379     void SetMask( bool mask 
= true ); 
 380     bool HasMask() const; 
 384     bool HasPalette() const; 
 385     const wxPalette
& GetPalette() const; 
 386     void SetPalette(const wxPalette
& palette
); 
 387 #endif // wxUSE_PALETTE 
 389     // Option functions (arbitrary name/value mapping) 
 390     void SetOption(const wxString
& name
, const wxString
& value
); 
 391     void SetOption(const wxString
& name
, int value
); 
 392     wxString 
GetOption(const wxString
& name
) const; 
 393     int GetOptionInt(const wxString
& name
) const; 
 394     bool HasOption(const wxString
& name
) const; 
 396     unsigned long CountColours( unsigned long stopafter 
= (unsigned long) -1 ) const; 
 398     // Computes the histogram of the image and fills a hash table, indexed 
 399     // with integer keys built as 0xRRGGBB, containing wxImageHistogramEntry 
 400     // objects. Each of them contains an 'index' (useful to build a palette 
 401     // with the image colours) and a 'value', which is the number of pixels 
 402     // in the image with that colour. 
 403     // Returned value: # of entries in the histogram 
 404     unsigned long ComputeHistogram( wxImageHistogram 
&h 
) const; 
 406     // Rotates the hue of each pixel of the image. angle is a double in the range 
 407     // -1.0..1.0 where -1.0 is -360 degrees and 1.0 is 360 degrees 
 408     void RotateHue(double angle
); 
 410     static wxList
& GetHandlers() { return sm_handlers
; } 
 411     static void AddHandler( wxImageHandler 
*handler 
); 
 412     static void InsertHandler( wxImageHandler 
*handler 
); 
 413     static bool RemoveHandler( const wxString
& name 
); 
 414     static wxImageHandler 
*FindHandler( const wxString
& name 
); 
 415     static wxImageHandler 
*FindHandler( const wxString
& extension
, long imageType 
); 
 416     static wxImageHandler 
*FindHandler( long imageType 
); 
 417     static wxImageHandler 
*FindHandlerMime( const wxString
& mimetype 
); 
 419     static wxString 
GetImageExtWildcard(); 
 421     static void CleanUpHandlers(); 
 422     static void InitStandardHandlers(); 
 424     static HSVValue 
RGBtoHSV(const RGBValue
& rgb
); 
 425     static RGBValue 
HSVtoRGB(const HSVValue
& hsv
); 
 429     static wxList   sm_handlers
; 
 431     // return the index of the point with the given coordinates or -1 if the 
 432     // image is invalid of the coordinates are out of range 
 434     // note that index must be multiplied by 3 when using it with RGB array 
 435     long XYToIndex(int x
, int y
) const; 
 437     virtual wxObjectRefData
* CreateRefData() const; 
 438     virtual wxObjectRefData
* CloneRefData(const wxObjectRefData
* data
) const; 
 441     friend class WXDLLIMPEXP_FWD_CORE wxImageHandler
; 
 443     DECLARE_DYNAMIC_CLASS(wxImage
) 
 447 extern void WXDLLEXPORT 
wxInitAllImageHandlers(); 
 449 extern WXDLLEXPORT_DATA(wxImage
)    wxNullImage
; 
 451 //----------------------------------------------------------------------------- 
 453 //----------------------------------------------------------------------------- 
 455 #include "wx/imagbmp.h" 
 456 #include "wx/imagpng.h" 
 457 #include "wx/imaggif.h" 
 458 #include "wx/imagpcx.h" 
 459 #include "wx/imagjpeg.h" 
 460 #include "wx/imagtga.h" 
 461 #include "wx/imagtiff.h" 
 462 #include "wx/imagpnm.h" 
 463 #include "wx/imagxpm.h" 
 464 #include "wx/imagiff.h" 
 466 #endif // wxUSE_IMAGE