| 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 | 
|---|