#include "tiio_movX.h"
#include "traster.h"
#include "trasterimage.h"
#include "tsound.h"
#ifdef NOTE
Supportato solo RGBM32, sotto NT sarebbe possibile supportare anche 1bpp,
ma mancano i TRaster opportuni...
viene supportata
solo(LA PRIMA) traccia video !!!!
#endif
// namespace {
enum QTLibError {
QTNoError = 0x0000,
QTCantCreateParams = 0x0101,
QTCantSetParams = 0x0102,
QTPixelTypeError = 0x0103,
QTCheckLibError = 0x0104
};
string buildQTErrorString(int ec) {
switch (ec) {
case QTCantCreateParams:
return "Unable to create default params";
case QTCantSetParams:
return "Unable to set default params";
case QTPixelTypeError:
return "Unsupported pixel type";
case QTCheckLibError:
return mvGetErrorStr(mvGetErrno());
default:
return "Unknown error";
}
}
//}
//-----------------------------------------------------------
// TImageWriterMov
//-----------------------------------------------------------
class TImageWriterMov : public TImageWriter {
public:
TImageWriterMov(const TFilePath &, int frameIndex, TLevelWriterMov *);
~TImageWriterMov() {}
bool is64bitOutputSupported() { return false; }
private:
// not implemented
TImageWriterMov(const TImageWriterMov &);
TImageWriterMov &operator=(const TImageWriterMov &src);
public:
void save(const TImageP &);
int m_frameIndex;
private:
TLevelWriterMov *m_lwm;
};
//-----------------------------------------------------------
// TImageReaderMov
//-----------------------------------------------------------
class TImageReaderMov : public TImageReader {
public:
TImageReaderMov(const TFilePath &, int frameIndex, TLevelReaderMov *);
~TImageReaderMov() {}
private:
// not implemented
TImageReaderMov(const TImageReaderMov &);
TImageReaderMov &operator=(const TImageReaderMov &src);
public:
TImageP load();
int m_frameIndex;
TDimension getSize() const { return TDimension(m_lrm->m_lx, m_lrm->m_ly); }
TRect getBBox() const {
return TRect(0, 0, m_lrm->m_lx - 1, m_lrm->m_ly - 1);
}
private:
TLevelReaderMov *m_lrm;
};
//-----------------------------------------------------------
//-----------------------------------------------------------
// TImageWriterMov
//-----------------------------------------------------------
TImageWriterMov::TImageWriterMov(const TFilePath &path, int frameIndex,
TLevelWriterMov *lwm)
: TImageWriter(path), m_lwm(lwm), m_frameIndex(frameIndex) {}
//-----------------------------------------------------------
void TImageWriterMov::save(const TImageP &img) {
TRasterImageP image(img);
int lx = image->getRaster()->getLx();
int ly = image->getRaster()->getLy();
void *buffer = image->getRaster()->getRawData();
int pixSize = image->getRaster()->getPixelSize();
if (pixSize != 4)
throw TImageException(m_lwm->getFilePath(), "Unsupported pixel type");
if (!m_lwm->m_initDone) {
DMparams *imageTrackParams;
if (dmParamsCreate(&imageTrackParams) != DM_SUCCESS)
throw TImageException(m_lwm->getFilePath(),
"Unable to create image track params");
if (dmSetImageDefaults(imageTrackParams, lx, ly, DM_IMAGE_PACKING_XRGB) !=
DM_SUCCESS) {
dmParamsDestroy(imageTrackParams);
throw TImageException(m_lwm->getFilePath(),
"Unable to set image defaults");
}
if (dmParamsSetFloat(imageTrackParams, DM_IMAGE_RATE,
(double)m_lwm->m_rate) != DM_SUCCESS) {
dmParamsDestroy(imageTrackParams);
throw TImageException(m_lwm->getFilePath(), "Unable to set frame rate");
}
if (dmParamsSetEnum(imageTrackParams, DM_IMAGE_ORIENTATION,
DM_TOP_TO_BOTTOM) != DM_SUCCESS) {
dmParamsDestroy(imageTrackParams);
throw TImageException(m_lwm->getFilePath(), "Unable to set frame rate");
}
if (dmParamsSetFloat(imageTrackParams, DM_IMAGE_QUALITY_SPATIAL,
m_lwm->quality) != DM_SUCCESS) {
dmParamsDestroy(imageTrackParams);
throw TImageException(m_lwm->getFilePath(), "Unable to set quality");
}
if (dmParamsSetString(imageTrackParams, DM_IMAGE_COMPRESSION,
m_lwm->compression) != DM_SUCCESS) {
dmParamsDestroy(imageTrackParams);
throw TImageException(m_lwm->getFilePath(), "Unable to set compression");
}
if (mvAddTrack(m_lwm->id, DM_IMAGE, imageTrackParams, NULL,
&(m_lwm->imageTrack)) == DM_FAILURE) {
dmParamsDestroy(imageTrackParams);
throw TImageException(m_lwm->getFilePath(),
"Unable to add image track to movie");
}
dmParamsDestroy(imageTrackParams);
m_lwm->m_initDone = true;
}
if (mvInsertFrames(m_lwm->imageTrack, m_frameIndex, 1, lx * ly * pixSize,
buffer) != DM_SUCCESS) {
throw TImageException(m_lwm->getFilePath(),
"Unable to write image to movie");
}
}
//-----------------------------------------------------------
// TLevelWriterMov
//-----------------------------------------------------------
class TWriterInfoMov : public TWriterInfo {
public:
TWriterInfoMov() : TWriterInfo() { assert(!"Not implemented"); }
~TWriterInfoMov() {}
private:
};
//-----------------------------------------------------------
TLevelWriterMov::TLevelWriterMov(const TFilePath &path)
: TLevelWriter(path)
, m_initDone(false)
, m_rate(25)
, m_IOError(QTNoError)
, quality(DM_IMAGE_QUALITY_NORMAL)
, compression(DM_IMAGE_QT_ANIM)
, m_writerInfo(new TWriterInfoMov()) {
DMparams *movieParams;
if (dmParamsCreate(&movieParams) != DM_SUCCESS) {
m_IOError = QTCantCreateParams;
return;
}
if (mvSetMovieDefaults(movieParams, MV_FORMAT_QT) != DM_SUCCESS) {
dmParamsDestroy(movieParams);
m_IOError = QTCantCreateParams;
return;
}
if (mvCreateFile(path.getFullPath().c_str(), movieParams, NULL, &id) ==
DM_FAILURE) {
static char m[1024];
dmParamsDestroy(movieParams);
m_IOError = QTCheckLibError;
return;
}
dmParamsDestroy(movieParams);
}
//-----------------------------------------------------------
void TLevelWriterMov::saveSoundTrack(TSoundTrack *) {
throw TImageException(m_path,
"TLevelWriterMov::saveSoundTrack not Implemented");
}
//-----------------------------------------------------------
TWriterInfo *TLevelWriterMov::getWriterInfo() const { return m_writerInfo; }
//-----------------------------------------------------------
TLevelWriterMov::~TLevelWriterMov() {
bool rc = (mvClose(id) == DM_SUCCESS);
// if (!rc)
// throw TImageException(getFilePath(), "Error closing mov file");
}
//-----------------------------------------------------------
TImageWriterP TLevelWriterMov::getFrameWriter(TFrameId fid) {
if (m_IOError) throw TImageException(m_path, buildQTErrorString(m_IOError));
if (fid.getLetter() != 0) return TImageWriterP(0);
int index = fid.getNumber() - 1;
TImageWriterMov *iwm = new TImageWriterMov(m_path, index, this);
return TImageWriterP(iwm);
}
//-----------------------------------------------------------
TLevelReaderMov::TLevelReaderMov(const TFilePath &path)
: TLevelReader(path)
, IOError(QTNoError)
{
m_status = mvOpenFile(path.getFullPath().c_str(), O_RDONLY, &movie);
if (m_status != DM_SUCCESS) {
IOError = QTCheckLibError;
return;
}
track = 0;
m_status = mvFindTrackByMedium(movie, DM_IMAGE, &track);
if (m_status != DM_SUCCESS) {
IOError = QTCheckLibError;
return;
}
m_lx = mvGetImageWidth(track);
m_ly = mvGetImageHeight(track);
DMpacking packing = mvGetImagePacking(track);
if (packing != DM_IMAGE_PACKING_XRGB) {
IOError = QTPixelTypeError;
return;
}
}
//------------------------------------------------
//------------------------------------------------
// TImageReaderMov
//------------------------------------------------
TImageReaderMov::TImageReaderMov(const TFilePath &path, int frameIndex,
TLevelReaderMov *lrm)
: TImageReader(path), m_lrm(lrm), m_frameIndex(frameIndex) {}
//------------------------------------------------
TLevelReaderMov::~TLevelReaderMov() { mvClose(movie); }
//------------------------------------------------
TLevelP TLevelReaderMov::loadInfo() {
TLevelP level;
if (IOError != QTNoError)
throw TImageException(m_path, buildQTErrorString(IOError));
if (!track)
throw TImageException(getFilePath().getFullPath().c_str(),
" error reading info");
MVframe nFrames = mvGetTrackLength(track);
if (nFrames == -1) return level;
for (int i = 1; i <= nFrames; i++) level->setFrame(TFrameId(i), TImageP());
return level;
}
//------------------------------------------------
TImageP TImageReaderMov::load() {
TRaster32P ret(m_lrm->m_lx, m_lrm->m_ly);
DMstatus status =
mvReadFrames(m_lrm->track, m_frameIndex, 1, m_lrm->m_lx * m_lrm->m_ly * 4,
ret->getRawData());
if (status != DM_SUCCESS) {
throw TImageException(getFilePath().getFullPath().c_str(),
mvGetErrorStr(mvGetErrno()));
}
// getImage()->setRaster(ret);
return TRasterImageP(ret);
}
//------------------------------------------------
TImageReaderP TLevelReaderMov::getFrameReader(TFrameId fid) {
if (IOError != QTNoError)
throw TImageException(m_path, buildQTErrorString(IOError));
if (fid.getLetter() != 0) return TImageReaderP(0);
int index = fid.getNumber() - 1;
TImageReaderMov *irm = new TImageReaderMov(m_path, index, this);
return TImageReaderP(irm);
}