2019-06-06 07:55:32 +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.
|
|
|
|
*
|
|
|
|
* 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.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "common/system.h"
|
|
|
|
|
2019-07-16 17:30:38 +00:00
|
|
|
#include "petka/sound.h"
|
2019-06-17 16:41:55 +00:00
|
|
|
#include "petka/flc.h"
|
2019-06-06 07:55:32 +00:00
|
|
|
#include "petka/petka.h"
|
2019-06-07 08:01:16 +00:00
|
|
|
#include "petka/q_system.h"
|
2019-07-16 17:30:38 +00:00
|
|
|
#include "petka/interfaces/main.h"
|
|
|
|
#include "petka/interfaces/dialog_interface.h"
|
2019-06-20 19:06:52 +00:00
|
|
|
#include "petka/objects/object.h"
|
2019-06-07 08:01:16 +00:00
|
|
|
#include "petka/video.h"
|
2019-06-06 07:55:32 +00:00
|
|
|
|
|
|
|
namespace Petka {
|
|
|
|
|
2020-10-02 18:48:29 +00:00
|
|
|
// COMPLETED
|
|
|
|
|
2020-05-20 18:58:51 +00:00
|
|
|
const uint kShakeTime = 30;
|
|
|
|
const int kShakeOffset = 3;
|
|
|
|
|
2020-06-05 18:01:15 +00:00
|
|
|
VideoSystem::VideoSystem(PetkaEngine &vm) :
|
|
|
|
_vm(vm), _shake(false), _shift(false), _shakeTime(0), _time(0) {
|
2019-07-03 19:28:12 +00:00
|
|
|
makeAllDirty();
|
2019-07-07 17:54:30 +00:00
|
|
|
_time = g_system->getMillis();
|
2020-05-24 18:53:21 +00:00
|
|
|
_allowAddingRects = true;
|
2019-06-07 08:01:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void VideoSystem::update() {
|
2020-10-02 18:48:29 +00:00
|
|
|
QSystem *sys = _vm.getQSystem();
|
|
|
|
Interface *interface = sys->_currInterface;
|
2020-05-20 18:58:51 +00:00
|
|
|
uint32 time = g_system->getMillis();
|
2019-06-07 08:01:16 +00:00
|
|
|
if (interface) {
|
2020-10-02 18:48:29 +00:00
|
|
|
if (sys->_currInterface == sys->_mainInterface.get()) {
|
|
|
|
int xOff = sys->_xOffset;
|
2020-10-02 19:34:37 +00:00
|
|
|
int reqOffset = sys->_reqOffset;
|
|
|
|
if (xOff != reqOffset && ((xOff != sys->_sceneWidth - 640 && xOff < reqOffset) || (xOff > 0 && xOff > reqOffset))) {
|
|
|
|
if (xOff <= reqOffset) {
|
2020-10-02 18:48:29 +00:00
|
|
|
xOff += 8;
|
2020-10-02 19:34:37 +00:00
|
|
|
xOff = MIN<int>(xOff, reqOffset);
|
2020-10-02 18:48:29 +00:00
|
|
|
} else {
|
|
|
|
xOff -= 8;
|
2020-10-02 19:34:37 +00:00
|
|
|
xOff = MAX<int>(xOff, reqOffset);
|
2020-10-02 18:48:29 +00:00
|
|
|
}
|
|
|
|
sys->_xOffset = CLIP(xOff, 0, sys->_sceneWidth - 640);
|
|
|
|
}
|
|
|
|
makeAllDirty();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-05-25 08:44:53 +00:00
|
|
|
for (uint i = interface->_startIndex; i < interface->_objs.size(); ++i) {
|
2019-06-30 15:11:13 +00:00
|
|
|
interface->_objs[i]->update(time - _time);
|
2019-06-09 15:09:38 +00:00
|
|
|
}
|
2019-06-28 08:13:19 +00:00
|
|
|
|
|
|
|
for (uint i = 0; i < interface->_objs.size(); ++i) {
|
|
|
|
interface->_objs[i]->updateZ();
|
|
|
|
}
|
|
|
|
|
2020-05-20 15:36:58 +00:00
|
|
|
sort();
|
2019-06-28 08:13:19 +00:00
|
|
|
|
2020-05-24 18:53:21 +00:00
|
|
|
mergeDirtyRects();
|
|
|
|
|
|
|
|
_allowAddingRects = false;
|
2019-06-07 08:01:16 +00:00
|
|
|
for (uint i = 0; i < interface->_objs.size(); ++i) {
|
|
|
|
interface->_objs[i]->draw();
|
|
|
|
}
|
2020-05-24 18:53:21 +00:00
|
|
|
_allowAddingRects = true;
|
|
|
|
|
|
|
|
for (Common::List<Common::Rect>::iterator i = _dirtyRects.begin(); i != _dirtyRects.end(); ++i) {
|
|
|
|
const Common::Rect &r = *i;
|
|
|
|
const byte *srcP = (const byte *)getBasePtr(r.left, r.top);
|
|
|
|
g_system->copyRectToScreen(srcP, pitch, r.left, r.top,
|
|
|
|
r.width(), r.height());
|
|
|
|
}
|
|
|
|
|
|
|
|
_dirtyRects.clear();
|
2019-06-07 08:01:16 +00:00
|
|
|
}
|
2020-05-20 18:58:51 +00:00
|
|
|
|
2019-06-30 15:11:13 +00:00
|
|
|
_time = time;
|
2019-06-06 07:55:32 +00:00
|
|
|
|
|
|
|
if (_shake) {
|
2020-05-20 18:58:51 +00:00
|
|
|
g_system->setShakePos(_shift ? kShakeOffset : 0, 0);
|
|
|
|
if (time - _shakeTime > kShakeTime) {
|
2019-06-06 07:55:32 +00:00
|
|
|
_shift = !_shift;
|
|
|
|
_shakeTime = time;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-05-24 18:53:21 +00:00
|
|
|
g_system->updateScreen();
|
2019-06-06 07:55:32 +00:00
|
|
|
}
|
|
|
|
|
2020-05-24 18:53:21 +00:00
|
|
|
void VideoSystem::addDirtyRect(const Common::Rect &rect) {
|
|
|
|
if (_allowAddingRects) {
|
|
|
|
Graphics::Screen::addDirtyRect(rect);
|
|
|
|
}
|
2019-06-06 07:55:32 +00:00
|
|
|
}
|
|
|
|
|
2020-05-24 18:53:21 +00:00
|
|
|
void VideoSystem::addDirtyRect(Common::Point pos, Common::Rect rect) {
|
|
|
|
rect.translate(pos.x, pos.y);
|
|
|
|
addDirtyRect(rect);
|
2019-06-06 07:55:32 +00:00
|
|
|
}
|
|
|
|
|
2019-06-17 16:41:55 +00:00
|
|
|
void VideoSystem::addDirtyRect(Common::Point pos, FlicDecoder &flc) {
|
2020-10-01 16:32:49 +00:00
|
|
|
pos.x = pos.x - g_vm->getQSystem()->_xOffset;
|
2020-05-24 18:53:21 +00:00
|
|
|
addDirtyRect(pos, flc.getBounds());
|
2019-06-17 16:41:55 +00:00
|
|
|
}
|
|
|
|
|
2020-05-24 18:53:21 +00:00
|
|
|
void VideoSystem::addDirtyMskRects(Common::Point pos, FlicDecoder &flc) {
|
2019-07-07 17:19:47 +00:00
|
|
|
const Common::Array<Common::Rect> &rects = flc.getMskRects();
|
|
|
|
for (uint i = 0; i < rects.size(); ++i) {
|
2020-05-24 18:53:21 +00:00
|
|
|
addDirtyRect(pos, rects[i]);
|
2019-07-07 17:19:47 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-05-24 18:53:21 +00:00
|
|
|
void VideoSystem::addDirtyMskRects(FlicDecoder &flc) {
|
|
|
|
addDirtyMskRects(Common::Point(0, 0), flc);
|
|
|
|
}
|
|
|
|
|
|
|
|
const Common::List<Common::Rect> VideoSystem::rects() const {
|
|
|
|
return _dirtyRects;
|
2019-07-03 19:28:12 +00:00
|
|
|
}
|
|
|
|
|
2019-07-07 17:54:30 +00:00
|
|
|
void VideoSystem::updateTime() {
|
|
|
|
_time = g_system->getMillis();
|
|
|
|
}
|
|
|
|
|
2020-05-17 18:33:47 +00:00
|
|
|
void VideoSystem::setShake(bool shake) {
|
|
|
|
_shake = shake;
|
2020-05-20 18:58:51 +00:00
|
|
|
g_system->setShakePos(0, 0);
|
2020-05-17 18:33:47 +00:00
|
|
|
}
|
|
|
|
|
2020-05-20 15:36:58 +00:00
|
|
|
void VideoSystem::sort() {
|
2020-06-05 18:01:15 +00:00
|
|
|
Common::Array<QVisibleObject *> &objs = _vm.getQSystem()->_currInterface->_objs;
|
2020-05-20 15:36:58 +00:00
|
|
|
for (uint i = 0; i < objs.size() - 1; ++i) {
|
2020-05-21 20:38:56 +00:00
|
|
|
uint minIndex = i;
|
2020-05-20 15:36:58 +00:00
|
|
|
for (uint j = i + 1; j < objs.size(); ++j) {
|
|
|
|
if (objs[j]->_z < objs[minIndex]->_z) {
|
|
|
|
minIndex = j;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (i != minIndex) {
|
|
|
|
QVisibleObject *tmp = objs[i];
|
|
|
|
objs[i] = objs[minIndex];
|
|
|
|
objs[minIndex] = tmp;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-05-16 20:51:35 +00:00
|
|
|
}
|