scummvm/engines/access/video.cpp
2023-12-24 13:19:25 +01:00

197 lines
5.3 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 3 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, see <http://www.gnu.org/licenses/>.
*
*/
#include "access/video.h"
#include "access/access.h"
namespace Access {
VideoPlayer::VideoPlayer(AccessEngine *vm) : Manager(vm) {
_vidSurface = nullptr;
_videoData = nullptr;
_startCoord = nullptr;
_frameCount = 0;
_xCount = 0;
_scanCount = 0;
_frameSize = 0;
_videoFrame = 0;
_soundFlag = false;
_soundFrame = 0;
_videoEnd = false;
_header._frameCount = 0;
_header._width = _header._height = 0;
_header._flags = VIDEOFLAG_NONE;
}
VideoPlayer::~VideoPlayer() {
closeVideo();
}
void VideoPlayer::setVideo(BaseSurface *vidSurface, const Common::Point &pt, int rate) {
_vidSurface = vidSurface;
vidSurface->_orgX1 = pt.x;
vidSurface->_orgY1 = pt.y;
_vm->_timers[31]._timer = rate;
_vm->_timers[31]._initTm = rate;
// Load in header
_header._frameCount = _videoData->_stream->readUint16LE();
_header._width = _videoData->_stream->readUint16LE();
_header._height = _videoData->_stream->readUint16LE();
_videoData->_stream->skip(1);
_header._flags = (VideoFlags)_videoData->_stream->readByte();
_startCoord = (byte *)vidSurface->getBasePtr(pt.x, pt.y);
_frameCount = _header._frameCount - 2;
_xCount = _header._width;
_scanCount = _header._height;
_videoFrame = 0;
_videoBounds = Common::Rect(pt.x, pt.y, pt.x + _header._width, pt.y + _header._height);
getFrame();
if (_header._flags == VIDEOFLAG_BG) {
// Draw the background
for (int y = 0; y < _scanCount; ++y) {
byte *pDest = (byte *)vidSurface->getBasePtr(pt.x, pt.y + y);
_videoData->_stream->read(pDest, _xCount);
}
if (vidSurface == _vm->_screen)
_vm->_newRects.push_back(Common::Rect(pt.x, pt.y, pt.x + _xCount, pt.y + _scanCount));
getFrame();
}
_videoEnd = false;
}
void VideoPlayer::setVideo(BaseSurface *vidSurface, const Common::Point &pt, const Common::Path &filename, int rate) {
// Open up video stream
_videoData = _vm->_files->loadFile(filename);
setVideo(vidSurface, pt, rate);
}
void VideoPlayer::setVideo(BaseSurface *vidSurface, const Common::Point &pt, const FileIdent &videoFile, int rate) {
// Open up video stream
_videoData = _vm->_files->loadFile(videoFile);
setVideo(vidSurface, pt, rate);
}
void VideoPlayer::closeVideo() {
delete _videoData;
_videoData = nullptr;
}
void VideoPlayer::getFrame() {
_frameSize = _videoData->_stream->readUint16LE();
}
void VideoPlayer::playVideo() {
if (_vm->_timers[31]._flag)
return;
++_vm->_timers[31]._flag;
byte *pDest = _startCoord;
byte *pLine = _startCoord;
uint32 frameEnd = _videoData->_stream->pos() + _frameSize;
while ((uint32)_videoData->_stream->pos() < frameEnd) {
int count = _videoData->_stream->readByte();
if (count & 0x80) {
count &= 0x7f;
// Skip count number of pixels
// Loop across lines if necessary
while (count >= (pLine + _xCount - pDest)) {
count -= (pLine + _xCount - pDest);
pLine += _vidSurface->pitch;
pDest = pLine;
}
// Skip any remaining pixels in the new line
pDest += count;
} else {
// Read count number of pixels
// Load across lines if necessary
while (count >= (pLine + _xCount - pDest)) {
int lineCount = (pLine + _xCount - pDest);
_videoData->_stream->read(pDest, lineCount);
count -= lineCount;
pLine += _vidSurface->pitch;
pDest = pLine;
}
// Load remainder of pixels on line
if (count > 0) {
_videoData->_stream->read(pDest, count);
pDest += count;
}
}
}
// If the video is playing on the screen surface, add a dirty rect
if (_vidSurface == _vm->_screen)
_vm->_screen->markAllDirty();
getFrame();
if (++_videoFrame == _frameCount) {
closeVideo();
_videoEnd = true;
}
}
void VideoPlayer::copyVideo() {
_vm->_player->calcPlayer();
// Figure out the dirty rect area for the video frame
Common::Rect r = Common::Rect(_vm->_vidX - _vm->_screen->_bufferStart.x,
_vm->_vidY - _vm->_screen->_bufferStart.y,
_vm->_vidX - _vm->_screen->_bufferStart.x + _header._width,
_vm->_vidY - _vm->_screen->_bufferStart.y + _header._height);
if (!_vm->_screen->clip(r))
return;
_vm->_newRects.push_back(r);
int vh = _header._height;
int vw = _header._width;
int destIdx = _vm->_vidX - _vm->_screen->_bufferStart.x;
int srcIdx = _vm->_screen->_leftSkip;
for (int i = 0; i < _vm->_screen->_topSkip; i++)
destIdx += 160;
const byte *srcP = (const byte *)_vm->_vidBuf.getPixels() + srcIdx;
byte *destP = (byte *)_vm->_buffer2.getPixels() + destIdx;
for (int i = 0; i < vh; i++) {
Common::copy(srcP, srcP + vw, destP);
srcP += _vm->_vidBuf.pitch;
destP += _vm->_buffer2.pitch;
}
}
} // End of namespace Access