-rw-r--r-- | noncore/graphics/opie-eye/slave/png_slave.cpp | 3 |
1 files changed, 2 insertions, 1 deletions
diff --git a/noncore/graphics/opie-eye/slave/png_slave.cpp b/noncore/graphics/opie-eye/slave/png_slave.cpp index 86e1cdc..21555b4 100644 --- a/noncore/graphics/opie-eye/slave/png_slave.cpp +++ b/noncore/graphics/opie-eye/slave/png_slave.cpp | |||
@@ -1,211 +1,212 @@ | |||
1 | #include "png_slave.h" | 1 | #include "png_slave.h" |
2 | 2 | ||
3 | #include "thumbnailtool.h" | 3 | #include "thumbnailtool.h" |
4 | 4 | ||
5 | #include <qobject.h> | 5 | #include <qobject.h> |
6 | #include <qfile.h> | 6 | #include <qfile.h> |
7 | #include <qimage.h> | 7 | #include <qimage.h> |
8 | #include <qpixmap.h> | 8 | #include <qpixmap.h> |
9 | #include <qstring.h> | 9 | #include <qstring.h> |
10 | 10 | ||
11 | /* | 11 | /* |
12 | * GPLv2 from kfile plugin | 12 | * GPLv2 from kfile plugin |
13 | */ | 13 | */ |
14 | PHUNK_VIEW_INTERFACE( "PNG", PNGSlave ); | 14 | PHUNK_VIEW_INTERFACE( "PNG", PNGSlave ); |
15 | 15 | ||
16 | #define CHUNK_SIZE(data, index) ((data[index ]<<24) + (data[index+1]<<16) + \ | 16 | #define CHUNK_SIZE(data, index) ((data[index ]<<24) + (data[index+1]<<16) + \ |
17 | (data[index+2]<< 8) + data[index+3]) | 17 | (data[index+2]<< 8) + data[index+3]) |
18 | #define CHUNK_TYPE(data, index) &data[index+4] | 18 | #define CHUNK_TYPE(data, index) &data[index+4] |
19 | #define CHUNK_HEADER_SIZE 12 | 19 | #define CHUNK_HEADER_SIZE 12 |
20 | #define CHUNK_DATA(data, index, offset) data[8+index+offset] | 20 | #define CHUNK_DATA(data, index, offset) data[8+index+offset] |
21 | 21 | ||
22 | /* TRANSLATOR QObject */ | 22 | /* TRANSLATOR QObject */ |
23 | 23 | ||
24 | // known translations for common png keys | 24 | // known translations for common png keys |
25 | static const char* knownTranslations[] | 25 | static const char* knownTranslations[] |
26 | #ifdef __GNUC__ | 26 | #ifdef __GNUC__ |
27 | __attribute__((unused)) | 27 | __attribute__((unused)) |
28 | #endif | 28 | #endif |
29 | = { | 29 | = { |
30 | QT_TR_NOOP("Title"), | 30 | QT_TR_NOOP("Title"), |
31 | QT_TR_NOOP("Author"), | 31 | QT_TR_NOOP("Author"), |
32 | QT_TR_NOOP("Description"), | 32 | QT_TR_NOOP("Description"), |
33 | QT_TR_NOOP("Copyright"), | 33 | QT_TR_NOOP("Copyright"), |
34 | QT_TR_NOOP("Creation Time"), | 34 | QT_TR_NOOP("Creation Time"), |
35 | QT_TR_NOOP("Software"), | 35 | QT_TR_NOOP("Software"), |
36 | QT_TR_NOOP("Disclaimer"), | 36 | QT_TR_NOOP("Disclaimer"), |
37 | QT_TR_NOOP("Warning"), | 37 | QT_TR_NOOP("Warning"), |
38 | QT_TR_NOOP("Source"), | 38 | QT_TR_NOOP("Source"), |
39 | QT_TR_NOOP("Comment") | 39 | QT_TR_NOOP("Comment") |
40 | }; | 40 | }; |
41 | 41 | ||
42 | // and for the colors | 42 | // and for the colors |
43 | static const char* colors[] = { | 43 | static const char* colors[] = { |
44 | QT_TR_NOOP("Grayscale"), | 44 | QT_TR_NOOP("Grayscale"), |
45 | QT_TR_NOOP("Unknown"), | 45 | QT_TR_NOOP("Unknown"), |
46 | QT_TR_NOOP("RGB"), | 46 | QT_TR_NOOP("RGB"), |
47 | QT_TR_NOOP("Palette"), | 47 | QT_TR_NOOP("Palette"), |
48 | QT_TR_NOOP("Grayscale/Alpha"), | 48 | QT_TR_NOOP("Grayscale/Alpha"), |
49 | QT_TR_NOOP("Unknown"), | 49 | QT_TR_NOOP("Unknown"), |
50 | QT_TR_NOOP("RGB/Alpha") | 50 | QT_TR_NOOP("RGB/Alpha") |
51 | }; | 51 | }; |
52 | 52 | ||
53 | // and compressions | 53 | // and compressions |
54 | static const char* compressions[] = | 54 | static const char* compressions[] = |
55 | { | 55 | { |
56 | QT_TR_NOOP("Deflate") | 56 | QT_TR_NOOP("Deflate") |
57 | }; | 57 | }; |
58 | 58 | ||
59 | // interlaced modes | 59 | // interlaced modes |
60 | static const char* interlaceModes[] = { | 60 | static const char* interlaceModes[] = { |
61 | QT_TR_NOOP("None"), | 61 | QT_TR_NOOP("None"), |
62 | QT_TR_NOOP("Adam7") | 62 | QT_TR_NOOP("Adam7") |
63 | }; | 63 | }; |
64 | 64 | ||
65 | 65 | ||
66 | static void read_comment( const QString& inf, | 66 | static void read_comment( const QString& inf, |
67 | bool readComments, QString& str ) { | 67 | bool readComments, QString& str ) { |
68 | QFile f(inf); | 68 | QFile f(inf); |
69 | f.open(IO_ReadOnly); | 69 | if (!f.exists()) return; |
70 | if (!f.open(IO_ReadOnly)) return; | ||
70 | 71 | ||
71 | if (f.size() < 26) return; | 72 | if (f.size() < 26) return; |
72 | // the technical group will be read from the first 26 bytes. If the file | 73 | // the technical group will be read from the first 26 bytes. If the file |
73 | // is smaller, we can't even read this. | 74 | // is smaller, we can't even read this. |
74 | 75 | ||
75 | uchar *data = new uchar[f.size()+1]; | 76 | uchar *data = new uchar[f.size()+1]; |
76 | f.readBlock(reinterpret_cast<char*>(data), f.size()); | 77 | f.readBlock(reinterpret_cast<char*>(data), f.size()); |
77 | data[f.size()]='\n'; | 78 | data[f.size()]='\n'; |
78 | 79 | ||
79 | // find the start | 80 | // find the start |
80 | if (data[0] == 137 && data[1] == 80 && data[2] == 78 && data[3] == 71 && | 81 | if (data[0] == 137 && data[1] == 80 && data[2] == 78 && data[3] == 71 && |
81 | data[4] == 13 && data[5] == 10 && data[6] == 26 && data[7] == 10 ) | 82 | data[4] == 13 && data[5] == 10 && data[6] == 26 && data[7] == 10 ) |
82 | { | 83 | { |
83 | // ok | 84 | // ok |
84 | // the IHDR chunk should be the first | 85 | // the IHDR chunk should be the first |
85 | if (!strncmp((char*)&data[12], "IHDR", 4)) | 86 | if (!strncmp((char*)&data[12], "IHDR", 4)) |
86 | { | 87 | { |
87 | // we found it, get the dimensions | 88 | // we found it, get the dimensions |
88 | ulong x,y; | 89 | ulong x,y; |
89 | x = (data[16]<<24) + (data[17]<<16) + (data[18]<<8) + data[19]; | 90 | x = (data[16]<<24) + (data[17]<<16) + (data[18]<<8) + data[19]; |
90 | y = (data[20]<<24) + (data[21]<<16) + (data[22]<<8) + data[23]; | 91 | y = (data[20]<<24) + (data[21]<<16) + (data[22]<<8) + data[23]; |
91 | 92 | ||
92 | uint type = data[25]; | 93 | uint type = data[25]; |
93 | uint bpp = data[24]; | 94 | uint bpp = data[24]; |
94 | 95 | ||
95 | // the bpp are only per channel, so we need to multiply the with | 96 | // the bpp are only per channel, so we need to multiply the with |
96 | // the channel count | 97 | // the channel count |
97 | switch (type) | 98 | switch (type) |
98 | { | 99 | { |
99 | case 0: break; // Grayscale | 100 | case 0: break; // Grayscale |
100 | case 2: bpp *= 3; break; // RGB | 101 | case 2: bpp *= 3; break; // RGB |
101 | case 3: break; // palette | 102 | case 3: break; // palette |
102 | case 4: bpp *= 2; break; // grayscale w. alpha | 103 | case 4: bpp *= 2; break; // grayscale w. alpha |
103 | case 6: bpp *= 4; break; // RGBA | 104 | case 6: bpp *= 4; break; // RGBA |
104 | 105 | ||
105 | default: // we don't get any sensible value here | 106 | default: // we don't get any sensible value here |
106 | bpp = 0; | 107 | bpp = 0; |
107 | } | 108 | } |
108 | 109 | ||
109 | 110 | ||
110 | str = QObject::tr("Dimensions: %1x%2\n" ).arg(x).arg(y); | 111 | str = QObject::tr("Dimensions: %1x%2\n" ).arg(x).arg(y); |
111 | str += QObject::tr("Depth: %1\n" ).arg(bpp); | 112 | str += QObject::tr("Depth: %1\n" ).arg(bpp); |
112 | str += QObject::tr("ColorMode: %1\n").arg( | 113 | str += QObject::tr("ColorMode: %1\n").arg( |
113 | (type < sizeof(colors)/sizeof(colors[0])) | 114 | (type < sizeof(colors)/sizeof(colors[0])) |
114 | ? QObject::tr(colors[data[25]]) : QObject::tr("Unknown") ); | 115 | ? QObject::tr(colors[data[25]]) : QObject::tr("Unknown") ); |
115 | 116 | ||
116 | str += QObject::tr("Compression: %1\n").arg( | 117 | str += QObject::tr("Compression: %1\n").arg( |
117 | (data[26] < sizeof(compressions)/sizeof(compressions[0])) | 118 | (data[26] < sizeof(compressions)/sizeof(compressions[0])) |
118 | ? QObject::tr(compressions[data[26]]) : QObject::tr("Unknown") ); | 119 | ? QObject::tr(compressions[data[26]]) : QObject::tr("Unknown") ); |
119 | 120 | ||
120 | str += QObject::tr("InterlaceMode: %1\n" ).arg( | 121 | str += QObject::tr("InterlaceMode: %1\n" ).arg( |
121 | (data[28] < sizeof(interlaceModes)/sizeof(interlaceModes[0])) | 122 | (data[28] < sizeof(interlaceModes)/sizeof(interlaceModes[0])) |
122 | ? QObject::tr(interlaceModes[data[28]]) : QObject::tr("Unknown")); | 123 | ? QObject::tr(interlaceModes[data[28]]) : QObject::tr("Unknown")); |
123 | } | 124 | } |
124 | 125 | ||
125 | if ( readComments ) { | 126 | if ( readComments ) { |
126 | uint index = 8; | 127 | uint index = 8; |
127 | index += CHUNK_SIZE(data, index) + CHUNK_HEADER_SIZE; | 128 | index += CHUNK_SIZE(data, index) + CHUNK_HEADER_SIZE; |
128 | 129 | ||
129 | while(index<f.size()-12) | 130 | while(index<f.size()-12) |
130 | { | 131 | { |
131 | while (index < f.size() - 12 && | 132 | while (index < f.size() - 12 && |
132 | strncmp((char*)CHUNK_TYPE(data,index), "tEXt", 4)) | 133 | strncmp((char*)CHUNK_TYPE(data,index), "tEXt", 4)) |
133 | { | 134 | { |
134 | if (!strncmp((char*)CHUNK_TYPE(data,index), "IEND", 4)) | 135 | if (!strncmp((char*)CHUNK_TYPE(data,index), "IEND", 4)) |
135 | goto end; | 136 | goto end; |
136 | 137 | ||
137 | index += CHUNK_SIZE(data, index) + CHUNK_HEADER_SIZE; | 138 | index += CHUNK_SIZE(data, index) + CHUNK_HEADER_SIZE; |
138 | } | 139 | } |
139 | 140 | ||
140 | if (index < f.size() - 12) | 141 | if (index < f.size() - 12) |
141 | { | 142 | { |
142 | // we found a tEXt field | 143 | // we found a tEXt field |
143 | // get the key, it's a null terminated string at the | 144 | // get the key, it's a null terminated string at the |
144 | // chunk start | 145 | // chunk start |
145 | 146 | ||
146 | uchar* key = &CHUNK_DATA(data,index,0); | 147 | uchar* key = &CHUNK_DATA(data,index,0); |
147 | 148 | ||
148 | int keysize=0; | 149 | int keysize=0; |
149 | for (;key[keysize]!=0; keysize++) | 150 | for (;key[keysize]!=0; keysize++) |
150 | // look if we reached the end of the file | 151 | // look if we reached the end of the file |
151 | // (it might be corrupted) | 152 | // (it might be corrupted) |
152 | if (8+index+keysize>=f.size()) | 153 | if (8+index+keysize>=f.size()) |
153 | goto end; | 154 | goto end; |
154 | 155 | ||
155 | // the text comes after the key, but isn't null terminated | 156 | // the text comes after the key, but isn't null terminated |
156 | uchar* text = &CHUNK_DATA(data,index, keysize+1); | 157 | uchar* text = &CHUNK_DATA(data,index, keysize+1); |
157 | uint textsize = CHUNK_SIZE(data, index)-keysize-1; | 158 | uint textsize = CHUNK_SIZE(data, index)-keysize-1; |
158 | 159 | ||
159 | // security check, also considering overflow wraparound from the addition -- | 160 | // security check, also considering overflow wraparound from the addition -- |
160 | // we may endup with a /smaller/ index if we wrap all the way around | 161 | // we may endup with a /smaller/ index if we wrap all the way around |
161 | uint firstIndex = (uint)(text - data); | 162 | uint firstIndex = (uint)(text - data); |
162 | uint onePastLastIndex = firstIndex + textsize; | 163 | uint onePastLastIndex = firstIndex + textsize; |
163 | 164 | ||
164 | if ( onePastLastIndex > f.size() || onePastLastIndex <= firstIndex) | 165 | if ( onePastLastIndex > f.size() || onePastLastIndex <= firstIndex) |
165 | goto end; | 166 | goto end; |
166 | 167 | ||
167 | QByteArray arr(textsize); | 168 | QByteArray arr(textsize); |
168 | arr = QByteArray(textsize).duplicate((const char*)text, | 169 | arr = QByteArray(textsize).duplicate((const char*)text, |
169 | textsize); | 170 | textsize); |
170 | str += QObject::tr( | 171 | str += QObject::tr( |
171 | QString(reinterpret_cast<char*>(key)), | 172 | QString(reinterpret_cast<char*>(key)), |
172 | QString(arr) ); | 173 | QString(arr) ); |
173 | 174 | ||
174 | index += CHUNK_SIZE(data, index) + CHUNK_HEADER_SIZE; | 175 | index += CHUNK_SIZE(data, index) + CHUNK_HEADER_SIZE; |
175 | } | 176 | } |
176 | } | 177 | } |
177 | } | 178 | } |
178 | } | 179 | } |
179 | end: | 180 | end: |
180 | delete[] data; | 181 | delete[] data; |
181 | 182 | ||
182 | } | 183 | } |
183 | 184 | ||
184 | 185 | ||
185 | PNGSlave::PNGSlave() | 186 | PNGSlave::PNGSlave() |
186 | : SlaveInterface("png") | 187 | : SlaveInterface("png") |
187 | { | 188 | { |
188 | } | 189 | } |
189 | PNGSlave::~PNGSlave() { | 190 | PNGSlave::~PNGSlave() { |
190 | } | 191 | } |
191 | QString PNGSlave::iconViewName( const QString& path) { | 192 | QString PNGSlave::iconViewName( const QString& path) { |
192 | QString str; | 193 | QString str; |
193 | read_comment( path, false, str ); | 194 | read_comment( path, false, str ); |
194 | return str; | 195 | return str; |
195 | } | 196 | } |
196 | 197 | ||
197 | QString PNGSlave::fullImageInfo( const QString& path) { | 198 | QString PNGSlave::fullImageInfo( const QString& path) { |
198 | QString str = "<qt>"; | 199 | QString str = "<qt>"; |
199 | read_comment( path, true, str ); | 200 | read_comment( path, true, str ); |
200 | str += "</qt>"; | 201 | str += "</qt>"; |
201 | return str; | 202 | return str; |
202 | } | 203 | } |
203 | 204 | ||
204 | 205 | ||
205 | QPixmap PNGSlave::pixmap( const QString& path, int width, int height) { | 206 | QPixmap PNGSlave::pixmap( const QString& path, int width, int height) { |
206 | QImage img; img.load( path ); | 207 | QImage img; img.load( path ); |
207 | if ( img.isNull() ) | 208 | if ( img.isNull() ) |
208 | return QPixmap(); | 209 | return QPixmap(); |
209 | else | 210 | else |
210 | return ThumbNailTool::scaleImage( img, width,height ); | 211 | return ThumbNailTool::scaleImage( img, width,height ); |
211 | } | 212 | } |