Blob Blame Raw


#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);
}