| 1 | /****************************************************************************
|
|---|
| 2 | ** $Id: qmngio.cpp 2 2005-11-16 15:49:26Z dmik $
|
|---|
| 3 | **
|
|---|
| 4 | ** Implementation of MNG QImage IOHandler
|
|---|
| 5 | **
|
|---|
| 6 | ** Created : 970521
|
|---|
| 7 | **
|
|---|
| 8 | ** Copyright (C) 1997-2004 Trolltech AS. All rights reserved.
|
|---|
| 9 | **
|
|---|
| 10 | ** This file is part of the kernel module of the Qt GUI Toolkit.
|
|---|
| 11 | **
|
|---|
| 12 | ** This file may be distributed under the terms of the Q Public License
|
|---|
| 13 | ** as defined by Trolltech AS of Norway and appearing in the file
|
|---|
| 14 | ** LICENSE.QPL included in the packaging of this file.
|
|---|
| 15 | **
|
|---|
| 16 | ** This file may be distributed and/or modified under the terms of the
|
|---|
| 17 | ** GNU General Public License version 2 as published by the Free Software
|
|---|
| 18 | ** Foundation and appearing in the file LICENSE.GPL included in the
|
|---|
| 19 | ** packaging of this file.
|
|---|
| 20 | **
|
|---|
| 21 | ** Licensees holding valid Qt Enterprise Edition or Qt Professional Edition
|
|---|
| 22 | ** licenses may use this file in accordance with the Qt Commercial License
|
|---|
| 23 | ** Agreement provided with the Software.
|
|---|
| 24 | **
|
|---|
| 25 | ** This file is provided AS IS with NO WARRANTY OF ANY KIND, INCLUDING THE
|
|---|
| 26 | ** WARRANTY OF DESIGN, MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE.
|
|---|
| 27 | **
|
|---|
| 28 | ** See http://www.trolltech.com/pricing.html or email sales@trolltech.com for
|
|---|
| 29 | ** information about Qt Commercial License Agreements.
|
|---|
| 30 | ** See http://www.trolltech.com/qpl/ for QPL licensing information.
|
|---|
| 31 | ** See http://www.trolltech.com/gpl/ for GPL licensing information.
|
|---|
| 32 | **
|
|---|
| 33 | ** Contact info@trolltech.com if any conditions of this licensing are
|
|---|
| 34 | ** not clear to you.
|
|---|
| 35 | **
|
|---|
| 36 | **********************************************************************/
|
|---|
| 37 |
|
|---|
| 38 | #ifndef QT_CLEAN_NAMESPACE
|
|---|
| 39 | #define QT_CLEAN_NAMESPACE
|
|---|
| 40 | #endif
|
|---|
| 41 |
|
|---|
| 42 | #include "qdatetime.h"
|
|---|
| 43 |
|
|---|
| 44 | #ifndef QT_NO_IMAGEIO_MNG
|
|---|
| 45 |
|
|---|
| 46 | #include "qimage.h"
|
|---|
| 47 | #include "qasyncimageio.h"
|
|---|
| 48 | #include "qiodevice.h"
|
|---|
| 49 | #include "qmngio.h"
|
|---|
| 50 |
|
|---|
| 51 | // Define XMD_H prohibits the included headers of libmng.h to typedef INT32.
|
|---|
| 52 | // This is needed for Borland with STL support, since in that case, INT32 is
|
|---|
| 53 | // already defined by some Borland header.
|
|---|
| 54 | #define XMD_H
|
|---|
| 55 | #if defined(Q_OS_UNIXWARE)
|
|---|
| 56 | # define HAVE_BOOLEAN // libjpeg under Unixware seems to need this
|
|---|
| 57 | #endif
|
|---|
| 58 | #include <libmng.h>
|
|---|
| 59 | #include <stdlib.h>
|
|---|
| 60 |
|
|---|
| 61 |
|
|---|
| 62 | #ifndef QT_NO_ASYNC_IMAGE_IO
|
|---|
| 63 |
|
|---|
| 64 | class QMNGFormat : public QImageFormat {
|
|---|
| 65 | public:
|
|---|
| 66 | QMNGFormat();
|
|---|
| 67 | virtual ~QMNGFormat();
|
|---|
| 68 |
|
|---|
| 69 | int decode(QImage& img, QImageConsumer* consumer,
|
|---|
| 70 | const uchar* buffer, int length);
|
|---|
| 71 |
|
|---|
| 72 | bool openstream()
|
|---|
| 73 | {
|
|---|
| 74 | // ### We should figure out how many loops an MNG has, but for now always assume infinite.
|
|---|
| 75 | if (consumer)
|
|---|
| 76 | consumer->setLooping(0);
|
|---|
| 77 | return TRUE;
|
|---|
| 78 | }
|
|---|
| 79 | bool closestream( )
|
|---|
| 80 | {
|
|---|
| 81 | if (consumer)
|
|---|
| 82 | consumer->end();
|
|---|
| 83 | return TRUE;
|
|---|
| 84 | }
|
|---|
| 85 | bool readdata( mng_ptr pBuf, mng_uint32 iBuflen, mng_uint32p pRead )
|
|---|
| 86 | {
|
|---|
| 87 | uint m = ndata + nbuffer - ubuffer;
|
|---|
| 88 | if ( iBuflen > m ) {
|
|---|
| 89 | iBuflen = m;
|
|---|
| 90 | }
|
|---|
| 91 | *pRead = iBuflen;
|
|---|
| 92 | uint n = nbuffer-ubuffer;
|
|---|
| 93 | if ( iBuflen < n ) {
|
|---|
| 94 | // enough in buffer
|
|---|
| 95 | memcpy(pBuf, buffer+ubuffer, iBuflen);
|
|---|
| 96 | ubuffer += iBuflen;
|
|---|
| 97 | return TRUE;
|
|---|
| 98 | }
|
|---|
| 99 | if ( n ) {
|
|---|
| 100 | // consume buffer
|
|---|
| 101 | memcpy(pBuf, buffer+ubuffer, n );
|
|---|
| 102 | pBuf = (mng_ptr)((char*)pBuf + n);
|
|---|
| 103 | iBuflen -= n;
|
|---|
| 104 | ubuffer = nbuffer;
|
|---|
| 105 | }
|
|---|
| 106 | if ( iBuflen ) {
|
|---|
| 107 | // fill from incoming data
|
|---|
| 108 | memcpy(pBuf, data, iBuflen);
|
|---|
| 109 | data += iBuflen;
|
|---|
| 110 | ndata -= iBuflen;
|
|---|
| 111 | }
|
|---|
| 112 | return TRUE;
|
|---|
| 113 | }
|
|---|
| 114 | bool errorproc( mng_int32 iErrorcode,
|
|---|
| 115 | mng_int8 /*iSeverity*/,
|
|---|
| 116 | mng_chunkid iChunkname,
|
|---|
| 117 | mng_uint32 /*iChunkseq*/,
|
|---|
| 118 | mng_int32 iExtra1,
|
|---|
| 119 | mng_int32 iExtra2,
|
|---|
| 120 | mng_pchar zErrortext )
|
|---|
| 121 | {
|
|---|
| 122 | qWarning("MNG error %d: %s; chunk %c%c%c%c; subcode %d:%d",
|
|---|
| 123 | iErrorcode,zErrortext,
|
|---|
| 124 | (iChunkname>>24)&0xff,
|
|---|
| 125 | (iChunkname>>16)&0xff,
|
|---|
| 126 | (iChunkname>>8)&0xff,
|
|---|
| 127 | (iChunkname>>0)&0xff,
|
|---|
| 128 | iExtra1,iExtra2);
|
|---|
| 129 | return TRUE;
|
|---|
| 130 | }
|
|---|
| 131 | bool processheader( mng_uint32 iWidth, mng_uint32 iHeight )
|
|---|
| 132 | {
|
|---|
| 133 | image->create(iWidth,iHeight,32);
|
|---|
| 134 | image->setAlphaBuffer(TRUE);
|
|---|
| 135 | memset(image->bits(),0,iWidth*iHeight*4);
|
|---|
| 136 | consumer->setSize(iWidth,iHeight);
|
|---|
| 137 | mng_set_canvasstyle(handle,
|
|---|
| 138 | QImage::systemByteOrder() == QImage::LittleEndian
|
|---|
| 139 | ? MNG_CANVAS_BGRA8 : MNG_CANVAS_ARGB8 );
|
|---|
| 140 | return TRUE;
|
|---|
| 141 | }
|
|---|
| 142 | mng_ptr getcanvasline( mng_uint32 iLinenr )
|
|---|
| 143 | {
|
|---|
| 144 | return image->scanLine(iLinenr);
|
|---|
| 145 | }
|
|---|
| 146 | mng_bool refresh( mng_uint32 x, mng_uint32 y, mng_uint32 w, mng_uint32 h )
|
|---|
| 147 | {
|
|---|
| 148 | QRect r(x,y,w,h);
|
|---|
| 149 | consumer->changed(r);
|
|---|
| 150 | consumer->setFramePeriod(0);
|
|---|
| 151 | consumer->frameDone();
|
|---|
| 152 | return TRUE;
|
|---|
| 153 | }
|
|---|
| 154 | mng_uint32 gettickcount( )
|
|---|
| 155 | {
|
|---|
| 156 | return timer.elapsed() - losttime;
|
|---|
| 157 | }
|
|---|
| 158 | bool settimer( mng_uint32 iMsecs )
|
|---|
| 159 | {
|
|---|
| 160 | consumer->setFramePeriod(iMsecs);
|
|---|
| 161 | consumer->frameDone();
|
|---|
| 162 | state = Time;
|
|---|
| 163 | losingtimer.start();
|
|---|
| 164 | losttime -= iMsecs;
|
|---|
| 165 | return TRUE;
|
|---|
| 166 | }
|
|---|
| 167 |
|
|---|
| 168 | private:
|
|---|
| 169 | // Animation-level information
|
|---|
| 170 | enum { MovieStart, Time, Data, Data2 } state;
|
|---|
| 171 |
|
|---|
| 172 | // Image-level information
|
|---|
| 173 | mng_handle handle;
|
|---|
| 174 |
|
|---|
| 175 | // For storing unused data
|
|---|
| 176 | uchar *buffer;
|
|---|
| 177 | uint maxbuffer;
|
|---|
| 178 | uint nbuffer;
|
|---|
| 179 |
|
|---|
| 180 | // Timing
|
|---|
| 181 | QTime timer;
|
|---|
| 182 | QTime losingtimer;
|
|---|
| 183 | int losttime;
|
|---|
| 184 |
|
|---|
| 185 | void enlargeBuffer(uint n)
|
|---|
| 186 | {
|
|---|
| 187 | if ( n > maxbuffer ) {
|
|---|
| 188 | maxbuffer = n;
|
|---|
| 189 | buffer = (uchar*)realloc(buffer,n);
|
|---|
| 190 | }
|
|---|
| 191 | }
|
|---|
| 192 |
|
|---|
| 193 | // Temporary locals during single data-chunk processing
|
|---|
| 194 | const uchar* data;
|
|---|
| 195 | uint ndata;
|
|---|
| 196 | uint ubuffer;
|
|---|
| 197 | QImageConsumer* consumer;
|
|---|
| 198 | QImage* image;
|
|---|
| 199 | };
|
|---|
| 200 |
|
|---|
| 201 | class QMNGFormatType : public QImageFormatType
|
|---|
| 202 | {
|
|---|
| 203 | QImageFormat* decoderFor(const uchar* buffer, int length);
|
|---|
| 204 | const char* formatName() const;
|
|---|
| 205 | };
|
|---|
| 206 |
|
|---|
| 207 |
|
|---|
| 208 | /*
|
|---|
| 209 | \class QMNGFormat qmngio.h
|
|---|
| 210 | \brief Incremental image decoder for MNG image format.
|
|---|
| 211 |
|
|---|
| 212 | \ingroup images
|
|---|
| 213 | \ingroup graphics
|
|---|
| 214 |
|
|---|
| 215 | This subclass of QImageFormat decodes MNG format images,
|
|---|
| 216 | including animated MNGs.
|
|---|
| 217 |
|
|---|
| 218 | Animated MNG images are standard MNG images. The MNG standard
|
|---|
| 219 | defines two extension chunks that are useful for animations:
|
|---|
| 220 |
|
|---|
| 221 | <dl>
|
|---|
| 222 | <dt>gIFg - GIF-like Graphic Control Extension
|
|---|
| 223 | <dd>Includes frame disposal, user input flag (we ignore this),
|
|---|
| 224 | and inter-frame delay.
|
|---|
| 225 | <dt>gIFx - GIF-like Application Extension
|
|---|
| 226 | <dd>Multi-purpose, but we just use the Netscape extension
|
|---|
| 227 | which specifies looping.
|
|---|
| 228 | </dl>
|
|---|
| 229 |
|
|---|
| 230 | The subimages usually contain a offset chunk (oFFs) but need not.
|
|---|
| 231 |
|
|---|
| 232 | The first image defines the "screen" size. Any subsequent image that
|
|---|
| 233 | doesn't fit is clipped.
|
|---|
| 234 |
|
|---|
| 235 | TODO: decide on this point. gIFg gives disposal types, so it can be done.
|
|---|
| 236 | All images paste (\e not composite, just place all-channel copying)
|
|---|
| 237 | over the previous image to produce a subsequent frame.
|
|---|
| 238 | */
|
|---|
| 239 |
|
|---|
| 240 | /*
|
|---|
| 241 | \class QMNGFormatType qasyncimageio.h
|
|---|
| 242 | \brief Incremental image decoder for MNG image format.
|
|---|
| 243 |
|
|---|
| 244 | \ingroup images
|
|---|
| 245 | \ingroup graphics
|
|---|
| 246 | \ingroup io
|
|---|
| 247 |
|
|---|
| 248 | This subclass of QImageFormatType recognizes MNG
|
|---|
| 249 | format images, creating a QMNGFormat when required. An instance
|
|---|
| 250 | of this class is created automatically before any other factories,
|
|---|
| 251 | so you should have no need for such objects.
|
|---|
| 252 | */
|
|---|
| 253 |
|
|---|
| 254 | QImageFormat* QMNGFormatType::decoderFor( const uchar* buffer, int length )
|
|---|
| 255 | {
|
|---|
| 256 | if (length < 8) return 0;
|
|---|
| 257 |
|
|---|
| 258 | if (buffer[0]==138 // MNG signature
|
|---|
| 259 | && buffer[1]=='M'
|
|---|
| 260 | && buffer[2]=='N'
|
|---|
| 261 | && buffer[3]=='G'
|
|---|
| 262 | && buffer[4]==13
|
|---|
| 263 | && buffer[5]==10
|
|---|
| 264 | && buffer[6]==26
|
|---|
| 265 | && buffer[7]==10
|
|---|
| 266 | || buffer[0]==139 // JNG signature
|
|---|
| 267 | && buffer[1]=='J'
|
|---|
| 268 | && buffer[2]=='N'
|
|---|
| 269 | && buffer[3]=='G'
|
|---|
| 270 | && buffer[4]==13
|
|---|
| 271 | && buffer[5]==10
|
|---|
| 272 | && buffer[6]==26
|
|---|
| 273 | && buffer[7]==10
|
|---|
| 274 | #ifdef QT_NO_IMAGEIO_PNG // if we don't have native PNG support use libmng
|
|---|
| 275 | || buffer[0]==137 // PNG signature
|
|---|
| 276 | && buffer[1]=='P'
|
|---|
| 277 | && buffer[2]=='N'
|
|---|
| 278 | && buffer[3]=='G'
|
|---|
| 279 | && buffer[4]==13
|
|---|
| 280 | && buffer[5]==10
|
|---|
| 281 | && buffer[6]==26
|
|---|
| 282 | && buffer[7]==10
|
|---|
| 283 | #endif
|
|---|
| 284 | )
|
|---|
| 285 | return new QMNGFormat;
|
|---|
| 286 | return 0;
|
|---|
| 287 | }
|
|---|
| 288 |
|
|---|
| 289 | const char* QMNGFormatType::formatName() const
|
|---|
| 290 | {
|
|---|
| 291 | return "MNG";
|
|---|
| 292 | }
|
|---|
| 293 |
|
|---|
| 294 |
|
|---|
| 295 | /*!
|
|---|
| 296 | Constructs a QMNGFormat.
|
|---|
| 297 | */
|
|---|
| 298 | QMNGFormat::QMNGFormat()
|
|---|
| 299 | {
|
|---|
| 300 | state = MovieStart;
|
|---|
| 301 | handle = 0;
|
|---|
| 302 | nbuffer = 0;
|
|---|
| 303 | maxbuffer = 0;
|
|---|
| 304 | buffer = 0;
|
|---|
| 305 | losttime = 0;
|
|---|
| 306 | }
|
|---|
| 307 |
|
|---|
| 308 | /*
|
|---|
| 309 | Destroys a QMNGFormat.
|
|---|
| 310 | */
|
|---|
| 311 | QMNGFormat::~QMNGFormat()
|
|---|
| 312 | {
|
|---|
| 313 | mng_cleanup(&handle);
|
|---|
| 314 | }
|
|---|
| 315 |
|
|---|
| 316 |
|
|---|
| 317 | // C-callback to C++-member-function conversion
|
|---|
| 318 | //
|
|---|
| 319 | static mng_bool openstream( mng_handle handle )
|
|---|
| 320 | {
|
|---|
| 321 | return ((QMNGFormat*)mng_get_userdata(handle))->openstream();
|
|---|
| 322 | }
|
|---|
| 323 | static mng_bool closestream( mng_handle handle )
|
|---|
| 324 | {
|
|---|
| 325 | return ((QMNGFormat*)mng_get_userdata(handle))->closestream();
|
|---|
| 326 | }
|
|---|
| 327 | static mng_bool readdata( mng_handle handle, mng_ptr pBuf, mng_uint32 iBuflen, mng_uint32p pRead )
|
|---|
| 328 | {
|
|---|
| 329 | return ((QMNGFormat*)mng_get_userdata(handle))->readdata(pBuf,iBuflen,pRead);
|
|---|
| 330 | }
|
|---|
| 331 | static mng_bool errorproc( mng_handle handle,
|
|---|
| 332 | mng_int32 iErrorcode,
|
|---|
| 333 | mng_int8 iSeverity,
|
|---|
| 334 | mng_chunkid iChunkname,
|
|---|
| 335 | mng_uint32 iChunkseq,
|
|---|
| 336 | mng_int32 iExtra1,
|
|---|
| 337 | mng_int32 iExtra2,
|
|---|
| 338 | mng_pchar zErrortext )
|
|---|
| 339 | {
|
|---|
| 340 | return ((QMNGFormat*)mng_get_userdata(handle))->errorproc(iErrorcode,
|
|---|
| 341 | iSeverity,iChunkname,iChunkseq,iExtra1,iExtra2,zErrortext);
|
|---|
| 342 | }
|
|---|
| 343 | static mng_bool processheader( mng_handle handle,
|
|---|
| 344 | mng_uint32 iWidth, mng_uint32 iHeight )
|
|---|
| 345 | {
|
|---|
| 346 | return ((QMNGFormat*)mng_get_userdata(handle))->processheader(iWidth,iHeight);
|
|---|
| 347 | }
|
|---|
| 348 | static mng_ptr getcanvasline( mng_handle handle, mng_uint32 iLinenr )
|
|---|
| 349 | {
|
|---|
| 350 | return ((QMNGFormat*)mng_get_userdata(handle))->getcanvasline(iLinenr);
|
|---|
| 351 | }
|
|---|
| 352 | static mng_bool refresh( mng_handle handle,
|
|---|
| 353 | mng_uint32 iTop,
|
|---|
| 354 | mng_uint32 iLeft,
|
|---|
| 355 | mng_uint32 iBottom,
|
|---|
| 356 | mng_uint32 iRight )
|
|---|
| 357 | {
|
|---|
| 358 | return ((QMNGFormat*)mng_get_userdata(handle))->refresh(iTop,iLeft,iBottom,iRight);
|
|---|
| 359 | }
|
|---|
| 360 | static mng_uint32 gettickcount( mng_handle handle )
|
|---|
| 361 | {
|
|---|
| 362 | return ((QMNGFormat*)mng_get_userdata(handle))->gettickcount();
|
|---|
| 363 | }
|
|---|
| 364 | static mng_bool settimer( mng_handle handle, mng_uint32 iMsecs )
|
|---|
| 365 | {
|
|---|
| 366 | return ((QMNGFormat*)mng_get_userdata(handle))->settimer(iMsecs);
|
|---|
| 367 | }
|
|---|
| 368 |
|
|---|
| 369 | static mng_ptr memalloc( mng_size_t iLen )
|
|---|
| 370 | {
|
|---|
| 371 | return calloc(1,iLen);
|
|---|
| 372 | }
|
|---|
| 373 | static void memfree( mng_ptr iPtr, mng_size_t /*iLen*/ )
|
|---|
| 374 | {
|
|---|
| 375 | free(iPtr);
|
|---|
| 376 | }
|
|---|
| 377 |
|
|---|
| 378 | /*!
|
|---|
| 379 | This function decodes some data into image changes.
|
|---|
| 380 |
|
|---|
| 381 | Returns the number of bytes consumed.
|
|---|
| 382 | */
|
|---|
| 383 | int QMNGFormat::decode( QImage& img, QImageConsumer* cons,
|
|---|
| 384 | const uchar* buf, int length )
|
|---|
| 385 | {
|
|---|
| 386 | consumer = cons;
|
|---|
| 387 | image = &img;
|
|---|
| 388 |
|
|---|
| 389 | data = buf;
|
|---|
| 390 | ndata = length;
|
|---|
| 391 | ubuffer = 0;
|
|---|
| 392 |
|
|---|
| 393 | if ( state == MovieStart ) {
|
|---|
| 394 | handle = mng_initialize( (mng_ptr)this, ::memalloc, ::memfree, 0 );
|
|---|
| 395 | mng_set_suspensionmode( handle, MNG_TRUE );
|
|---|
| 396 | mng_setcb_openstream( handle, ::openstream );
|
|---|
| 397 | mng_setcb_closestream( handle, ::closestream );
|
|---|
| 398 | mng_setcb_readdata( handle, ::readdata );
|
|---|
| 399 | mng_setcb_errorproc( handle, ::errorproc );
|
|---|
| 400 | mng_setcb_processheader( handle, ::processheader );
|
|---|
| 401 | mng_setcb_getcanvasline( handle, ::getcanvasline );
|
|---|
| 402 | mng_setcb_refresh( handle, ::refresh );
|
|---|
| 403 | mng_setcb_gettickcount( handle, ::gettickcount );
|
|---|
| 404 | mng_setcb_settimer( handle, ::settimer );
|
|---|
| 405 | state = Data;
|
|---|
| 406 | mng_readdisplay(handle);
|
|---|
| 407 | losingtimer.start();
|
|---|
| 408 | }
|
|---|
| 409 |
|
|---|
| 410 | losttime += losingtimer.elapsed();
|
|---|
| 411 | if ( ndata || !length )
|
|---|
| 412 | mng_display_resume(handle);
|
|---|
| 413 | losingtimer.start();
|
|---|
| 414 |
|
|---|
| 415 | image = 0;
|
|---|
| 416 |
|
|---|
| 417 | nbuffer -= ubuffer;
|
|---|
| 418 | if ( nbuffer ) {
|
|---|
| 419 | // Move back unused tail
|
|---|
| 420 | memcpy(buffer,buffer+ubuffer,nbuffer);
|
|---|
| 421 | }
|
|---|
| 422 | if ( ndata ) {
|
|---|
| 423 | // Not all used.
|
|---|
| 424 | enlargeBuffer(nbuffer+ndata);
|
|---|
| 425 | memcpy(buffer+nbuffer,data,ndata);
|
|---|
| 426 | nbuffer += ndata;
|
|---|
| 427 | }
|
|---|
| 428 |
|
|---|
| 429 | return length;
|
|---|
| 430 | }
|
|---|
| 431 |
|
|---|
| 432 | static QMNGFormatType* globalMngFormatTypeObject = 0;
|
|---|
| 433 |
|
|---|
| 434 | #endif // QT_NO_ASYNC_IMAGE_IO
|
|---|
| 435 |
|
|---|
| 436 | #ifndef QT_NO_ASYNC_IMAGE_IO
|
|---|
| 437 | void qCleanupMngIO()
|
|---|
| 438 | {
|
|---|
| 439 | if ( globalMngFormatTypeObject ) {
|
|---|
| 440 | delete globalMngFormatTypeObject;
|
|---|
| 441 | globalMngFormatTypeObject = 0;
|
|---|
| 442 | }
|
|---|
| 443 | }
|
|---|
| 444 | #endif
|
|---|
| 445 |
|
|---|
| 446 | void qInitMngIO()
|
|---|
| 447 | {
|
|---|
| 448 | static bool done = FALSE;
|
|---|
| 449 | if ( !done ) {
|
|---|
| 450 | done = TRUE;
|
|---|
| 451 | #ifndef QT_NO_ASYNC_IMAGE_IO
|
|---|
| 452 | globalMngFormatTypeObject = new QMNGFormatType;
|
|---|
| 453 | qAddPostRoutine( qCleanupMngIO );
|
|---|
| 454 | #endif
|
|---|
| 455 | }
|
|---|
| 456 | }
|
|---|
| 457 |
|
|---|
| 458 | #endif // QT_NO_IMAGEIO_MNG
|
|---|