scummvm/engines/sci/video/vmd_decoder.cpp

131 lines
3.5 KiB
C++

/* ScummVM - Graphic Adventure Engine
*
* ScummVM is the legal property of its developers, whose names
* are too numerous to list here. Please refer to the COPYRIGHT
* file distributed with this source distribution.
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU General Public License
* as published by the Free Software Foundation; either version 2
* of the License, or (at your option) any later version.
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
*
* $URL$
* $Id$
*
*/
#ifdef ENABLE_SCI32
#include "sci/video/vmd_decoder.h"
#include "common/archive.h"
#include "common/endian.h"
#include "common/util.h"
#include "common/stream.h"
#include "common/system.h"
#include "graphics/dither.h"
#include "sound/mixer.h"
#include "sound/audiostream.h"
namespace Sci {
VMDDecoder::VMDDecoder(Audio::Mixer *mixer) : _mixer(mixer) {
_vmdDecoder = new Graphics::Vmd(new Graphics::PaletteLUT(5, Graphics::PaletteLUT::kPaletteYUV));
}
VMDDecoder::~VMDDecoder() {
closeFile();
}
uint32 VMDDecoder::getFrameWaitTime() {
return _vmdDecoder->getFrameWaitTime();
}
bool VMDDecoder::loadFile(const char *fileName) {
closeFile();
_fileStream = SearchMan.createReadStreamForMember(fileName);
if (!_fileStream)
return false;
if (!_vmdDecoder->load(*_fileStream))
return false;
if (_vmdDecoder->getFeatures() & Graphics::CoktelVideo::kFeaturesPalette) {
getPalette();
setPalette(_palette);
}
if (_vmdDecoder->getFeatures() & Graphics::CoktelVideo::kFeaturesSound)
_vmdDecoder->enableSound(*_mixer);
_videoInfo.width = _vmdDecoder->getWidth();
_videoInfo.height = _vmdDecoder->getHeight();
_videoInfo.frameCount = _vmdDecoder->getFramesCount();
_videoInfo.frameRate = _vmdDecoder->getFrameRate();
_videoInfo.frameDelay = _videoInfo.frameRate * 100;
_videoInfo.currentFrame = 0;
_videoInfo.firstframeOffset = 0; // not really necessary for VMDs
if (_vmdDecoder->hasExtraData())
warning("This VMD video has extra embedded data, which is currently not handled");
_videoFrameBuffer = new byte[_videoInfo.width * _videoInfo.height];
memset(_videoFrameBuffer, 0, _videoInfo.width * _videoInfo.height);
_vmdDecoder->setVideoMemory(_videoFrameBuffer, _videoInfo.width, _videoInfo.height);
return true;
}
void VMDDecoder::closeFile() {
if (!_fileStream)
return;
_vmdDecoder->unload();
delete _fileStream;
_fileStream = 0;
delete[] _videoFrameBuffer;
_videoFrameBuffer = 0;
}
bool VMDDecoder::decodeNextFrame() {
if (_videoInfo.currentFrame == 0)
_videoInfo.startTime = g_system->getMillis();
Graphics::CoktelVideo::State state = _vmdDecoder->nextFrame();
if (state.flags & Graphics::CoktelVideo::kStatePalette) {
getPalette();
setPalette(_palette);
}
return ++_videoInfo.currentFrame < _videoInfo.frameCount;
}
void VMDDecoder::getPalette() {
const byte *pal = _vmdDecoder->getPalette();
for (int i = 0; i < 256; i++) {
_palette[i * 3 + 0] = pal[i * 3 + 0] << 2;
_palette[i * 3 + 1] = pal[i * 3 + 1] << 2;
_palette[i * 3 + 2] = pal[i * 3 + 2] << 2;
}
}
} // End of namespace Graphics
#endif