458 lines
13 KiB
C++
458 lines
13 KiB
C++
// Copyright (C) 2016 The Qt Company Ltd.
|
|
// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR LGPL-3.0-only OR GPL-2.0-only OR GPL-3.0-only
|
|
|
|
#include "qmnghandler_p.h"
|
|
|
|
#include "qimage.h"
|
|
#include "qvariant.h"
|
|
#include "qcolor.h"
|
|
|
|
#define MNG_USE_SO
|
|
#include <libmng.h>
|
|
|
|
QT_BEGIN_NAMESPACE
|
|
|
|
class QMngHandlerPrivate
|
|
{
|
|
Q_DECLARE_PUBLIC(QMngHandler)
|
|
public:
|
|
bool haveReadNone;
|
|
bool haveReadAll;
|
|
mng_handle hMNG;
|
|
QImage image;
|
|
int elapsed;
|
|
int nextDelay;
|
|
int iterCount;
|
|
int frameIndex;
|
|
int nextIndex;
|
|
int frameCount;
|
|
mng_uint32 iStyle;
|
|
mng_bool readData(mng_ptr pBuf, mng_uint32 iSize, mng_uint32p pRead);
|
|
mng_bool writeData(mng_ptr pBuf, mng_uint32 iSize, mng_uint32p pWritten);
|
|
mng_bool processHeader(mng_uint32 iWidth, mng_uint32 iHeight);
|
|
QMngHandlerPrivate(QMngHandler *q_ptr);
|
|
~QMngHandlerPrivate();
|
|
bool getNextImage(QImage *result);
|
|
bool writeImage(const QImage &image);
|
|
int currentImageNumber() const;
|
|
int imageCount() const;
|
|
bool jumpToImage(int imageNumber);
|
|
bool jumpToNextImage();
|
|
int nextImageDelay() const;
|
|
bool setBackgroundColor(const QColor &color);
|
|
QColor backgroundColor() const;
|
|
QMngHandler *q_ptr;
|
|
};
|
|
|
|
static mng_bool MNG_DECL myerror(mng_handle /*hMNG*/,
|
|
mng_int32 iErrorcode,
|
|
mng_int8 /*iSeverity*/,
|
|
mng_chunkid iChunkname,
|
|
mng_uint32 /*iChunkseq*/,
|
|
mng_int32 iExtra1,
|
|
mng_int32 iExtra2,
|
|
mng_pchar zErrortext)
|
|
{
|
|
qWarning("MNG error %d: %s; chunk %c%c%c%c; subcode %d:%d",
|
|
iErrorcode,zErrortext,
|
|
(iChunkname>>24)&0xff,
|
|
(iChunkname>>16)&0xff,
|
|
(iChunkname>>8)&0xff,
|
|
(iChunkname>>0)&0xff,
|
|
iExtra1,iExtra2);
|
|
return MNG_TRUE;
|
|
}
|
|
|
|
static mng_ptr MNG_DECL myalloc(mng_size_t iSize)
|
|
{
|
|
return (mng_ptr)calloc(1, iSize);
|
|
}
|
|
|
|
static void MNG_DECL myfree(mng_ptr pPtr, mng_size_t /*iSize*/)
|
|
{
|
|
free(pPtr);
|
|
}
|
|
|
|
static mng_bool MNG_DECL myopenstream(mng_handle)
|
|
{
|
|
return MNG_TRUE;
|
|
}
|
|
|
|
static mng_bool MNG_DECL myclosestream(mng_handle hMNG)
|
|
{
|
|
QMngHandlerPrivate *pMydata = reinterpret_cast<QMngHandlerPrivate *>(mng_get_userdata(hMNG));
|
|
pMydata->haveReadAll = true;
|
|
return MNG_TRUE;
|
|
}
|
|
|
|
static mng_bool MNG_DECL myreaddata(mng_handle hMNG,
|
|
mng_ptr pBuf,
|
|
mng_uint32 iSize,
|
|
mng_uint32p pRead)
|
|
{
|
|
QMngHandlerPrivate *pMydata = reinterpret_cast<QMngHandlerPrivate *>(mng_get_userdata(hMNG));
|
|
return pMydata->readData(pBuf, iSize, pRead);
|
|
}
|
|
|
|
static mng_bool MNG_DECL mywritedata(mng_handle hMNG,
|
|
mng_ptr pBuf,
|
|
mng_uint32 iSize,
|
|
mng_uint32p pWritten)
|
|
{
|
|
QMngHandlerPrivate *pMydata = reinterpret_cast<QMngHandlerPrivate *>(mng_get_userdata(hMNG));
|
|
return pMydata->writeData(pBuf, iSize, pWritten);
|
|
}
|
|
|
|
static mng_bool MNG_DECL myprocessheader(mng_handle hMNG,
|
|
mng_uint32 iWidth,
|
|
mng_uint32 iHeight)
|
|
{
|
|
QMngHandlerPrivate *pMydata = reinterpret_cast<QMngHandlerPrivate *>(mng_get_userdata(hMNG));
|
|
return pMydata->processHeader(iWidth, iHeight);
|
|
}
|
|
|
|
static mng_ptr MNG_DECL mygetcanvasline(mng_handle hMNG,
|
|
mng_uint32 iLinenr)
|
|
{
|
|
QMngHandlerPrivate *pMydata = reinterpret_cast<QMngHandlerPrivate *>(mng_get_userdata(hMNG));
|
|
return (mng_ptr)pMydata->image.scanLine(iLinenr);
|
|
}
|
|
|
|
static mng_bool MNG_DECL myrefresh(mng_handle /*hMNG*/,
|
|
mng_uint32 /*iX*/,
|
|
mng_uint32 /*iY*/,
|
|
mng_uint32 /*iWidth*/,
|
|
mng_uint32 /*iHeight*/)
|
|
{
|
|
return MNG_TRUE;
|
|
}
|
|
|
|
static mng_uint32 MNG_DECL mygettickcount(mng_handle hMNG)
|
|
{
|
|
QMngHandlerPrivate *pMydata = reinterpret_cast<QMngHandlerPrivate *>(mng_get_userdata(hMNG));
|
|
return pMydata->elapsed++;
|
|
}
|
|
|
|
static mng_bool MNG_DECL mysettimer(mng_handle hMNG,
|
|
mng_uint32 iMsecs)
|
|
{
|
|
QMngHandlerPrivate *pMydata = reinterpret_cast<QMngHandlerPrivate *>(mng_get_userdata(hMNG));
|
|
pMydata->elapsed += iMsecs;
|
|
pMydata->nextDelay = iMsecs;
|
|
return MNG_TRUE;
|
|
}
|
|
|
|
static mng_bool MNG_DECL myprocessterm(mng_handle hMNG,
|
|
mng_uint8 iTermaction,
|
|
mng_uint8 /*iIteraction*/,
|
|
mng_uint32 /*iDelay*/,
|
|
mng_uint32 iItermax)
|
|
{
|
|
QMngHandlerPrivate *pMydata = reinterpret_cast<QMngHandlerPrivate *>(mng_get_userdata(hMNG));
|
|
if (iTermaction == 3)
|
|
pMydata->iterCount = iItermax;
|
|
return MNG_TRUE;
|
|
}
|
|
|
|
static mng_bool MNG_DECL mytrace(mng_handle,
|
|
mng_int32 iFuncnr,
|
|
mng_int32 iFuncseq,
|
|
mng_pchar zFuncname)
|
|
{
|
|
qDebug("mng trace: iFuncnr: %d iFuncseq: %d zFuncname: %s", iFuncnr, iFuncseq, zFuncname);
|
|
return MNG_TRUE;
|
|
}
|
|
|
|
QMngHandlerPrivate::QMngHandlerPrivate(QMngHandler *q_ptr)
|
|
: haveReadNone(true), haveReadAll(false), elapsed(0), nextDelay(0), iterCount(1),
|
|
frameIndex(-1), nextIndex(0), frameCount(0), q_ptr(q_ptr)
|
|
{
|
|
iStyle = (QSysInfo::ByteOrder == QSysInfo::LittleEndian) ? MNG_CANVAS_BGRA8 : MNG_CANVAS_ARGB8;
|
|
// Initialize libmng
|
|
hMNG = mng_initialize((mng_ptr)this, myalloc, myfree, mytrace);
|
|
if (hMNG) {
|
|
// Set callback functions
|
|
mng_setcb_errorproc(hMNG, myerror);
|
|
mng_setcb_openstream(hMNG, myopenstream);
|
|
mng_setcb_closestream(hMNG, myclosestream);
|
|
mng_setcb_readdata(hMNG, myreaddata);
|
|
mng_setcb_writedata(hMNG, mywritedata);
|
|
mng_setcb_processheader(hMNG, myprocessheader);
|
|
mng_setcb_getcanvasline(hMNG, mygetcanvasline);
|
|
mng_setcb_refresh(hMNG, myrefresh);
|
|
mng_setcb_gettickcount(hMNG, mygettickcount);
|
|
mng_setcb_settimer(hMNG, mysettimer);
|
|
mng_setcb_processterm(hMNG, myprocessterm);
|
|
mng_set_doprogressive(hMNG, MNG_FALSE);
|
|
mng_set_suspensionmode(hMNG, MNG_TRUE);
|
|
}
|
|
}
|
|
|
|
QMngHandlerPrivate::~QMngHandlerPrivate()
|
|
{
|
|
mng_cleanup(&hMNG);
|
|
}
|
|
|
|
mng_bool QMngHandlerPrivate::readData(mng_ptr pBuf, mng_uint32 iSize, mng_uint32p pRead)
|
|
{
|
|
Q_Q(QMngHandler);
|
|
*pRead = q->device()->read((char *)pBuf, iSize);
|
|
return (*pRead > 0) ? MNG_TRUE : MNG_FALSE;
|
|
}
|
|
|
|
mng_bool QMngHandlerPrivate::writeData(mng_ptr pBuf, mng_uint32 iSize, mng_uint32p pWritten)
|
|
{
|
|
Q_Q(QMngHandler);
|
|
*pWritten = q->device()->write((char *)pBuf, iSize);
|
|
return MNG_TRUE;
|
|
}
|
|
|
|
mng_bool QMngHandlerPrivate::processHeader(mng_uint32 iWidth, mng_uint32 iHeight)
|
|
{
|
|
if (mng_set_canvasstyle(hMNG, iStyle) != MNG_NOERROR)
|
|
return MNG_FALSE;
|
|
if (!QImageIOHandler::allocateImage(QSize(iWidth, iHeight), QImage::Format_ARGB32, &image))
|
|
return MNG_FALSE;
|
|
image.fill(0);
|
|
return MNG_TRUE;
|
|
}
|
|
|
|
bool QMngHandlerPrivate::getNextImage(QImage *result)
|
|
{
|
|
mng_retcode ret;
|
|
const bool savedHaveReadAll = haveReadAll;
|
|
if (haveReadNone) {
|
|
haveReadNone = false;
|
|
ret = mng_readdisplay(hMNG);
|
|
} else {
|
|
ret = mng_display_resume(hMNG);
|
|
}
|
|
if ((MNG_NOERROR == ret) || (MNG_NEEDTIMERWAIT == ret)) {
|
|
*result = image;
|
|
|
|
// QTBUG-28894 -- libmng produces an extra frame at the end
|
|
// of the animation on the first loop only.
|
|
if (nextDelay == 1 && (!savedHaveReadAll && haveReadAll)) {
|
|
ret = mng_display_resume(hMNG);
|
|
}
|
|
|
|
frameIndex = nextIndex++;
|
|
if (haveReadAll && (frameCount == 0))
|
|
frameCount = nextIndex;
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
bool QMngHandlerPrivate::writeImage(const QImage &image)
|
|
{
|
|
mng_reset(hMNG);
|
|
if (mng_create(hMNG) != MNG_NOERROR)
|
|
return false;
|
|
|
|
this->image = image.convertToFormat(QImage::Format_ARGB32);
|
|
int w = image.width();
|
|
int h = image.height();
|
|
|
|
if (
|
|
// width, height, ticks, layercount, framecount, playtime, simplicity
|
|
(mng_putchunk_mhdr(hMNG, w, h, 1000, 0, 0, 0, 7) == MNG_NOERROR) &&
|
|
// termination_action, action_after_iterations, delay, iteration_max
|
|
(mng_putchunk_term(hMNG, 3, 0, 1, 0x7FFFFFFF) == MNG_NOERROR) &&
|
|
// width, height, bitdepth, colortype, compression, filter, interlace
|
|
(mng_putchunk_ihdr(hMNG, w, h, 8, 6, 0, 0, 0) == MNG_NOERROR) &&
|
|
// width, height, colortype, bitdepth, compression, filter, interlace, canvasstyle, getcanvasline
|
|
(mng_putimgdata_ihdr(hMNG, w, h, 6, 8, 0, 0, 0, iStyle, mygetcanvasline) == MNG_NOERROR) &&
|
|
(mng_putchunk_iend(hMNG) == MNG_NOERROR) &&
|
|
(mng_putchunk_mend(hMNG) == MNG_NOERROR) &&
|
|
(mng_write(hMNG) == MNG_NOERROR)
|
|
)
|
|
return true;
|
|
return false;
|
|
}
|
|
|
|
int QMngHandlerPrivate::currentImageNumber() const
|
|
{
|
|
// return mng_get_currentframe(hMNG) % imageCount(); not implemented, apparently
|
|
return frameIndex;
|
|
}
|
|
|
|
int QMngHandlerPrivate::imageCount() const
|
|
{
|
|
// return mng_get_totalframes(hMNG); not implemented, apparently
|
|
if (haveReadAll)
|
|
return frameCount;
|
|
return 0; // Don't know
|
|
}
|
|
|
|
bool QMngHandlerPrivate::jumpToImage(int imageNumber)
|
|
{
|
|
if (imageNumber == nextIndex)
|
|
return true;
|
|
|
|
if ((imageNumber == 0) && haveReadAll && (nextIndex == frameCount)) {
|
|
// Loop!
|
|
nextIndex = 0;
|
|
return true;
|
|
}
|
|
if (mng_display_freeze(hMNG) == MNG_NOERROR) {
|
|
if (mng_display_goframe(hMNG, imageNumber) == MNG_NOERROR) {
|
|
nextIndex = imageNumber;
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
bool QMngHandlerPrivate::jumpToNextImage()
|
|
{
|
|
const int numImages = imageCount();
|
|
return numImages > 1 && jumpToImage((currentImageNumber() + 1) % numImages);
|
|
}
|
|
|
|
int QMngHandlerPrivate::nextImageDelay() const
|
|
{
|
|
return nextDelay;
|
|
}
|
|
|
|
bool QMngHandlerPrivate::setBackgroundColor(const QColor &color)
|
|
{
|
|
mng_uint16 iRed = (mng_uint16)(color.red() << 8);
|
|
mng_uint16 iBlue = (mng_uint16)(color.blue() << 8);
|
|
mng_uint16 iGreen = (mng_uint16)(color.green() << 8);
|
|
return (mng_set_bgcolor(hMNG, iRed, iBlue, iGreen) == MNG_NOERROR);
|
|
}
|
|
|
|
QColor QMngHandlerPrivate::backgroundColor() const
|
|
{
|
|
mng_uint16 iRed;
|
|
mng_uint16 iBlue;
|
|
mng_uint16 iGreen;
|
|
if (mng_get_bgcolor(hMNG, &iRed, &iBlue, &iGreen) == MNG_NOERROR)
|
|
return QColor((iRed >> 8) & 0xFF, (iGreen >> 8) & 0xFF, (iBlue >> 8) & 0xFF);
|
|
return QColor();
|
|
}
|
|
|
|
QMngHandler::QMngHandler()
|
|
: d_ptr(new QMngHandlerPrivate(this))
|
|
{
|
|
}
|
|
|
|
QMngHandler::~QMngHandler()
|
|
{
|
|
}
|
|
|
|
/*! \reimp */
|
|
bool QMngHandler::canRead() const
|
|
{
|
|
Q_D(const QMngHandler);
|
|
if ((!d->haveReadNone
|
|
&& (!d->haveReadAll || (d->haveReadAll && (d->nextIndex < d->frameCount))))
|
|
|| canRead(device()))
|
|
{
|
|
setFormat("mng");
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
/*! \internal */
|
|
bool QMngHandler::canRead(QIODevice *device)
|
|
{
|
|
if (!device) {
|
|
qWarning("QMngHandler::canRead() called with no device");
|
|
return false;
|
|
}
|
|
|
|
return device->peek(8) == "\x8A\x4D\x4E\x47\x0D\x0A\x1A\x0A";
|
|
}
|
|
|
|
/*! \reimp */
|
|
bool QMngHandler::read(QImage *image)
|
|
{
|
|
Q_D(QMngHandler);
|
|
return canRead() ? d->getNextImage(image) : false;
|
|
}
|
|
|
|
/*! \reimp */
|
|
bool QMngHandler::write(const QImage &image)
|
|
{
|
|
Q_D(QMngHandler);
|
|
return d->writeImage(image);
|
|
}
|
|
|
|
/*! \reimp */
|
|
int QMngHandler::currentImageNumber() const
|
|
{
|
|
Q_D(const QMngHandler);
|
|
return d->currentImageNumber();
|
|
}
|
|
|
|
/*! \reimp */
|
|
int QMngHandler::imageCount() const
|
|
{
|
|
Q_D(const QMngHandler);
|
|
return d->imageCount();
|
|
}
|
|
|
|
/*! \reimp */
|
|
bool QMngHandler::jumpToImage(int imageNumber)
|
|
{
|
|
Q_D(QMngHandler);
|
|
return d->jumpToImage(imageNumber);
|
|
}
|
|
|
|
/*! \reimp */
|
|
bool QMngHandler::jumpToNextImage()
|
|
{
|
|
Q_D(QMngHandler);
|
|
return d->jumpToNextImage();
|
|
}
|
|
|
|
/*! \reimp */
|
|
int QMngHandler::loopCount() const
|
|
{
|
|
Q_D(const QMngHandler);
|
|
if (d->iterCount == 0x7FFFFFFF)
|
|
return -1; // infinite loop
|
|
return d->iterCount-1;
|
|
}
|
|
|
|
/*! \reimp */
|
|
int QMngHandler::nextImageDelay() const
|
|
{
|
|
Q_D(const QMngHandler);
|
|
return d->nextImageDelay();
|
|
}
|
|
|
|
/*! \reimp */
|
|
QVariant QMngHandler::option(ImageOption option) const
|
|
{
|
|
Q_D(const QMngHandler);
|
|
if (option == QImageIOHandler::Animation)
|
|
return true;
|
|
else if (option == QImageIOHandler::BackgroundColor)
|
|
return d->backgroundColor();
|
|
return QVariant();
|
|
}
|
|
|
|
/*! \reimp */
|
|
void QMngHandler::setOption(ImageOption option, const QVariant & value)
|
|
{
|
|
Q_D(QMngHandler);
|
|
if (option == QImageIOHandler::BackgroundColor)
|
|
d->setBackgroundColor(qvariant_cast<QColor>(value));
|
|
}
|
|
|
|
/*! \reimp */
|
|
bool QMngHandler::supportsOption(ImageOption option) const
|
|
{
|
|
if (option == QImageIOHandler::Animation)
|
|
return true;
|
|
else if (option == QImageIOHandler::BackgroundColor)
|
|
return true;
|
|
return false;
|
|
}
|
|
|
|
QT_END_NAMESPACE
|