2011-09-20 10:56:51 -04: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.
|
|
|
|
*
|
|
|
|
* Additional copyright for this file:
|
|
|
|
* Copyright (C) 1995-1997 Presto Studios, Inc.
|
|
|
|
*
|
|
|
|
* 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.
|
2016-09-03 12:46:38 +02:00
|
|
|
*
|
2011-09-20 10:56:51 -04:00
|
|
|
* 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.
|
2016-09-03 12:46:38 +02:00
|
|
|
*
|
2011-09-20 10:56:51 -04:00
|
|
|
* 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.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "common/system.h"
|
|
|
|
#include "graphics/surface.h"
|
|
|
|
#include "video/qt_decoder.h"
|
|
|
|
#include "video/video_decoder.h"
|
|
|
|
|
|
|
|
#include "pegasus/movie.h"
|
|
|
|
|
|
|
|
namespace Pegasus {
|
|
|
|
|
2011-12-16 14:17:50 -05:00
|
|
|
Movie::Movie(const DisplayElementID id) : Animation(id) {
|
2011-09-20 10:56:51 -04:00
|
|
|
_video = 0;
|
|
|
|
setScale(600);
|
|
|
|
}
|
|
|
|
|
|
|
|
Movie::~Movie() {
|
|
|
|
releaseMovie();
|
|
|
|
}
|
|
|
|
|
2011-12-20 18:26:25 -05:00
|
|
|
// *** Make sure this will stop displaying the movie.
|
2011-09-20 10:56:51 -04:00
|
|
|
|
|
|
|
void Movie::releaseMovie() {
|
|
|
|
if (_video) {
|
|
|
|
delete _video;
|
|
|
|
_video = 0;
|
|
|
|
disposeAllCallBacks();
|
|
|
|
deallocateSurface();
|
|
|
|
}
|
2011-10-28 15:37:17 -04:00
|
|
|
|
|
|
|
setBounds(Common::Rect(0, 0, 0, 0));
|
2011-09-20 10:56:51 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
void Movie::initFromMovieFile(const Common::String &fileName, bool transparent) {
|
|
|
|
_transparent = transparent;
|
|
|
|
|
|
|
|
releaseMovie();
|
|
|
|
_video = new Video::QuickTimeDecoder();
|
2011-10-20 17:35:25 -04:00
|
|
|
if (!_video->loadFile(fileName)) {
|
|
|
|
// Replace any colon with an underscore, since only Mac OS X
|
|
|
|
// supports that. See PegasusEngine::detectOpeningClosingDirectory()
|
|
|
|
// for more info.
|
|
|
|
Common::String newName(fileName);
|
|
|
|
if (newName.contains(':'))
|
|
|
|
for (uint i = 0; i < newName.size(); i++)
|
|
|
|
if (newName[i] == ':')
|
2012-04-06 09:46:04 -04:00
|
|
|
newName.setChar('_', i);
|
2011-10-20 17:35:25 -04:00
|
|
|
|
|
|
|
if (!_video->loadFile(newName))
|
|
|
|
error("Could not load video '%s'", fileName.c_str());
|
|
|
|
}
|
2011-09-20 10:56:51 -04:00
|
|
|
|
|
|
|
Common::Rect bounds(0, 0, _video->getWidth(), _video->getHeight());
|
2011-09-28 20:16:50 -04:00
|
|
|
sizeElement(_video->getWidth(), _video->getHeight());
|
2011-09-23 13:44:40 -04:00
|
|
|
_movieBox = bounds;
|
2011-09-20 10:56:51 -04:00
|
|
|
|
2011-09-28 20:16:50 -04:00
|
|
|
if (!isSurfaceValid())
|
|
|
|
allocateSurface(bounds);
|
|
|
|
|
2012-09-12 20:18:53 -04:00
|
|
|
setStart(0, getScale());
|
2012-08-31 22:06:56 -04:00
|
|
|
TimeBase::setStop(_video->getDuration().convertToFramerate(getScale()).totalNumberOfFrames(), getScale());
|
2011-09-20 10:56:51 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
void Movie::redrawMovieWorld() {
|
2011-10-30 19:23:33 -04:00
|
|
|
if (_video && _video->needsUpdate()) {
|
2011-09-20 10:56:51 -04:00
|
|
|
const Graphics::Surface *frame = _video->decodeNextFrame();
|
|
|
|
|
2011-09-22 10:53:53 -04:00
|
|
|
if (!frame)
|
|
|
|
return;
|
|
|
|
|
2012-06-16 01:47:36 -04:00
|
|
|
// Make sure we have a surface in the current pixel format
|
|
|
|
Graphics::Surface *convertedFrame = 0;
|
|
|
|
|
|
|
|
if (frame->format != g_system->getScreenFormat()) {
|
|
|
|
convertedFrame = frame->convertTo(g_system->getScreenFormat());
|
|
|
|
frame = convertedFrame;
|
|
|
|
}
|
|
|
|
|
2011-10-08 00:13:14 -04:00
|
|
|
// Copy to the surface using _movieBox
|
|
|
|
uint16 width = MIN<int>(frame->w, _movieBox.width());
|
|
|
|
uint16 height = MIN<int>(frame->h, _movieBox.height());
|
|
|
|
|
|
|
|
for (uint16 y = 0; y < height; y++)
|
|
|
|
memcpy((byte *)_surface->getBasePtr(_movieBox.left, _movieBox.top + y), (const byte *)frame->getBasePtr(0, y), width * frame->format.bytesPerPixel);
|
2011-09-20 13:04:27 -04:00
|
|
|
|
2012-06-16 01:47:36 -04:00
|
|
|
if (convertedFrame) {
|
|
|
|
convertedFrame->free();
|
|
|
|
delete convertedFrame;
|
|
|
|
}
|
|
|
|
|
2011-09-20 13:04:27 -04:00
|
|
|
triggerRedraw();
|
2011-09-20 10:56:51 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-09-26 04:10:32 +02:00
|
|
|
void Movie::draw(const Common::Rect &r) {
|
2011-09-28 20:16:50 -04:00
|
|
|
Common::Rect worldBounds = _movieBox;
|
|
|
|
Common::Rect elementBounds;
|
|
|
|
getBounds(elementBounds);
|
2011-09-21 21:39:07 -04:00
|
|
|
|
2011-09-28 20:16:50 -04:00
|
|
|
worldBounds.moveTo(elementBounds.left, elementBounds.top);
|
|
|
|
Common::Rect r1 = r.findIntersectingRect(worldBounds);
|
2011-09-21 21:39:07 -04:00
|
|
|
|
|
|
|
Common::Rect r2 = r1;
|
2011-09-28 20:16:50 -04:00
|
|
|
r2.translate(_movieBox.left - elementBounds.left, _movieBox.top - elementBounds.top);
|
2011-09-21 21:39:07 -04:00
|
|
|
drawImage(r2, r1);
|
2011-09-20 10:56:51 -04:00
|
|
|
}
|
|
|
|
|
2011-12-16 14:17:50 -05:00
|
|
|
void Movie::moveMovieBoxTo(const CoordType h, const CoordType v) {
|
2011-09-23 13:44:40 -04:00
|
|
|
_movieBox.moveTo(h, v);
|
|
|
|
}
|
|
|
|
|
2012-08-31 22:06:56 -04:00
|
|
|
void Movie::setStop(const TimeValue stopTime, const TimeScale scale) {
|
|
|
|
TimeBase::setStop(stopTime, scale);
|
|
|
|
|
|
|
|
if (_video)
|
|
|
|
_video->setEndTime(Audio::Timestamp(0, _stopTime, _stopScale));
|
|
|
|
}
|
|
|
|
|
2011-09-20 10:56:51 -04:00
|
|
|
void Movie::setVolume(uint16 volume) {
|
2012-05-28 17:04:56 -04:00
|
|
|
if (_video)
|
|
|
|
_video->setVolume(MIN<uint>(volume, 0xFF));
|
2011-09-20 10:56:51 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
void Movie::setTime(const TimeValue time, const TimeScale scale) {
|
|
|
|
if (_video) {
|
2011-09-22 10:53:53 -04:00
|
|
|
// Don't go past the ends of the movie
|
|
|
|
Common::Rational timeFrac = Common::Rational(time, ((scale == 0) ? getScale() : scale));
|
|
|
|
|
|
|
|
if (timeFrac < Common::Rational(_startTime, _startScale))
|
|
|
|
timeFrac = Common::Rational(_startTime, _startScale);
|
|
|
|
else if (timeFrac >= Common::Rational(_stopTime, _stopScale))
|
|
|
|
return;
|
|
|
|
|
2012-08-26 15:49:45 -04:00
|
|
|
_video->seek(Audio::Timestamp(0, timeFrac.getNumerator(), timeFrac.getDenominator()));
|
2011-09-22 10:53:53 -04:00
|
|
|
_time = timeFrac;
|
2011-10-02 23:16:30 -04:00
|
|
|
_lastMillis = 0;
|
2011-09-20 10:56:51 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Movie::setRate(const Common::Rational rate) {
|
2012-11-24 01:04:13 -05:00
|
|
|
if (_video) {
|
|
|
|
_video->setRate(rate);
|
|
|
|
|
|
|
|
TimeBase::setRate(_video->getRate());
|
2011-10-22 22:29:09 -04:00
|
|
|
return;
|
|
|
|
}
|
2011-09-20 13:04:27 -04:00
|
|
|
|
|
|
|
TimeBase::setRate(rate);
|
2011-09-20 10:56:51 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
void Movie::start() {
|
2012-08-26 15:49:45 -04:00
|
|
|
if (_video)
|
|
|
|
_video->start();
|
2011-09-20 10:56:51 -04:00
|
|
|
|
|
|
|
TimeBase::start();
|
|
|
|
}
|
|
|
|
|
|
|
|
void Movie::stop() {
|
2012-08-26 15:49:45 -04:00
|
|
|
if (_video)
|
|
|
|
_video->stop();
|
2011-09-20 10:56:51 -04:00
|
|
|
|
|
|
|
TimeBase::stop();
|
|
|
|
}
|
|
|
|
|
|
|
|
void Movie::resume() {
|
2012-09-22 18:13:31 -04:00
|
|
|
if (_paused) {
|
|
|
|
if (_video)
|
|
|
|
_video->pauseVideo(false);
|
2011-09-20 10:56:51 -04:00
|
|
|
|
2012-09-22 18:13:31 -04:00
|
|
|
_paused = false;
|
|
|
|
}
|
2011-09-20 10:56:51 -04:00
|
|
|
}
|
|
|
|
|
2011-09-30 16:26:31 -04:00
|
|
|
void Movie::pause() {
|
2012-09-22 18:13:31 -04:00
|
|
|
if (isRunning() && !_paused) {
|
|
|
|
if (_video)
|
|
|
|
_video->pauseVideo(true);
|
2011-09-30 16:26:31 -04:00
|
|
|
|
2012-09-22 18:13:31 -04:00
|
|
|
_paused = true;
|
|
|
|
_pauseStart = g_system->getMillis();
|
|
|
|
}
|
2011-09-30 16:26:31 -04:00
|
|
|
}
|
|
|
|
|
2011-10-07 23:01:53 -04:00
|
|
|
TimeValue Movie::getDuration(const TimeScale scale) const {
|
|
|
|
// Unlike TimeBase::getDuration(), this returns the whole duration of the movie
|
|
|
|
// The original source has a TODO to make this behave like TimeBase::getDuration(),
|
|
|
|
// but the problem is that too much code requires this function to behave this way...
|
|
|
|
|
|
|
|
if (_video)
|
2012-08-26 15:49:45 -04:00
|
|
|
return _video->getDuration().convertToFramerate(((scale == 0) ? getScale() : scale)).totalNumberOfFrames();
|
2011-10-07 23:01:53 -04:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2011-10-09 16:14:14 -04:00
|
|
|
void Movie::updateTime() {
|
|
|
|
// The reason why we overrode TimeBase's updateTime():
|
2011-09-21 20:22:26 -04:00
|
|
|
// Again, avoiding timers and handling it here
|
2012-08-31 22:06:56 -04:00
|
|
|
if (_video && _video->isPlaying() && !_video->isPaused()) {
|
2011-10-30 19:23:33 -04:00
|
|
|
redrawMovieWorld();
|
2011-09-22 10:39:31 -04:00
|
|
|
|
|
|
|
uint32 startTime = _startTime * getScale() / _startScale;
|
|
|
|
uint32 stopTime = _stopTime * getScale() / _stopScale;
|
2012-05-25 00:21:51 -04:00
|
|
|
uint32 actualTime = CLIP<int>(_video->getTime() * getScale() / 1000, startTime, stopTime);
|
2012-08-31 22:06:56 -04:00
|
|
|
|
|
|
|
// HACK: Due to the inaccuracy of the time, we won't actually allow us to hit
|
|
|
|
// the stop time unless we've actually reached the end of the segment.
|
|
|
|
if (actualTime == stopTime && !_video->endOfVideo())
|
|
|
|
actualTime--;
|
|
|
|
|
2011-09-22 10:39:31 -04:00
|
|
|
_time = Common::Rational(actualTime, getScale());
|
|
|
|
}
|
2011-09-20 10:56:51 -04:00
|
|
|
}
|
|
|
|
|
2011-12-16 14:17:50 -05:00
|
|
|
GlowingMovie::GlowingMovie(const DisplayElementID id) : Movie(id) {
|
2011-10-23 12:51:33 -04:00
|
|
|
_glowing = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
void GlowingMovie::draw(const Common::Rect &r) {
|
|
|
|
// Make sure the rectangles are clipped properly, OR guarantee that _bounds will
|
|
|
|
// never fall off the screen.
|
|
|
|
if (_glowing) {
|
|
|
|
Common::Rect bounds;
|
|
|
|
getBounds(bounds);
|
|
|
|
|
|
|
|
copyToCurrentPortTransparentGlow(_movieBox, bounds);
|
|
|
|
} else {
|
|
|
|
Movie::draw(r);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void GlowingMovie::setBounds(const Common::Rect &r) {
|
|
|
|
Common::Rect bounds;
|
|
|
|
getBounds(bounds);
|
|
|
|
|
|
|
|
if (r != bounds) {
|
|
|
|
// Avoid Movie::setBounds.
|
|
|
|
// clone2727 asks why, but goes along with it
|
|
|
|
Animation::setBounds(r);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-12-16 14:17:50 -05:00
|
|
|
ScalingMovie::ScalingMovie(const DisplayElementID id) : GlowingMovie(id) {
|
2011-10-23 12:51:33 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
void ScalingMovie::draw(const Common::Rect &) {
|
|
|
|
// Make sure the rectangles are clipped properly, OR guarantee that _bounds will
|
|
|
|
// never fall off the screen.
|
|
|
|
|
|
|
|
Common::Rect bounds;
|
|
|
|
getBounds(bounds);
|
|
|
|
|
|
|
|
if (_glowing)
|
|
|
|
scaleTransparentCopyGlow(_movieBox, bounds);
|
|
|
|
else
|
|
|
|
scaleTransparentCopy(_movieBox, bounds);
|
|
|
|
}
|
|
|
|
|
2011-09-20 10:56:51 -04:00
|
|
|
} // End of namespace Pegasus
|