]> git.saurik.com Git - wxWidgets.git/blame - src/common/imagpnm.cpp
added find performance test (see #9870) and the possibility to set the number of...
[wxWidgets.git] / src / common / imagpnm.cpp
CommitLineData
a8d9809f 1/////////////////////////////////////////////////////////////////////////////
9b5f1895 2// Name: src/common/imagpnm.cpp
a8d9809f
SB
3// Purpose: wxImage PNM handler
4// Author: Sylvain Bougnoux
5// RCS-ID: $Id$
6// Copyright: (c) Sylvain Bougnoux
65571936 7// Licence: wxWindows licence
a8d9809f
SB
8/////////////////////////////////////////////////////////////////////////////
9
a8d9809f
SB
10// For compilers that support precompilation, includes "wx.h".
11#include "wx/wxprec.h"
12
13#ifdef __BORLANDC__
88a7a4e1 14 #pragma hdrstop
b9b32d5c
GRG
15#endif
16
c96ea657 17#if wxUSE_IMAGE && wxUSE_PNM
b9b32d5c 18
8f493002 19#include "wx/imagpnm.h"
88a7a4e1
WS
20
21#ifndef WX_PRECOMP
22 #include "wx/intl.h"
e4db172a 23 #include "wx/log.h"
88a7a4e1
WS
24#endif
25
a8d9809f
SB
26#include "wx/txtstrm.h"
27
a8d9809f
SB
28//-----------------------------------------------------------------------------
29// wxBMPHandler
30//-----------------------------------------------------------------------------
31
a8d9809f 32IMPLEMENT_DYNAMIC_CLASS(wxPNMHandler,wxImageHandler)
a8d9809f 33
9ab6ee85 34#if wxUSE_STREAMS
a8d9809f
SB
35
36void Skip_Comment(wxInputStream &stream)
37{
2b5f62a0 38 wxTextInputStream text_stream(stream);
a8d9809f 39
2b5f62a0 40 if (stream.Peek()==wxT('#'))
a8d9809f 41 {
2b5f62a0
VZ
42 text_stream.ReadLine();
43 Skip_Comment(stream);
a8d9809f
SB
44 }
45}
46
58c837a4 47bool wxPNMHandler::LoadFile( wxImage *image, wxInputStream& stream, bool verbose, int WXUNUSED(index) )
a8d9809f
SB
48{
49 wxUint32 width, height;
50 wxUint16 maxval;
a8d9809f 51 char c(0);
995612e2 52
a8d9809f
SB
53 image->Destroy();
54
55 /*
56 * Read the PNM header
57 */
58
6319afe3
GL
59 wxBufferedInputStream buf_stream(stream);
60 wxTextInputStream text_stream(buf_stream);
a8d9809f 61
6319afe3 62 Skip_Comment(buf_stream);
223d09f6 63 if (buf_stream.GetC()==wxT('P')) c=buf_stream.GetC();
a8d9809f
SB
64
65 switch (c)
58c837a4 66 {
7448de8d 67 case wxT('2'): // ASCII Grey
3e8711ce
JS
68 case wxT('3'): // ASCII RGB
69 case wxT('5'): // RAW Grey
7beb59f3 70 case wxT('6'): break;
58c837a4
RR
71 default:
72 if (verbose) wxLogError(_("PNM: File format is not recognized."));
7beb59f3 73 return false;
58c837a4 74 }
a8d9809f 75
41d1f82d 76 text_stream.ReadLine(); // for the \n
6319afe3 77 Skip_Comment(buf_stream);
a8d9809f 78 text_stream >> width >> height ;
995612e2 79 Skip_Comment(buf_stream);
a8d9809f
SB
80 text_stream >> maxval;
81
a2b8bd55 82 //cout << line << " " << width << " " << height << " " << maxval << endl;
a8d9809f
SB
83 image->Create( width, height );
84 unsigned char *ptr = image->GetData();
85 if (!ptr)
86 {
58c837a4
RR
87 if (verbose)
88 wxLogError( _("PNM: Couldn't allocate memory.") );
7beb59f3 89 return false;
a8d9809f
SB
90 }
91
3e8711ce 92
7448de8d
WS
93 if (c=='2') // Ascii GREY
94 {
3e8711ce
JS
95 wxUint32 value, size=width*height;
96 for (wxUint32 i=0; i<size; ++i)
97 {
98 value=text_stream.Read32();
57beadd2
VZ
99 if ( maxval != 255 )
100 value = (255 * value)/maxval;
3e8711ce 101 *ptr++=(unsigned char)value; // R
7448de8d 102 *ptr++=(unsigned char)value; // G
3e8711ce
JS
103 *ptr++=(unsigned char)value; // B
104 if ( !buf_stream )
105 {
106 if (verbose) wxLogError(_("PNM: File seems truncated."));
107 return false;
108 }
109 }
110 }
7448de8d
WS
111 if (c=='3') // Ascii RBG
112 {
995612e2
VZ
113 wxUint32 value, size=3*width*height;
114 for (wxUint32 i=0; i<size; ++i)
115 {
116 //this is very slow !!!
117 //I wonder how we can make any better ?
118 value=text_stream.Read32();
57beadd2
VZ
119 if ( maxval != 255 )
120 value = (255 * value)/maxval;
995612e2
VZ
121 *ptr++=(unsigned char)value;
122
2b5f62a0 123 if ( !buf_stream )
995612e2 124 {
58c837a4 125 if (verbose) wxLogError(_("PNM: File seems truncated."));
7beb59f3 126 return false;
995612e2
VZ
127 }
128 }
7448de8d
WS
129 }
130 if (c=='5') // Raw GREY
131 {
3e8711ce
JS
132 wxUint32 size=width*height;
133 unsigned char value;
134 for (wxUint32 i=0; i<size; ++i)
135 {
136 buf_stream.Read(&value,1);
57beadd2
VZ
137 if ( maxval != 255 )
138 value = (255 * value)/maxval;
3e8711ce
JS
139 *ptr++=value; // R
140 *ptr++=value; // G
7448de8d 141 *ptr++=value; // B
3e8711ce
JS
142 if ( !buf_stream )
143 {
144 if (verbose) wxLogError(_("PNM: File seems truncated."));
145 return false;
146 }
147 }
148 }
57beadd2
VZ
149
150 if ( c=='6' ) // Raw RGB
151 {
152 buf_stream.Read(ptr, 3*width*height);
153 if ( maxval != 255 )
154 {
155 for ( unsigned i = 0; i < 3*width*height; i++ )
156 ptr[i] = (255 * ptr[i])/maxval;
157 }
158 }
a8d9809f 159
7beb59f3 160 image->SetMask( false );
a8d9809f 161
2b5f62a0
VZ
162 const wxStreamError err = buf_stream.GetLastError();
163 return err == wxSTREAM_NO_ERROR || err == wxSTREAM_EOF;
a8d9809f
SB
164}
165
166bool wxPNMHandler::SaveFile( wxImage *image, wxOutputStream& stream, bool WXUNUSED(verbose) )
167{
168 wxTextOutputStream text_stream(stream);
995612e2
VZ
169
170 //text_stream << "P6" << endl
171 //<< image->GetWidth() << " " << image->GetHeight() << endl
a8d9809f 172 //<< "255" << endl;
2b5f62a0 173 text_stream << wxT("P6\n") << image->GetWidth() << wxT(" ") << image->GetHeight() << wxT("\n255\n");
a8d9809f
SB
174 stream.Write(image->GetData(),3*image->GetWidth()*image->GetHeight());
175
2b5f62a0 176 return stream.IsOk();
a8d9809f
SB
177}
178
995612e2 179bool wxPNMHandler::DoCanRead( wxInputStream& stream )
0828c087 180{
93dfff5a
SB
181 Skip_Comment(stream);
182
995612e2
VZ
183 if ( stream.GetC() == 'P' )
184 {
cdf0ff0f 185 switch ( stream.GetC() )
995612e2 186 {
cdf0ff0f
VZ
187 case '2': // ASCII Grey
188 case '3': // ASCII RGB
189 case '5': // RAW Grey
190 case '6': // RAW RGB
7beb59f3 191 return true;
995612e2
VZ
192 }
193 }
93dfff5a 194
7beb59f3 195 return false;
0828c087
VS
196}
197
198
9ab6ee85 199#endif // wxUSE_STREAMS
a8d9809f 200
e4db172a 201#endif // wxUSE_IMAGE && wxUSE_PNM