2007-05-30 21:56:52 +00:00
|
|
|
/* 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.
|
2004-01-18 05:52:04 +00:00
|
|
|
*
|
|
|
|
* 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
|
2005-10-18 01:30:26 +00:00
|
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
2004-01-18 05:52:04 +00:00
|
|
|
*
|
2006-02-11 10:01:01 +00:00
|
|
|
* $URL$
|
|
|
|
* $Id$
|
2004-01-18 05:52:04 +00:00
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
2007-09-19 08:40:12 +00:00
|
|
|
|
2004-01-18 05:52:04 +00:00
|
|
|
#include "common/file.h"
|
2004-12-05 02:55:06 +00:00
|
|
|
#include "sword1/sword1.h"
|
2004-01-18 05:52:04 +00:00
|
|
|
#include "sword1/animation.h"
|
2004-12-05 02:55:06 +00:00
|
|
|
#include "sword1/credits.h"
|
2007-04-28 07:27:53 +00:00
|
|
|
#include "sword1/text.h"
|
2004-12-05 02:55:06 +00:00
|
|
|
#include "sound/vorbis.h"
|
2004-02-22 14:11:16 +00:00
|
|
|
|
2004-07-11 04:41:48 +00:00
|
|
|
#include "common/config-manager.h"
|
2006-08-26 11:34:35 +00:00
|
|
|
#include "common/endian.h"
|
2004-07-11 04:41:48 +00:00
|
|
|
#include "common/str.h"
|
2007-03-17 00:53:21 +00:00
|
|
|
#include "common/events.h"
|
2005-01-10 22:06:49 +00:00
|
|
|
#include "common/system.h"
|
2008-11-18 17:48:19 +00:00
|
|
|
#include "graphics/surface.h"
|
2004-12-05 02:55:06 +00:00
|
|
|
|
2004-01-18 05:52:04 +00:00
|
|
|
namespace Sword1 {
|
|
|
|
|
2006-08-26 11:34:35 +00:00
|
|
|
static const char *sequenceList[20] = {
|
|
|
|
"ferrari", // 0 CD2 ferrari running down fitz in sc19
|
|
|
|
"ladder", // 1 CD2 george walking down ladder to dig sc24->sc$
|
|
|
|
"steps", // 2 CD2 george walking down steps sc23->sc24
|
|
|
|
"sewer", // 3 CD1 george entering sewer sc2->sc6
|
|
|
|
"intro", // 4 CD1 intro sequence ->sc1
|
|
|
|
"river", // 5 CD1 george being thrown into river by flap & g$
|
|
|
|
"truck", // 6 CD2 truck arriving at bull's head sc45->sc53/4
|
|
|
|
"grave", // 7 BOTH george's grave in scotland, from sc73 + from sc38 $
|
|
|
|
"montfcon", // 8 CD2 monfaucon clue in ireland dig, sc25
|
|
|
|
"tapestry", // 9 CD2 tapestry room beyond spain well, sc61
|
|
|
|
"ireland", // 10 CD2 ireland establishing shot europe_map->sc19
|
|
|
|
"finale", // 11 CD2 grand finale at very end, from sc73
|
|
|
|
"history", // 12 CD1 George's history lesson from Nico, in sc10
|
|
|
|
"spanish", // 13 CD2 establishing shot for 1st visit to Spain, europe_m$
|
|
|
|
"well", // 14 CD2 first time being lowered down well in Spai$
|
|
|
|
"candle", // 15 CD2 Candle burning down in Spain mausoleum sc59
|
|
|
|
"geodrop", // 16 CD2 from sc54, George jumping down onto truck
|
|
|
|
"vulture", // 17 CD2 from sc54, vultures circling George's dead body
|
|
|
|
"enddemo", // 18 --- for end of single CD demo
|
|
|
|
"credits", // 19 CD2 credits, to follow "finale" sequence
|
|
|
|
};
|
2004-01-18 05:52:04 +00:00
|
|
|
|
2006-08-26 11:34:35 +00:00
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
|
|
// Basic movie player
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
2004-01-18 05:52:04 +00:00
|
|
|
|
2008-09-27 17:47:22 +00:00
|
|
|
MoviePlayer::MoviePlayer(SwordEngine *vm, Screen *screen, Text *textMan, Audio::Mixer *snd, OSystem *system)
|
|
|
|
: _vm(vm), _screen(screen), _textMan(textMan), _snd(snd), _system(system) {
|
2006-08-26 11:34:35 +00:00
|
|
|
_bgSoundStream = NULL;
|
|
|
|
_ticks = 0;
|
2007-04-28 07:27:53 +00:00
|
|
|
_black = 1;
|
|
|
|
_white = 255;
|
2006-08-26 11:34:35 +00:00
|
|
|
_currentFrame = 0;
|
|
|
|
_forceFrame = false;
|
|
|
|
_framesSkipped = 0;
|
2004-03-21 18:49:04 +00:00
|
|
|
}
|
2006-05-17 23:52:45 +00:00
|
|
|
|
2006-08-26 11:34:35 +00:00
|
|
|
MoviePlayer::~MoviePlayer(void) {
|
2004-12-05 02:55:06 +00:00
|
|
|
}
|
|
|
|
|
2006-08-26 11:34:35 +00:00
|
|
|
void MoviePlayer::updatePalette(byte *pal, bool packed) {
|
|
|
|
byte palette[4 * 256];
|
|
|
|
byte *p = palette;
|
2007-04-28 07:27:53 +00:00
|
|
|
|
|
|
|
uint32 maxWeight = 0;
|
|
|
|
uint32 minWeight = 0xFFFFFFFF;
|
|
|
|
|
2006-08-26 11:34:35 +00:00
|
|
|
for (int i = 0; i < 256; i++) {
|
2007-04-28 07:27:53 +00:00
|
|
|
int r = *pal++;
|
|
|
|
int g = *pal++;
|
|
|
|
int b = *pal++;
|
|
|
|
|
2006-08-26 11:34:35 +00:00
|
|
|
if (!packed)
|
2007-04-28 07:27:53 +00:00
|
|
|
pal++;
|
|
|
|
|
|
|
|
uint32 weight = 3 * r * r + 6 * g * g + 2 * b * b;
|
|
|
|
|
|
|
|
if (weight >= maxWeight) {
|
|
|
|
_white = i;
|
|
|
|
maxWeight = weight;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (weight <= minWeight) {
|
|
|
|
_black = i;
|
|
|
|
minWeight = i;
|
|
|
|
}
|
|
|
|
|
|
|
|
*p++ = r;
|
|
|
|
*p++ = g;
|
|
|
|
*p++ = b;
|
|
|
|
*p++ = 0;
|
2006-08-26 11:34:35 +00:00
|
|
|
}
|
2007-04-28 07:27:53 +00:00
|
|
|
|
2007-04-01 18:05:11 +00:00
|
|
|
_system->setPalette(palette, 0, 256);
|
2006-08-26 11:34:35 +00:00
|
|
|
_forceFrame = true;
|
2004-12-05 02:55:06 +00:00
|
|
|
}
|
|
|
|
|
2006-08-26 11:34:35 +00:00
|
|
|
void MoviePlayer::handleScreenChanged(void) {
|
2004-12-05 02:55:06 +00:00
|
|
|
}
|
|
|
|
|
2006-08-26 11:34:35 +00:00
|
|
|
bool MoviePlayer::initOverlays(uint32 id) {
|
|
|
|
return true;
|
2004-01-18 05:52:04 +00:00
|
|
|
}
|
|
|
|
|
2006-08-26 11:34:35 +00:00
|
|
|
bool MoviePlayer::checkSkipFrame(void) {
|
|
|
|
if (_forceFrame) {
|
|
|
|
_forceFrame = false;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (_framesSkipped > 10) {
|
|
|
|
warning("Forced frame %d to be displayed", _currentFrame);
|
|
|
|
_framesSkipped = 0;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (_bgSoundStream) {
|
|
|
|
if ((_snd->getSoundElapsedTime(_bgSoundHandle) * 12) / 1000 < _currentFrame + 1)
|
|
|
|
return false;
|
|
|
|
} else {
|
2007-04-01 18:05:11 +00:00
|
|
|
if (_system->getMillis() <= _ticks)
|
2006-08-26 11:34:35 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
_framesSkipped++;
|
|
|
|
return true;
|
2004-12-05 02:55:06 +00:00
|
|
|
}
|
|
|
|
|
2007-04-14 14:07:11 +00:00
|
|
|
bool MoviePlayer::syncFrame(void) {
|
2006-08-26 11:34:35 +00:00
|
|
|
_ticks += 83;
|
|
|
|
if (checkSkipFrame()) {
|
|
|
|
warning("Skipped frame %d", _currentFrame);
|
2007-04-14 14:07:11 +00:00
|
|
|
return false;
|
2006-08-26 11:34:35 +00:00
|
|
|
}
|
|
|
|
if (_bgSoundStream) {
|
|
|
|
while (_snd->isSoundHandleActive(_bgSoundHandle) && (_snd->getSoundElapsedTime(_bgSoundHandle) * 12) / 1000 < _currentFrame) {
|
2007-04-01 18:05:11 +00:00
|
|
|
_system->delayMillis(10);
|
2006-08-26 11:34:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// In case the background sound ends prematurely, update _ticks
|
|
|
|
// so that we can still fall back on the no-sound sync case for
|
|
|
|
// the subsequent frames.
|
|
|
|
|
2007-04-01 18:05:11 +00:00
|
|
|
_ticks = _system->getMillis();
|
2006-08-26 11:34:35 +00:00
|
|
|
} else {
|
2007-04-01 18:05:11 +00:00
|
|
|
while (_system->getMillis() < _ticks) {
|
|
|
|
_system->delayMillis(10);
|
2006-08-26 11:34:35 +00:00
|
|
|
}
|
|
|
|
}
|
2007-04-14 14:07:11 +00:00
|
|
|
return true;
|
2004-03-21 18:49:04 +00:00
|
|
|
}
|
|
|
|
|
2004-01-18 05:52:04 +00:00
|
|
|
/**
|
|
|
|
* Plays an animated cutscene.
|
2004-12-05 02:55:06 +00:00
|
|
|
* @param id the id of the file
|
2004-01-18 05:52:04 +00:00
|
|
|
*/
|
2006-08-26 11:34:35 +00:00
|
|
|
bool MoviePlayer::load(uint32 id) {
|
2007-04-28 07:27:53 +00:00
|
|
|
Common::File f;
|
|
|
|
char fileName[20];
|
|
|
|
|
2006-08-26 11:34:35 +00:00
|
|
|
_id = id;
|
|
|
|
_bgSoundStream = NULL;
|
2007-04-28 07:27:53 +00:00
|
|
|
|
|
|
|
if (SwordEngine::_systemVars.showText) {
|
|
|
|
sprintf(fileName, "%s.txt", sequenceList[id]);
|
|
|
|
if (f.open(fileName)) {
|
2008-12-01 21:13:02 +00:00
|
|
|
Common::String line;
|
2007-04-28 07:27:53 +00:00
|
|
|
int lineNo = 0;
|
|
|
|
int lastEnd = -1;
|
|
|
|
|
|
|
|
_movieTexts.clear();
|
2008-12-01 21:13:02 +00:00
|
|
|
while (!f.eos() && !f.err()) {
|
|
|
|
line = f.readLine();
|
2007-04-28 07:27:53 +00:00
|
|
|
lineNo++;
|
2008-12-01 21:13:02 +00:00
|
|
|
if (line.empty() || line[0] == '#') {
|
2007-04-28 07:27:53 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2008-12-01 21:13:02 +00:00
|
|
|
const char *ptr = line.c_str();
|
2007-04-28 07:27:53 +00:00
|
|
|
|
|
|
|
// TODO: Better error handling
|
2008-12-01 21:35:17 +00:00
|
|
|
int startFrame = strtoul(ptr, const_cast<char **>(&ptr), 10);
|
|
|
|
int endFrame = strtoul(ptr, const_cast<char **>(&ptr), 10);
|
2007-04-28 07:27:53 +00:00
|
|
|
|
|
|
|
while (*ptr && isspace(*ptr))
|
|
|
|
ptr++;
|
|
|
|
|
|
|
|
if (startFrame > endFrame) {
|
|
|
|
warning("%s:%d: startFrame (%d) > endFrame (%d)", fileName, lineNo, startFrame, endFrame);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (startFrame <= lastEnd) {
|
|
|
|
warning("%s:%d startFrame (%d) <= lastEnd (%d)", fileName, lineNo, startFrame, lastEnd);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
_movieTexts.push_back(new MovieText(startFrame, endFrame, ptr));
|
|
|
|
lastEnd = endFrame;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2004-12-05 02:55:06 +00:00
|
|
|
if (SwordEngine::_systemVars.cutscenePackVersion == 1) {
|
|
|
|
if ((id == SEQ_INTRO) || (id == SEQ_FINALE) || (id == SEQ_HISTORY) || (id == SEQ_FERRARI)) {
|
2006-08-26 11:34:35 +00:00
|
|
|
#ifdef USE_VORBIS
|
2004-12-05 02:55:06 +00:00
|
|
|
// these sequences are language specific
|
2007-04-28 07:27:53 +00:00
|
|
|
sprintf(fileName, "%s.snd", sequenceList[id]);
|
2005-05-10 22:56:25 +00:00
|
|
|
Common::File *oggSource = new Common::File();
|
2007-04-28 07:27:53 +00:00
|
|
|
if (oggSource->open(fileName)) {
|
2004-12-05 02:55:06 +00:00
|
|
|
SplittedAudioStream *sStream = new SplittedAudioStream();
|
|
|
|
uint32 numSegs = oggSource->readUint32LE(); // number of audio segments, either 1 or 2.
|
|
|
|
// for each segment and each of the 7 languages, we've got fileoffset and size
|
|
|
|
uint32 *header = (uint32*)malloc(numSegs * 7 * 2 * 4);
|
|
|
|
for (uint32 cnt = 0; cnt < numSegs * 7 * 2; cnt++)
|
|
|
|
header[cnt] = oggSource->readUint32LE();
|
|
|
|
for (uint32 segCnt = 0; segCnt < numSegs; segCnt++) {
|
|
|
|
oggSource->seek( header[SwordEngine::_systemVars.language * 2 + 0 + segCnt * 14]);
|
|
|
|
uint32 segSize = header[SwordEngine::_systemVars.language * 2 + 1 + segCnt * 14];
|
2007-02-25 00:03:47 +00:00
|
|
|
Common::MemoryReadStream *stream = oggSource->readStream(segSize);
|
2007-02-25 00:09:22 +00:00
|
|
|
Audio::AudioStream *apStream = Audio::makeVorbisStream(stream, true);
|
2004-12-05 02:55:06 +00:00
|
|
|
if (!apStream)
|
2007-04-28 07:27:53 +00:00
|
|
|
error("Can't create Vorbis Stream from file %s", fileName);
|
2004-12-05 02:55:06 +00:00
|
|
|
sStream->appendStream(apStream);
|
|
|
|
}
|
|
|
|
free(header);
|
2006-08-26 11:34:35 +00:00
|
|
|
_bgSoundStream = sStream;
|
2004-12-05 02:55:06 +00:00
|
|
|
} else
|
2007-04-28 07:27:53 +00:00
|
|
|
warning("Sound file \"%s\" not found", fileName);
|
2007-02-25 00:03:47 +00:00
|
|
|
delete oggSource;
|
2006-08-26 11:34:35 +00:00
|
|
|
#endif
|
|
|
|
initOverlays(id);
|
2004-12-05 02:55:06 +00:00
|
|
|
}
|
|
|
|
}
|
2006-08-26 11:34:35 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void MoviePlayer::play(void) {
|
2007-04-28 07:27:53 +00:00
|
|
|
_screen->clearScreen();
|
2006-08-26 11:34:35 +00:00
|
|
|
_framesSkipped = 0;
|
2007-04-01 18:05:11 +00:00
|
|
|
_ticks = _system->getMillis();
|
2006-08-26 12:29:00 +00:00
|
|
|
_bgSoundStream = Audio::AudioStream::openStreamFile(sequenceList[_id]);
|
2006-08-26 11:34:35 +00:00
|
|
|
if (_bgSoundStream) {
|
|
|
|
_snd->playInputStream(Audio::Mixer::kSFXSoundType, &_bgSoundHandle, _bgSoundStream);
|
|
|
|
}
|
|
|
|
_currentFrame = 0;
|
2006-08-28 04:53:21 +00:00
|
|
|
bool terminated = false;
|
2007-04-01 18:05:11 +00:00
|
|
|
Common::EventManager *eventMan = _system->getEventManager();
|
2008-09-27 17:51:22 +00:00
|
|
|
while (!terminated && decodeFrame()) {
|
2007-04-28 07:27:53 +00:00
|
|
|
if (!_movieTexts.empty()) {
|
|
|
|
if (_currentFrame == _movieTexts[0]->_startFrame) {
|
|
|
|
_textMan->makeTextSprite(2, (uint8 *)_movieTexts[0]->_text, 600, LETTER_COL);
|
|
|
|
|
|
|
|
FrameHeader *frame = _textMan->giveSpriteData(2);
|
|
|
|
_textWidth = frame->width;
|
|
|
|
_textHeight = frame->height;
|
|
|
|
_textX = 320 - _textWidth / 2;
|
|
|
|
_textY = 420 - _textHeight;
|
|
|
|
}
|
|
|
|
if (_currentFrame == _movieTexts[0]->_endFrame) {
|
2008-10-13 18:41:12 +00:00
|
|
|
_textMan->releaseText(2, false);
|
2007-04-28 07:27:53 +00:00
|
|
|
delete _movieTexts.remove_at(0);
|
|
|
|
}
|
|
|
|
}
|
2006-08-26 11:34:35 +00:00
|
|
|
processFrame();
|
2007-04-14 14:07:11 +00:00
|
|
|
if (syncFrame())
|
|
|
|
updateScreen();
|
2006-08-26 11:34:35 +00:00
|
|
|
_currentFrame++;
|
2007-03-17 19:02:05 +00:00
|
|
|
Common::Event event;
|
2007-03-17 00:53:21 +00:00
|
|
|
while (eventMan->pollEvent(event)) {
|
2006-08-26 11:34:35 +00:00
|
|
|
switch (event.type) {
|
2007-03-17 19:02:05 +00:00
|
|
|
case Common::EVENT_SCREEN_CHANGED:
|
2006-08-26 11:34:35 +00:00
|
|
|
handleScreenChanged();
|
|
|
|
break;
|
2007-03-17 19:02:05 +00:00
|
|
|
case Common::EVENT_KEYDOWN:
|
2008-09-27 17:51:22 +00:00
|
|
|
if (event.kbd.keycode == Common::KEYCODE_ESCAPE)
|
2006-08-28 04:53:21 +00:00
|
|
|
terminated = true;
|
2006-08-26 11:34:35 +00:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
2004-02-08 14:11:11 +00:00
|
|
|
}
|
2004-01-18 05:52:04 +00:00
|
|
|
}
|
2008-09-30 12:27:38 +00:00
|
|
|
if (_vm->shouldQuit())
|
2008-09-27 17:51:22 +00:00
|
|
|
terminated = true;
|
2004-01-18 05:52:04 +00:00
|
|
|
}
|
2007-04-28 07:27:53 +00:00
|
|
|
|
2008-09-27 17:51:22 +00:00
|
|
|
if (terminated)
|
2008-09-27 17:47:22 +00:00
|
|
|
_snd->stopHandle(_bgSoundHandle);
|
|
|
|
|
2008-11-18 17:48:19 +00:00
|
|
|
_textMan->releaseText(2, false);
|
2007-04-28 07:27:53 +00:00
|
|
|
|
2008-10-13 18:41:12 +00:00
|
|
|
while (!_movieTexts.empty())
|
|
|
|
delete _movieTexts.remove_at(_movieTexts.size() - 1);
|
|
|
|
|
2006-08-26 11:34:35 +00:00
|
|
|
while (_snd->isSoundHandleActive(_bgSoundHandle))
|
2007-04-01 18:05:11 +00:00
|
|
|
_system->delayMillis(100);
|
2004-12-05 02:55:06 +00:00
|
|
|
|
2006-08-26 11:34:35 +00:00
|
|
|
// It's tempting to call _screen->fullRefresh() here to restore the old
|
|
|
|
// palette. However, that causes glitches with DXA movies, here the
|
|
|
|
// previous location would be momentarily drawn, before switching to
|
|
|
|
// the new one. Work around this by setting the palette to black.
|
2004-01-18 05:52:04 +00:00
|
|
|
|
2006-08-26 11:34:35 +00:00
|
|
|
byte pal[3 * 256];
|
|
|
|
memset(pal, 0, sizeof(pal));
|
|
|
|
updatePalette(pal, true);
|
2004-01-18 05:52:04 +00:00
|
|
|
}
|
|
|
|
|
2004-12-05 02:55:06 +00:00
|
|
|
SplittedAudioStream::SplittedAudioStream(void) {
|
|
|
|
_queue = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
SplittedAudioStream::~SplittedAudioStream(void) {
|
|
|
|
while (_queue) {
|
|
|
|
delete _queue->stream;
|
|
|
|
FileQueue *que = _queue->next;
|
|
|
|
delete _queue;
|
|
|
|
_queue = que;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
int SplittedAudioStream::getRate(void) const {
|
|
|
|
if (_queue)
|
|
|
|
return _queue->stream->getRate();
|
|
|
|
else
|
|
|
|
return 22050;
|
|
|
|
}
|
|
|
|
|
2006-04-29 22:33:31 +00:00
|
|
|
void SplittedAudioStream::appendStream(Audio::AudioStream *stream) {
|
2004-12-05 02:55:06 +00:00
|
|
|
FileQueue **que = &_queue;
|
|
|
|
while (*que)
|
|
|
|
que = &((*que)->next);
|
|
|
|
*que = new FileQueue;
|
|
|
|
(*que)->stream = stream;
|
|
|
|
(*que)->next = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool SplittedAudioStream::endOfData(void) const {
|
|
|
|
if (_queue)
|
|
|
|
return _queue->stream->endOfData();
|
|
|
|
else
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool SplittedAudioStream::isStereo(void) const {
|
|
|
|
if (_queue)
|
|
|
|
return _queue->stream->isStereo();
|
|
|
|
else
|
|
|
|
return false; // all the BS1 files are mono, anyways.
|
|
|
|
}
|
|
|
|
|
|
|
|
int SplittedAudioStream::readBuffer(int16 *buffer, const int numSamples) {
|
|
|
|
int retVal = 0;
|
|
|
|
int needSamples = numSamples;
|
|
|
|
while (needSamples && _queue) {
|
|
|
|
int retSmp = _queue->stream->readBuffer(buffer, needSamples);
|
|
|
|
needSamples -= retSmp;
|
|
|
|
retVal += retSmp;
|
|
|
|
buffer += retSmp;
|
|
|
|
if (_queue->stream->endOfData()) {
|
|
|
|
delete _queue->stream;
|
|
|
|
FileQueue *que = _queue->next;
|
|
|
|
delete _queue;
|
|
|
|
_queue = que;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return retVal;
|
|
|
|
}
|
|
|
|
|
2006-08-26 11:34:35 +00:00
|
|
|
#ifdef USE_ZLIB
|
|
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
|
|
// Movie player for the new DXA movies
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
2008-09-27 17:47:22 +00:00
|
|
|
MoviePlayerDXA::MoviePlayerDXA(SwordEngine *vm, Screen *screen, Text *textMan, Audio::Mixer *snd, OSystem *system)
|
|
|
|
: MoviePlayer(vm, screen, textMan, snd, system) {
|
2006-08-26 11:34:35 +00:00
|
|
|
debug(0, "Creating DXA cutscene player");
|
|
|
|
}
|
|
|
|
|
|
|
|
MoviePlayerDXA::~MoviePlayerDXA(void) {
|
2006-08-27 10:40:09 +00:00
|
|
|
closeFile();
|
2006-08-26 11:34:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool MoviePlayerDXA::load(uint32 id) {
|
|
|
|
if (!MoviePlayer::load(id))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
char filename[20];
|
|
|
|
snprintf(filename, sizeof(filename), "%s.dxa", sequenceList[id]);
|
2007-11-01 06:00:16 +00:00
|
|
|
if (loadFile(filename)) {
|
2006-08-26 11:34:35 +00:00
|
|
|
// The Broken Sword games always use external audio tracks.
|
2008-12-14 23:41:48 +00:00
|
|
|
if (_fileStream->readUint32BE() != MKID_BE('NULL'))
|
2006-08-26 11:34:35 +00:00
|
|
|
return false;
|
|
|
|
_frameWidth = getWidth();
|
|
|
|
_frameHeight = getHeight();
|
|
|
|
_frameX = (640 - _frameWidth) / 2;
|
|
|
|
_frameY = (480 - _frameHeight) / 2;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool MoviePlayerDXA::initOverlays(uint32 id) {
|
|
|
|
// TODO
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void MoviePlayerDXA::setPalette(byte *pal) {
|
|
|
|
updatePalette(pal, true);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool MoviePlayerDXA::decodeFrame(void) {
|
2009-01-06 17:49:41 +00:00
|
|
|
if ((uint32)_currentFrame < (uint32)getFrameCount()) {
|
2006-08-26 11:34:35 +00:00
|
|
|
decodeNextFrame();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
void MoviePlayerDXA::processFrame(void) {
|
2008-11-18 17:48:19 +00:00
|
|
|
}
|
2007-04-28 07:27:53 +00:00
|
|
|
|
2008-11-18 17:48:19 +00:00
|
|
|
void MoviePlayerDXA::updateScreen(void) {
|
2009-01-06 17:49:41 +00:00
|
|
|
Graphics::Surface *frameBuffer = _system->lockScreen();
|
2009-02-14 15:52:31 +00:00
|
|
|
copyFrameToBuffer((byte *)frameBuffer->pixels, _frameX, _frameY, 640);
|
2007-04-28 07:27:53 +00:00
|
|
|
|
2008-11-18 17:48:19 +00:00
|
|
|
// TODO: Handle the advanced cutscene packs. Do they really exist?
|
2007-04-28 07:27:53 +00:00
|
|
|
|
2008-11-18 17:48:19 +00:00
|
|
|
// We cannot draw the text to _drawBuffer, since that's one of the
|
|
|
|
// decoder's internal buffers, and besides it may be much smaller than
|
|
|
|
// the screen, so it's possible that the subtitles don't fit on it.
|
|
|
|
// Instead, we get the frame buffer from the backend, after copying the
|
|
|
|
// frame to it, and draw on that.
|
2007-04-28 07:27:53 +00:00
|
|
|
|
2008-11-18 17:48:19 +00:00
|
|
|
if (_textMan->giveSpriteData(2)) {
|
2007-04-28 07:27:53 +00:00
|
|
|
byte *src = (byte *)_textMan->giveSpriteData(2) + sizeof(FrameHeader);
|
2008-11-18 17:48:19 +00:00
|
|
|
byte *dst = (byte *)frameBuffer->getBasePtr(_textX, _textY);
|
2007-04-28 07:27:53 +00:00
|
|
|
|
2008-11-18 17:48:19 +00:00
|
|
|
for (int y = 0; y < _textHeight; y++) {
|
|
|
|
for (int x = 0; x < _textWidth; x++) {
|
2007-04-28 07:27:53 +00:00
|
|
|
switch (src[x]) {
|
|
|
|
case BORDER_COL:
|
|
|
|
dst[x] = _black;
|
|
|
|
break;
|
|
|
|
case LETTER_COL:
|
|
|
|
dst[x] = _white;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
src += _textWidth;
|
2008-11-18 17:48:19 +00:00
|
|
|
dst += frameBuffer->pitch;
|
2007-04-28 07:27:53 +00:00
|
|
|
}
|
2006-08-26 11:34:35 +00:00
|
|
|
|
2007-04-28 07:27:53 +00:00
|
|
|
}
|
2008-11-18 17:48:19 +00:00
|
|
|
|
2009-01-06 17:49:41 +00:00
|
|
|
_system->unlockScreen();
|
|
|
|
|
2007-04-01 18:05:11 +00:00
|
|
|
_system->updateScreen();
|
2006-08-26 11:34:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef USE_MPEG2
|
|
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
|
|
// Movie player for the old MPEG movies
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
2008-09-27 17:47:22 +00:00
|
|
|
MoviePlayerMPEG::MoviePlayerMPEG(SwordEngine *vm, Screen *screen, Text *textMan, Audio::Mixer *snd, OSystem *system)
|
|
|
|
: MoviePlayer(vm, screen, textMan, snd, system) {
|
2006-08-26 11:34:35 +00:00
|
|
|
#ifdef BACKEND_8BIT
|
|
|
|
debug(0, "Creating MPEG cutscene player (8-bit)");
|
|
|
|
#else
|
|
|
|
debug(0, "Creating MPEG cutscene player (16-bit)");
|
|
|
|
#endif
|
|
|
|
for (uint8 cnt = 0; cnt < INTRO_LOGO_OVLS; cnt++)
|
|
|
|
_logoOvls[cnt] = NULL;
|
|
|
|
_introPal = NULL;
|
|
|
|
_anim = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
MoviePlayerMPEG::~MoviePlayerMPEG(void) {
|
|
|
|
free(_introPal);
|
|
|
|
for (uint8 cnt = 0; cnt < INTRO_LOGO_OVLS; cnt++)
|
|
|
|
free(_logoOvls[cnt]);
|
|
|
|
delete _anim;
|
|
|
|
}
|
|
|
|
|
|
|
|
void MoviePlayerMPEG::handleScreenChanged(void) {
|
|
|
|
_anim->handleScreenChanged();
|
|
|
|
}
|
|
|
|
|
|
|
|
void MoviePlayerMPEG::insertOverlay(OverlayColor *buf, uint8 *ovl, OverlayColor *pal) {
|
|
|
|
if (ovl != NULL)
|
|
|
|
for (uint32 cnt = 0; cnt < 640 * 400; cnt++)
|
|
|
|
if (ovl[cnt])
|
|
|
|
buf[cnt] = pal[ovl[cnt]];
|
|
|
|
}
|
|
|
|
|
|
|
|
bool MoviePlayerMPEG::load(uint32 id) {
|
|
|
|
if (MoviePlayer::load(id)) {
|
2007-04-28 07:27:53 +00:00
|
|
|
_anim = new AnimationState(this, _screen, _system);
|
2006-08-26 12:29:00 +00:00
|
|
|
return _anim->init(sequenceList[id]);
|
2006-08-26 11:34:35 +00:00
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool MoviePlayerMPEG::initOverlays(uint32 id) {
|
|
|
|
if (id == SEQ_INTRO) {
|
|
|
|
ArcFile ovlFile;
|
|
|
|
if (!ovlFile.open("intro.dat")) {
|
|
|
|
warning("\"intro.dat\" not found");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
ovlFile.enterPath(SwordEngine::_systemVars.language);
|
|
|
|
for (uint8 fcnt = 0; fcnt < 12; fcnt++) {
|
|
|
|
_logoOvls[fcnt] = ovlFile.decompressFile(fcnt);
|
|
|
|
if (fcnt > 0)
|
|
|
|
for (uint32 cnt = 0; cnt < 640 * 400; cnt++)
|
|
|
|
if (_logoOvls[fcnt - 1][cnt] && !_logoOvls[fcnt][cnt])
|
|
|
|
_logoOvls[fcnt][cnt] = _logoOvls[fcnt - 1][cnt];
|
|
|
|
}
|
|
|
|
uint8 *pal = ovlFile.fetchFile(12);
|
2008-11-06 15:02:50 +00:00
|
|
|
_introPal = (OverlayColor *)malloc(256 * sizeof(OverlayColor));
|
|
|
|
Graphics::PixelFormat format = _system->getOverlayFormat();
|
2006-08-26 11:34:35 +00:00
|
|
|
for (uint16 cnt = 0; cnt < 256; cnt++)
|
2009-01-22 04:35:10 +00:00
|
|
|
_introPal[cnt] = format.RGBToColor(pal[cnt * 3 + 0], pal[cnt * 3 + 1], pal[cnt * 3 + 2]);
|
2006-08-26 11:34:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool MoviePlayerMPEG::decodeFrame(void) {
|
|
|
|
return _anim->decodeFrame();
|
|
|
|
}
|
|
|
|
|
|
|
|
void MoviePlayerMPEG::updateScreen(void) {
|
|
|
|
_anim->updateScreen();
|
|
|
|
}
|
|
|
|
|
|
|
|
void MoviePlayerMPEG::processFrame(void) {
|
|
|
|
#ifndef BACKEND_8BIT
|
|
|
|
if ((_id != 4) || (SwordEngine::_systemVars.cutscenePackVersion == 0))
|
|
|
|
return;
|
|
|
|
OverlayColor *buf = _anim->giveRgbBuffer();
|
|
|
|
if ((_currentFrame > 397) && (_currentFrame < 444)) { // Broken Sword Logo
|
|
|
|
if (_currentFrame <= 403)
|
|
|
|
insertOverlay(buf, _logoOvls[_currentFrame - 398], _introPal); // fade up
|
|
|
|
else if (_currentFrame <= 437)
|
|
|
|
insertOverlay(buf, _logoOvls[(_currentFrame - 404) % 6 + 6], _introPal); // animation
|
|
|
|
else {
|
|
|
|
insertOverlay(buf, _logoOvls[5 - (_currentFrame - 438)], _introPal); // fade down
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2007-04-28 07:27:53 +00:00
|
|
|
AnimationState::AnimationState(MoviePlayer *player, Screen *screen, OSystem *system)
|
|
|
|
: BaseAnimationState(system, 640, 400), _player(player), _screen(screen) {
|
2006-08-26 11:34:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
AnimationState::~AnimationState(void) {
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef BACKEND_8BIT
|
|
|
|
void AnimationState::setPalette(byte *pal) {
|
|
|
|
_player->updatePalette(pal, false);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
void AnimationState::drawYUV(int width, int height, byte *const *dat) {
|
|
|
|
_frameWidth = width;
|
|
|
|
_frameHeight = height;
|
|
|
|
|
|
|
|
#ifdef BACKEND_8BIT
|
2007-04-28 07:27:53 +00:00
|
|
|
_screen->plotYUV(_lut, width, height, dat);
|
2006-08-26 11:34:35 +00:00
|
|
|
#else
|
|
|
|
plotYUV(width, height, dat);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
OverlayColor *AnimationState::giveRgbBuffer(void) {
|
|
|
|
#ifdef BACKEND_8BIT
|
|
|
|
return NULL;
|
|
|
|
#else
|
|
|
|
return _overlay;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
Audio::AudioStream *AnimationState::createAudioStream(const char *name, void *arg) {
|
|
|
|
if (arg)
|
|
|
|
return (Audio::AudioStream*)arg;
|
|
|
|
else
|
|
|
|
return Audio::AudioStream::openStreamFile(name);
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
|
|
// Factory function for creating the appropriate cutscene player
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
2008-09-27 17:47:22 +00:00
|
|
|
MoviePlayer *makeMoviePlayer(uint32 id, SwordEngine *vm, Screen *screen, Text *textMan, Audio::Mixer *snd, OSystem *system) {
|
2006-08-26 11:34:35 +00:00
|
|
|
#if defined(USE_ZLIB) || defined(USE_MPEG2)
|
|
|
|
char filename[20];
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef USE_ZLIB
|
|
|
|
snprintf(filename, sizeof(filename), "%s.dxa", sequenceList[id]);
|
|
|
|
|
|
|
|
if (Common::File::exists(filename)) {
|
2008-09-27 17:47:22 +00:00
|
|
|
return new MoviePlayerDXA(vm, screen, textMan, snd, system);
|
2006-08-26 11:34:35 +00:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef USE_MPEG2
|
|
|
|
snprintf(filename, sizeof(filename), "%s.mp2", sequenceList[id]);
|
|
|
|
|
|
|
|
if (Common::File::exists(filename)) {
|
2008-09-27 17:47:22 +00:00
|
|
|
return new MoviePlayerMPEG(vm, screen, textMan, snd, system);
|
2006-08-26 11:34:35 +00:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
2004-12-05 02:55:06 +00:00
|
|
|
|
2004-07-11 04:41:48 +00:00
|
|
|
} // End of namespace Sword1
|