author | Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com> |
Tue, 02 Feb 2010 00:43:10 +0200 | |
changeset 3 | 41300fa6a67c |
parent 0 | 1918ee327afb |
child 4 | 3b1da2848fc7 |
permissions | -rw-r--r-- |
0 | 1 |
/**************************************************************************** |
2 |
** |
|
3 |
** Copyright (C) 2009 Nokia Corporation and/or its subsidiary(-ies). |
|
4 |
** All rights reserved. |
|
5 |
** Contact: Nokia Corporation (qt-info@nokia.com) |
|
6 |
** |
|
7 |
** This file is part of the plugins of the Qt Toolkit. |
|
8 |
** |
|
9 |
** $QT_BEGIN_LICENSE:LGPL$ |
|
10 |
** No Commercial Usage |
|
11 |
** This file contains pre-release code and may not be distributed. |
|
12 |
** You may use this file in accordance with the terms and conditions |
|
13 |
** contained in the Technology Preview License Agreement accompanying |
|
14 |
** this package. |
|
15 |
** |
|
16 |
** GNU Lesser General Public License Usage |
|
17 |
** Alternatively, this file may be used under the terms of the GNU Lesser |
|
18 |
** General Public License version 2.1 as published by the Free Software |
|
19 |
** Foundation and appearing in the file LICENSE.LGPL included in the |
|
20 |
** packaging of this file. Please review the following information to |
|
21 |
** ensure the GNU Lesser General Public License version 2.1 requirements |
|
22 |
** will be met: http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html. |
|
23 |
** |
|
24 |
** In addition, as a special exception, Nokia gives you certain additional |
|
25 |
** rights. These rights are described in the Nokia Qt LGPL Exception |
|
26 |
** version 1.1, included in the file LGPL_EXCEPTION.txt in this package. |
|
27 |
** |
|
28 |
** If you have questions regarding the use of this file, please contact |
|
29 |
** Nokia at qt-info@nokia.com. |
|
30 |
** |
|
31 |
** |
|
32 |
** |
|
33 |
** |
|
34 |
** |
|
35 |
** |
|
36 |
** |
|
37 |
** |
|
38 |
** $QT_END_LICENSE$ |
|
39 |
** |
|
40 |
****************************************************************************/ |
|
41 |
||
42 |
#include "qtiffhandler.h" |
|
43 |
#include <qvariant.h> |
|
44 |
#include <qdebug.h> |
|
45 |
#include <qimage.h> |
|
46 |
#include <qglobal.h> |
|
47 |
extern "C" { |
|
48 |
#include "tiffio.h" |
|
49 |
} |
|
50 |
||
51 |
QT_BEGIN_NAMESPACE |
|
52 |
||
53 |
tsize_t qtiffReadProc(thandle_t fd, tdata_t buf, tsize_t size) |
|
54 |
{ |
|
55 |
QIODevice* device = static_cast<QTiffHandler*>(fd)->device(); |
|
56 |
return device->isReadable() ? device->read(static_cast<char *>(buf), size) : -1; |
|
57 |
} |
|
58 |
||
59 |
tsize_t qtiffWriteProc(thandle_t fd, tdata_t buf, tsize_t size) |
|
60 |
{ |
|
61 |
return static_cast<QTiffHandler*>(fd)->device()->write(static_cast<char *>(buf), size); |
|
62 |
} |
|
63 |
||
64 |
toff_t qtiffSeekProc(thandle_t fd, toff_t off, int whence) |
|
65 |
{ |
|
66 |
QIODevice *device = static_cast<QTiffHandler*>(fd)->device(); |
|
67 |
switch (whence) { |
|
68 |
case SEEK_SET: |
|
69 |
device->seek(off); |
|
70 |
break; |
|
71 |
case SEEK_CUR: |
|
72 |
device->seek(device->pos() + off); |
|
73 |
break; |
|
74 |
case SEEK_END: |
|
75 |
device->seek(device->size() + off); |
|
76 |
break; |
|
77 |
} |
|
78 |
||
79 |
return device->pos(); |
|
80 |
} |
|
81 |
||
82 |
int qtiffCloseProc(thandle_t /*fd*/) |
|
83 |
{ |
|
84 |
return 0; |
|
85 |
} |
|
86 |
||
87 |
toff_t qtiffSizeProc(thandle_t fd) |
|
88 |
{ |
|
89 |
return static_cast<QTiffHandler*>(fd)->device()->size(); |
|
90 |
} |
|
91 |
||
92 |
int qtiffMapProc(thandle_t /*fd*/, tdata_t* /*pbase*/, toff_t* /*psize*/) |
|
93 |
{ |
|
94 |
return 0; |
|
95 |
} |
|
96 |
||
97 |
void qtiffUnmapProc(thandle_t /*fd*/, tdata_t /*base*/, toff_t /*size*/) |
|
98 |
{ |
|
99 |
} |
|
100 |
||
101 |
// for 32 bits images |
|
102 |
inline void rotate_right_mirror_horizontal(QImage *const image)// rotate right->mirrored horizontal |
|
103 |
{ |
|
104 |
const int height = image->height(); |
|
105 |
const int width = image->width(); |
|
106 |
QImage generated(/* width = */ height, /* height = */ width, image->format()); |
|
107 |
const uint32 *originalPixel = reinterpret_cast<const uint32*>(image->bits()); |
|
108 |
uint32 *const generatedPixels = reinterpret_cast<uint32*>(generated.bits()); |
|
109 |
for (int row=0; row < height; ++row) { |
|
110 |
for (int col=0; col < width; ++col) { |
|
111 |
int idx = col * height + row; |
|
112 |
generatedPixels[idx] = *originalPixel; |
|
113 |
++originalPixel; |
|
114 |
} |
|
115 |
} |
|
116 |
*image = generated; |
|
117 |
} |
|
118 |
||
119 |
inline void rotate_right_mirror_vertical(QImage *const image) // rotate right->mirrored vertical |
|
120 |
{ |
|
121 |
const int height = image->height(); |
|
122 |
const int width = image->width(); |
|
123 |
QImage generated(/* width = */ height, /* height = */ width, image->format()); |
|
124 |
const int lastCol = width - 1; |
|
125 |
const int lastRow = height - 1; |
|
126 |
const uint32 *pixel = reinterpret_cast<const uint32*>(image->bits()); |
|
127 |
uint32 *const generatedBits = reinterpret_cast<uint32*>(generated.bits()); |
|
128 |
for (int row=0; row < height; ++row) { |
|
129 |
for (int col=0; col < width; ++col) { |
|
130 |
int idx = (lastCol - col) * height + (lastRow - row); |
|
131 |
generatedBits[idx] = *pixel; |
|
132 |
++pixel; |
|
133 |
} |
|
134 |
} |
|
135 |
*image = generated; |
|
136 |
} |
|
137 |
||
138 |
QTiffHandler::QTiffHandler() : QImageIOHandler() |
|
139 |
{ |
|
140 |
compression = NoCompression; |
|
141 |
} |
|
142 |
||
143 |
bool QTiffHandler::canRead() const |
|
144 |
{ |
|
145 |
if (canRead(device())) { |
|
146 |
setFormat("tiff"); |
|
147 |
return true; |
|
148 |
} |
|
149 |
return false; |
|
150 |
} |
|
151 |
||
152 |
bool QTiffHandler::canRead(QIODevice *device) |
|
153 |
{ |
|
154 |
if (!device) { |
|
155 |
qWarning("QTiffHandler::canRead() called with no device"); |
|
156 |
return false; |
|
157 |
} |
|
158 |
||
159 |
// current implementation uses TIFFClientOpen which needs to be |
|
160 |
// able to seek, so sequential devices are not supported |
|
161 |
QByteArray header = device->peek(4); |
|
162 |
return header == QByteArray::fromRawData("\x49\x49\x2A\x00", 4) |
|
163 |
|| header == QByteArray::fromRawData("\x4D\x4D\x00\x2A", 4); |
|
164 |
} |
|
165 |
||
166 |
bool QTiffHandler::read(QImage *image) |
|
167 |
{ |
|
168 |
if (!canRead()) |
|
169 |
return false; |
|
170 |
||
171 |
TIFF *const tiff = TIFFClientOpen("foo", |
|
172 |
"r", |
|
173 |
this, |
|
174 |
qtiffReadProc, |
|
175 |
qtiffWriteProc, |
|
176 |
qtiffSeekProc, |
|
177 |
qtiffCloseProc, |
|
178 |
qtiffSizeProc, |
|
179 |
qtiffMapProc, |
|
180 |
qtiffUnmapProc); |
|
181 |
||
182 |
if (!tiff) { |
|
183 |
return false; |
|
184 |
} |
|
185 |
uint32 width; |
|
186 |
uint32 height; |
|
187 |
uint16 photometric; |
|
188 |
if (!TIFFGetField(tiff, TIFFTAG_IMAGEWIDTH, &width) |
|
189 |
|| !TIFFGetField(tiff, TIFFTAG_IMAGELENGTH, &height) |
|
190 |
|| !TIFFGetField(tiff, TIFFTAG_PHOTOMETRIC, &photometric)) { |
|
191 |
TIFFClose(tiff); |
|
192 |
return false; |
|
193 |
} |
|
194 |
||
3
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
195 |
uint16 bitPerSample; |
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
196 |
if (!TIFFGetField(tiff, TIFFTAG_BITSPERSAMPLE, &bitPerSample)) { |
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
197 |
TIFFClose(tiff); |
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
198 |
return false; |
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
199 |
} |
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
200 |
|
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
201 |
bool grayscale = photometric == PHOTOMETRIC_MINISBLACK || photometric == PHOTOMETRIC_MINISWHITE; |
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
202 |
if (grayscale && bitPerSample == 1) { |
0 | 203 |
if (image->size() != QSize(width, height) || image->format() != QImage::Format_Mono) |
204 |
*image = QImage(width, height, QImage::Format_Mono); |
|
205 |
QVector<QRgb> colortable(2); |
|
206 |
if (photometric == PHOTOMETRIC_MINISBLACK) { |
|
207 |
colortable[0] = 0xff000000; |
|
208 |
colortable[1] = 0xffffffff; |
|
209 |
} else { |
|
210 |
colortable[0] = 0xffffffff; |
|
211 |
colortable[1] = 0xff000000; |
|
212 |
} |
|
213 |
image->setColorTable(colortable); |
|
214 |
||
215 |
if (!image->isNull()) { |
|
216 |
for (uint32 y=0; y<height; ++y) { |
|
217 |
if (TIFFReadScanline(tiff, image->scanLine(y), y, 0) < 0) { |
|
3
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
218 |
TIFFClose(tiff); |
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
219 |
return false; |
0 | 220 |
} |
221 |
} |
|
222 |
} |
|
223 |
} else { |
|
3
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
224 |
if ((grayscale || photometric == PHOTOMETRIC_PALETTE) && bitPerSample == 8) { |
0 | 225 |
if (image->size() != QSize(width, height) || image->format() != QImage::Format_Indexed8) |
226 |
*image = QImage(width, height, QImage::Format_Indexed8); |
|
227 |
if (!image->isNull()) { |
|
228 |
const uint16 tableSize = 256; |
|
3
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
229 |
QVector<QRgb> qtColorTable(tableSize); |
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
230 |
if (grayscale) { |
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
231 |
for (int i = 0; i<tableSize; ++i) { |
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
232 |
const int c = (photometric == PHOTOMETRIC_MINISBLACK) ? i : (255 - i); |
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
233 |
qtColorTable[i] = qRgb(c, c, c); |
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
234 |
} |
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
235 |
} else { |
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
236 |
// create the color table |
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
237 |
uint16 *redTable = static_cast<uint16 *>(qMalloc(tableSize * sizeof(uint16))); |
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
238 |
uint16 *greenTable = static_cast<uint16 *>(qMalloc(tableSize * sizeof(uint16))); |
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
239 |
uint16 *blueTable = static_cast<uint16 *>(qMalloc(tableSize * sizeof(uint16))); |
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
240 |
if (!redTable || !greenTable || !blueTable) { |
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
241 |
TIFFClose(tiff); |
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
242 |
return false; |
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
243 |
} |
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
244 |
if (!TIFFGetField(tiff, TIFFTAG_COLORMAP, &redTable, &greenTable, &blueTable)) { |
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
245 |
TIFFClose(tiff); |
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
246 |
return false; |
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
247 |
} |
0 | 248 |
|
3
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
249 |
for (int i = 0; i<tableSize ;++i) { |
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
250 |
const int red = redTable[i] / 257; |
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
251 |
const int green = greenTable[i] / 257; |
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
252 |
const int blue = blueTable[i] / 257; |
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
253 |
qtColorTable[i] = qRgb(red, green, blue); |
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
254 |
} |
0 | 255 |
} |
256 |
||
257 |
image->setColorTable(qtColorTable); |
|
258 |
for (uint32 y=0; y<height; ++y) { |
|
259 |
if (TIFFReadScanline(tiff, image->scanLine(y), y, 0) < 0) { |
|
260 |
TIFFClose(tiff); |
|
261 |
return false; |
|
262 |
} |
|
263 |
} |
|
264 |
||
265 |
// free redTable, greenTable and greenTable done by libtiff |
|
266 |
} |
|
267 |
} else { |
|
268 |
if (image->size() != QSize(width, height) || image->format() != QImage::Format_ARGB32) |
|
269 |
*image = QImage(width, height, QImage::Format_ARGB32); |
|
270 |
if (!image->isNull()) { |
|
271 |
const int stopOnError = 1; |
|
272 |
if (TIFFReadRGBAImageOriented(tiff, width, height, reinterpret_cast<uint32 *>(image->bits()), ORIENTATION_TOPLEFT, stopOnError)) { |
|
273 |
for (uint32 y=0; y<height; ++y) |
|
274 |
convert32BitOrder(image->scanLine(y), width); |
|
275 |
} else { |
|
276 |
TIFFClose(tiff); |
|
277 |
return false; |
|
278 |
} |
|
279 |
} |
|
280 |
} |
|
281 |
} |
|
282 |
||
283 |
if (image->isNull()) { |
|
284 |
TIFFClose(tiff); |
|
285 |
return false; |
|
286 |
} |
|
287 |
||
288 |
float resX = 0; |
|
289 |
float resY = 0; |
|
290 |
uint16 resUnit = RESUNIT_NONE; |
|
291 |
if (TIFFGetField(tiff, TIFFTAG_RESOLUTIONUNIT, &resUnit) |
|
292 |
&& TIFFGetField(tiff, TIFFTAG_XRESOLUTION, &resX) |
|
293 |
&& TIFFGetField(tiff, TIFFTAG_YRESOLUTION, &resY)) { |
|
294 |
||
295 |
switch(resUnit) { |
|
296 |
case RESUNIT_CENTIMETER: |
|
297 |
image->setDotsPerMeterX(qRound(resX * 100)); |
|
298 |
image->setDotsPerMeterY(qRound(resY * 100)); |
|
299 |
break; |
|
300 |
case RESUNIT_INCH: |
|
301 |
image->setDotsPerMeterX(qRound(resX * (100 / 2.54))); |
|
302 |
image->setDotsPerMeterY(qRound(resY * (100 / 2.54))); |
|
303 |
break; |
|
304 |
default: |
|
305 |
// do nothing as defaults have already |
|
306 |
// been set within the QImage class |
|
307 |
break; |
|
308 |
} |
|
309 |
} |
|
310 |
||
311 |
// rotate the image if the orientation is defined in the file |
|
312 |
uint16 orientationTag; |
|
313 |
if (TIFFGetField(tiff, TIFFTAG_ORIENTATION, &orientationTag)) { |
|
314 |
if (image->format() == QImage::Format_ARGB32) { |
|
315 |
// TIFFReadRGBAImageOriented() flip the image but does not rotate them |
|
316 |
switch (orientationTag) { |
|
317 |
case 5: |
|
318 |
rotate_right_mirror_horizontal(image); |
|
319 |
break; |
|
320 |
case 6: |
|
321 |
rotate_right_mirror_vertical(image); |
|
322 |
break; |
|
323 |
case 7: |
|
324 |
rotate_right_mirror_horizontal(image); |
|
325 |
break; |
|
326 |
case 8: |
|
327 |
rotate_right_mirror_vertical(image); |
|
328 |
break; |
|
329 |
} |
|
330 |
} else { |
|
331 |
switch (orientationTag) { |
|
332 |
case 1: // default orientation |
|
333 |
break; |
|
334 |
case 2: // mirror horizontal |
|
335 |
*image = image->mirrored(true, false); |
|
336 |
break; |
|
337 |
case 3: // mirror both |
|
338 |
*image = image->mirrored(true, true); |
|
339 |
break; |
|
340 |
case 4: // mirror vertical |
|
341 |
*image = image->mirrored(false, true); |
|
342 |
break; |
|
343 |
case 5: // rotate right mirror horizontal |
|
344 |
{ |
|
345 |
QMatrix transformation; |
|
346 |
transformation.rotate(90); |
|
347 |
*image = image->transformed(transformation); |
|
348 |
*image = image->mirrored(true, false); |
|
349 |
break; |
|
350 |
} |
|
351 |
case 6: // rotate right |
|
352 |
{ |
|
353 |
QMatrix transformation; |
|
354 |
transformation.rotate(90); |
|
355 |
*image = image->transformed(transformation); |
|
356 |
break; |
|
357 |
} |
|
358 |
case 7: // rotate right, mirror vertical |
|
359 |
{ |
|
360 |
QMatrix transformation; |
|
361 |
transformation.rotate(90); |
|
362 |
*image = image->transformed(transformation); |
|
363 |
*image = image->mirrored(false, true); |
|
364 |
break; |
|
365 |
} |
|
366 |
case 8: // rotate left |
|
367 |
{ |
|
368 |
QMatrix transformation; |
|
369 |
transformation.rotate(270); |
|
370 |
*image = image->transformed(transformation); |
|
371 |
break; |
|
372 |
} |
|
373 |
} |
|
374 |
} |
|
375 |
} |
|
376 |
||
377 |
||
378 |
TIFFClose(tiff); |
|
379 |
return true; |
|
380 |
} |
|
381 |
||
3
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
382 |
static bool checkGrayscale(const QVector<QRgb> &colorTable) |
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
383 |
{ |
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
384 |
if (colorTable.size() != 256) |
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
385 |
return false; |
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
386 |
|
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
387 |
const bool increasing = (colorTable.at(0) == 0xff000000); |
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
388 |
for (int i = 0; i < 256; ++i) { |
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
389 |
if (increasing && colorTable.at(i) != qRgb(i, i, i) |
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
390 |
|| !increasing && colorTable.at(i) != qRgb(255 - i, 255 - i, 255 - i)) |
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
391 |
return false; |
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
392 |
} |
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
393 |
return true; |
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
394 |
} |
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
395 |
|
0 | 396 |
bool QTiffHandler::write(const QImage &image) |
397 |
{ |
|
398 |
if (!device()->isWritable()) |
|
399 |
return false; |
|
400 |
||
401 |
TIFF *const tiff = TIFFClientOpen("foo", |
|
402 |
"w", |
|
403 |
this, |
|
404 |
qtiffReadProc, |
|
405 |
qtiffWriteProc, |
|
406 |
qtiffSeekProc, |
|
407 |
qtiffCloseProc, |
|
408 |
qtiffSizeProc, |
|
409 |
qtiffMapProc, |
|
410 |
qtiffUnmapProc); |
|
411 |
if (!tiff) |
|
412 |
return false; |
|
413 |
||
414 |
const int width = image.width(); |
|
415 |
const int height = image.height(); |
|
416 |
||
417 |
if (!TIFFSetField(tiff, TIFFTAG_IMAGEWIDTH, width) |
|
418 |
|| !TIFFSetField(tiff, TIFFTAG_IMAGELENGTH, height) |
|
419 |
|| !TIFFSetField(tiff, TIFFTAG_PLANARCONFIG, PLANARCONFIG_CONTIG)) { |
|
420 |
TIFFClose(tiff); |
|
421 |
return false; |
|
422 |
} |
|
423 |
||
424 |
// set the resolution |
|
425 |
bool resolutionSet = false; |
|
426 |
const int dotPerMeterX = image.dotsPerMeterX(); |
|
427 |
const int dotPerMeterY = image.dotsPerMeterY(); |
|
428 |
if ((dotPerMeterX % 100) == 0 |
|
429 |
&& (dotPerMeterY % 100) == 0) { |
|
430 |
resolutionSet = TIFFSetField(tiff, TIFFTAG_RESOLUTIONUNIT, RESUNIT_CENTIMETER) |
|
431 |
&& TIFFSetField(tiff, TIFFTAG_XRESOLUTION, dotPerMeterX/100.0) |
|
432 |
&& TIFFSetField(tiff, TIFFTAG_YRESOLUTION, dotPerMeterY/100.0); |
|
433 |
} else { |
|
434 |
resolutionSet = TIFFSetField(tiff, TIFFTAG_RESOLUTIONUNIT, RESUNIT_INCH) |
|
435 |
&& TIFFSetField(tiff, TIFFTAG_XRESOLUTION, static_cast<float>(image.logicalDpiX())) |
|
436 |
&& TIFFSetField(tiff, TIFFTAG_YRESOLUTION, static_cast<float>(image.logicalDpiY())); |
|
437 |
} |
|
438 |
if (!resolutionSet) { |
|
439 |
TIFFClose(tiff); |
|
440 |
return false; |
|
441 |
} |
|
442 |
||
443 |
// configure image depth |
|
444 |
const QImage::Format format = image.format(); |
|
445 |
if (format == QImage::Format_Mono || format == QImage::Format_MonoLSB) { |
|
446 |
uint16 photometric = PHOTOMETRIC_MINISBLACK; |
|
447 |
if (image.colorTable().at(0) == 0xffffffff) |
|
448 |
photometric = PHOTOMETRIC_MINISWHITE; |
|
449 |
if (!TIFFSetField(tiff, TIFFTAG_PHOTOMETRIC, photometric) |
|
3
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
450 |
|| !TIFFSetField(tiff, TIFFTAG_COMPRESSION, compression == NoCompression ? COMPRESSION_NONE : COMPRESSION_CCITTRLE) |
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
451 |
|| !TIFFSetField(tiff, TIFFTAG_BITSPERSAMPLE, 1)) { |
0 | 452 |
TIFFClose(tiff); |
453 |
return false; |
|
454 |
} |
|
455 |
||
456 |
// try to do the conversion in chunks no greater than 16 MB |
|
457 |
int chunks = (width * height / (1024 * 1024 * 16)) + 1; |
|
458 |
int chunkHeight = qMax(height / chunks, 1); |
|
459 |
||
460 |
int y = 0; |
|
461 |
while (y < height) { |
|
462 |
QImage chunk = image.copy(0, y, width, qMin(chunkHeight, height - y)).convertToFormat(QImage::Format_Mono); |
|
463 |
||
464 |
int chunkStart = y; |
|
465 |
int chunkEnd = y + chunk.height(); |
|
466 |
while (y < chunkEnd) { |
|
467 |
if (TIFFWriteScanline(tiff, reinterpret_cast<uint32 *>(chunk.scanLine(y - chunkStart)), y) != 1) { |
|
468 |
TIFFClose(tiff); |
|
469 |
return false; |
|
470 |
} |
|
471 |
++y; |
|
472 |
} |
|
473 |
} |
|
474 |
TIFFClose(tiff); |
|
475 |
} else if (format == QImage::Format_Indexed8) { |
|
476 |
const QVector<QRgb> colorTable = image.colorTable(); |
|
3
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
477 |
bool isGrayscale = checkGrayscale(colorTable); |
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
478 |
if (isGrayscale) { |
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
479 |
uint16 photometric = PHOTOMETRIC_MINISBLACK; |
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
480 |
if (image.colorTable().at(0) == 0xffffffff) |
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
481 |
photometric = PHOTOMETRIC_MINISWHITE; |
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
482 |
if (!TIFFSetField(tiff, TIFFTAG_PHOTOMETRIC, photometric) |
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
483 |
|| !TIFFSetField(tiff, TIFFTAG_COMPRESSION, compression == NoCompression ? COMPRESSION_NONE : COMPRESSION_PACKBITS) |
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
484 |
|| !TIFFSetField(tiff, TIFFTAG_BITSPERSAMPLE, 8)) { |
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
485 |
TIFFClose(tiff); |
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
486 |
return false; |
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
487 |
} |
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
488 |
} else { |
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
489 |
if (!TIFFSetField(tiff, TIFFTAG_PHOTOMETRIC, PHOTOMETRIC_PALETTE) |
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
490 |
|| !TIFFSetField(tiff, TIFFTAG_COMPRESSION, compression == NoCompression ? COMPRESSION_NONE : COMPRESSION_PACKBITS) |
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
491 |
|| !TIFFSetField(tiff, TIFFTAG_BITSPERSAMPLE, 8)) { |
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
492 |
TIFFClose(tiff); |
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
493 |
return false; |
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
494 |
} |
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
495 |
//// write the color table |
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
496 |
// allocate the color tables |
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
497 |
uint16 *redTable = static_cast<uint16 *>(qMalloc(256 * sizeof(uint16))); |
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
498 |
uint16 *greenTable = static_cast<uint16 *>(qMalloc(256 * sizeof(uint16))); |
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
499 |
uint16 *blueTable = static_cast<uint16 *>(qMalloc(256 * sizeof(uint16))); |
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
500 |
if (!redTable || !greenTable || !blueTable) { |
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
501 |
TIFFClose(tiff); |
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
502 |
return false; |
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
503 |
} |
0 | 504 |
|
3
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
505 |
// set the color table |
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
506 |
const int tableSize = colorTable.size(); |
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
507 |
Q_ASSERT(tableSize <= 256); |
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
508 |
for (int i = 0; i<tableSize; ++i) { |
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
509 |
const QRgb color = colorTable.at(i); |
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
510 |
redTable[i] = qRed(color) * 257; |
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
511 |
greenTable[i] = qGreen(color) * 257; |
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
512 |
blueTable[i] = qBlue(color) * 257; |
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
513 |
} |
0 | 514 |
|
3
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
515 |
const bool setColorTableSuccess = TIFFSetField(tiff, TIFFTAG_COLORMAP, redTable, greenTable, blueTable); |
0 | 516 |
|
3
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
517 |
qFree(redTable); |
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
518 |
qFree(greenTable); |
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
519 |
qFree(blueTable); |
0 | 520 |
|
3
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
521 |
if (!setColorTableSuccess) { |
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
522 |
TIFFClose(tiff); |
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
523 |
return false; |
41300fa6a67c
Revision: 201003
Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
parents:
0
diff
changeset
|
524 |
} |
0 | 525 |
} |
526 |
||
527 |
//// write the data |
|
528 |
// try to do the conversion in chunks no greater than 16 MB |
|
529 |
int chunks = (width * height/ (1024 * 1024 * 16)) + 1; |
|
530 |
int chunkHeight = qMax(height / chunks, 1); |
|
531 |
||
532 |
int y = 0; |
|
533 |
while (y < height) { |
|
534 |
QImage chunk = image.copy(0, y, width, qMin(chunkHeight, height - y)); |
|
535 |
||
536 |
int chunkStart = y; |
|
537 |
int chunkEnd = y + chunk.height(); |
|
538 |
while (y < chunkEnd) { |
|
539 |
if (TIFFWriteScanline(tiff, reinterpret_cast<uint32 *>(chunk.scanLine(y - chunkStart)), y) != 1) { |
|
540 |
TIFFClose(tiff); |
|
541 |
return false; |
|
542 |
} |
|
543 |
++y; |
|
544 |
} |
|
545 |
} |
|
546 |
TIFFClose(tiff); |
|
547 |
||
548 |
} else { |
|
549 |
if (!TIFFSetField(tiff, TIFFTAG_PHOTOMETRIC, PHOTOMETRIC_RGB) |
|
550 |
|| !TIFFSetField(tiff, TIFFTAG_COMPRESSION, compression == NoCompression ? COMPRESSION_NONE : COMPRESSION_LZW) |
|
551 |
|| !TIFFSetField(tiff, TIFFTAG_SAMPLESPERPIXEL, 4) |
|
552 |
|| !TIFFSetField(tiff, TIFFTAG_BITSPERSAMPLE, 8)) { |
|
553 |
TIFFClose(tiff); |
|
554 |
return false; |
|
555 |
} |
|
556 |
// try to do the ARGB32 conversion in chunks no greater than 16 MB |
|
557 |
int chunks = (width * height * 4 / (1024 * 1024 * 16)) + 1; |
|
558 |
int chunkHeight = qMax(height / chunks, 1); |
|
559 |
||
560 |
int y = 0; |
|
561 |
while (y < height) { |
|
562 |
QImage chunk = image.copy(0, y, width, qMin(chunkHeight, height - y)).convertToFormat(QImage::Format_ARGB32); |
|
563 |
||
564 |
int chunkStart = y; |
|
565 |
int chunkEnd = y + chunk.height(); |
|
566 |
while (y < chunkEnd) { |
|
567 |
if (QSysInfo::ByteOrder == QSysInfo::LittleEndian) |
|
568 |
convert32BitOrder(chunk.scanLine(y - chunkStart), width); |
|
569 |
else |
|
570 |
convert32BitOrderBigEndian(chunk.scanLine(y - chunkStart), width); |
|
571 |
||
572 |
if (TIFFWriteScanline(tiff, reinterpret_cast<uint32 *>(chunk.scanLine(y - chunkStart)), y) != 1) { |
|
573 |
TIFFClose(tiff); |
|
574 |
return false; |
|
575 |
} |
|
576 |
++y; |
|
577 |
} |
|
578 |
} |
|
579 |
TIFFClose(tiff); |
|
580 |
} |
|
581 |
||
582 |
return true; |
|
583 |
} |
|
584 |
||
585 |
QByteArray QTiffHandler::name() const |
|
586 |
{ |
|
587 |
return "tiff"; |
|
588 |
} |
|
589 |
||
590 |
QVariant QTiffHandler::option(ImageOption option) const |
|
591 |
{ |
|
592 |
if (option == Size && canRead()) { |
|
593 |
QSize imageSize; |
|
594 |
qint64 pos = device()->pos(); |
|
595 |
TIFF *tiff = TIFFClientOpen("foo", |
|
596 |
"r", |
|
597 |
const_cast<QTiffHandler*>(this), |
|
598 |
qtiffReadProc, |
|
599 |
qtiffWriteProc, |
|
600 |
qtiffSeekProc, |
|
601 |
qtiffCloseProc, |
|
602 |
qtiffSizeProc, |
|
603 |
qtiffMapProc, |
|
604 |
qtiffUnmapProc); |
|
605 |
||
606 |
if (tiff) { |
|
607 |
uint32 width = 0; |
|
608 |
uint32 height = 0; |
|
609 |
TIFFGetField(tiff, TIFFTAG_IMAGEWIDTH, &width); |
|
610 |
TIFFGetField(tiff, TIFFTAG_IMAGELENGTH, &height); |
|
611 |
imageSize = QSize(width, height); |
|
612 |
} |
|
613 |
device()->seek(pos); |
|
614 |
if (imageSize.isValid()) |
|
615 |
return imageSize; |
|
616 |
} else if (option == CompressionRatio) { |
|
617 |
return compression; |
|
618 |
} else if (option == ImageFormat) { |
|
619 |
return QImage::Format_ARGB32; |
|
620 |
} |
|
621 |
return QVariant(); |
|
622 |
} |
|
623 |
||
624 |
void QTiffHandler::setOption(ImageOption option, const QVariant &value) |
|
625 |
{ |
|
626 |
if (option == CompressionRatio && value.type() == QVariant::Int) |
|
627 |
compression = value.toInt(); |
|
628 |
} |
|
629 |
||
630 |
bool QTiffHandler::supportsOption(ImageOption option) const |
|
631 |
{ |
|
632 |
return option == CompressionRatio |
|
633 |
|| option == Size |
|
634 |
|| option == ImageFormat; |
|
635 |
} |
|
636 |
||
637 |
void QTiffHandler::convert32BitOrder(void *buffer, int width) |
|
638 |
{ |
|
639 |
uint32 *target = reinterpret_cast<uint32 *>(buffer); |
|
640 |
for (int32 x=0; x<width; ++x) { |
|
641 |
uint32 p = target[x]; |
|
642 |
// convert between ARGB and ABGR |
|
643 |
target[x] = (p & 0xff000000) |
|
644 |
| ((p & 0x00ff0000) >> 16) |
|
645 |
| (p & 0x0000ff00) |
|
646 |
| ((p & 0x000000ff) << 16); |
|
647 |
} |
|
648 |
} |
|
649 |
||
650 |
void QTiffHandler::convert32BitOrderBigEndian(void *buffer, int width) |
|
651 |
{ |
|
652 |
uint32 *target = reinterpret_cast<uint32 *>(buffer); |
|
653 |
for (int32 x=0; x<width; ++x) { |
|
654 |
uint32 p = target[x]; |
|
655 |
target[x] = (p & 0xff000000) >> 24 |
|
656 |
| (p & 0x00ff0000) << 8 |
|
657 |
| (p & 0x0000ff00) << 8 |
|
658 |
| (p & 0x000000ff) << 8; |
|
659 |
} |
|
660 |
} |
|
661 |
||
662 |
QT_END_NAMESPACE |