scummvm/engines/petka/objects/heroes.cpp

428 lines
10 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 "common/system.h"
#include "petka/petka.h"
#include "petka/q_manager.h"
#include "petka/q_system.h"
#include "petka/flc.h"
#include "petka/video.h"
#include "petka/sound.h"
#include "petka/objects/heroes.h"
#include "petka/interfaces/panel.h"
namespace Petka {
QObjectPetka::QObjectPetka()
: _walk(nullptr) {
_field7C = 1;
_reaction = nullptr;
_heroReaction = nullptr;
_sender = nullptr;
_isWalking = false;
_x = 574;
_y = 444;
_z = 200;
// _surfId = -5;
_surfH = 0;
_surfW = 0;
_k = 1.0;
_x_ = _y_ = 0;
_destX = _destY = 0;
_imageId = 0;
_fieldB4 = 0;
}
void QObjectPetka::processMessage(const QMessage &arg) {
QMessage msg = arg;
if (msg.opcode == kImage) {
msg.opcode = kSet;
_imageId = msg.arg1;
_walk.reset(new Walk(_imageId + 10));
QObjectBG *room = g_vm->getQSystem()->_room;
if (room)
_walk->setBackground(g_vm->resMgr()->findResourceName(room->_resourceId));
}
if (msg.opcode == kSaid || msg.opcode == kStand) {
msg.opcode = kSet;
msg.arg1 = _imageId;
msg.arg2 = 1;
}
if (msg.opcode == kSay) {
msg.opcode = kSet;
msg.arg1 = _imageId + 1;
msg.arg2 = 1;
}
if (msg.opcode == kSet || msg.opcode == kPlay) {
_field7C = msg.arg1 == _imageId || msg.opcode == kPlay;
}
if (msg.opcode != kWalk) {
if (msg.opcode == kWalked && _heroReaction) {
QReaction *reaction = _heroReaction;
_heroReaction = nullptr;
_sender->processReaction(reaction);
}
QMessageObject::processMessage(msg);
if (msg.opcode == kSet || msg.opcode == kPlay) {
initSurface();
if (!g_vm->getQSystem()->_totalInit) {
setPos(Common::Point(_x_, _y_), false);
}
}
}
}
void QObjectPetka::initSurface() {
QManager *resMgr = g_vm->resMgr();
FlicDecoder *flc = resMgr->getFlic(_resourceId);
_surfW = flc->getWidth() * _k;
_surfH = flc->getHeight() * _k;
}
void QObjectPetka::walk(int x, int y) {
Common::Point walkPos(x, y);
if (!_isShown) {
setPos(walkPos, false);
return;
}
Common::Point currPos;
if (_isWalking) {
currPos = _walk->currPos();
} else {
currPos.x = _x_;
currPos.y = _y_;
}
if (currPos.sqrDist(walkPos) >= 25) {
_walk->init(currPos, walkPos);
_destX = x;
_destY = y;
_resourceId = _imageId + _walk->getSpriteId() + 10;
_isWalking = true;
_animate = true;
initSurface();
FlicDecoder *flc = g_vm->resMgr()->getFlic(_resourceId);
flc->setFrame(1);
sub_408940();
g_vm->videoSystem()->makeAllDirty();
_field7C = 0;
_time = 0;
_holdMessages = true;
}
}
void QObjectPetka::draw() {
if (!_isShown || _resourceId == -1) {
return;
}
if (_animate && _startSound) {
if (_sound) {
_sound->play(_loopedSound);
if (_loopedSound) {
_sound = nullptr;
}
}
_startSound = false;
}
FlicDecoder *flc = g_vm->resMgr()->getFlic(_resourceId);
if (!flc) {
return;
}
Graphics::Surface *conv = flc->getCurrentFrame()->convertTo(g_system->getScreenFormat(), flc->getPalette());
Common::Rect srcRect(0, 0, conv->w, conv->h);
Common::Rect dstRect(0, 0, _surfW, _surfH);
dstRect.translate(_x - g_vm->getQSystem()->_xOffset, _y);
g_vm->videoSystem()->transBlitFrom(*conv, srcRect, dstRect, flc->getTransColor(conv->format));
conv->free();
delete conv;
}
void QObjectPetka::setPos(Common::Point p, bool) {
QSystem *sys = g_vm->getQSystem();
int xOff = sys->_xOffset;
Common::Rect dirty(_x - xOff, _y, _surfW + _x - xOff, _surfH + _y);
g_vm->videoSystem()->addDirtyRect(dirty);
p.y = MIN<int16>(p.y, 480);
FlicDecoder *flc = g_vm->resMgr()->getFlic(_resourceId);
_k = calcPerspective(p.y);
_surfH = flc->getHeight() * _k;
_surfW = flc->getWidth() * _k;
_x_ = p.x;
_y_ = p.y;
_x = p.x - _surfW / 2;
_y = p.y - _surfH;
recalcOffset();
g_vm->videoSystem()->addDirtyRect(Common::Rect(_x - xOff, _y, _surfW + _x - xOff, _surfH + _y));
}
double QObjectPetka::calcPerspective(int y) {
QSystem *qsys = g_vm->getQSystem();
y = MIN(y, 480);
const Perspective &pers = qsys->_room->_persp;
double res = (y - pers.y0) * pers.k / (pers.y1 - pers.y0);
if (res < 0.0)
res = 0.0;
if (res + pers.f0 > pers.f1)
return pers.f1;
return res + pers.f0;
}
void QObjectPetka::updateWalk() {
if (!_isWalking)
return;
int v = _walk->sub_423350();
switch (v) {
case 0: {
_isWalking = false;
setPos(Common::Point(_walk->destX, _walk->destY), false);
QMessage msg(_id, kSet, (uint16) _imageId, 1, 0, nullptr, 0);
if (_heroReaction) {
uint i;
for (i = 0; i < _heroReaction->messages.size(); ++i) {
if (_heroReaction->messages[i].opcode == kGoTo || _heroReaction->messages[i].opcode == kSetSeq) {
_resourceId = _imageId + _walk->getSpriteId() + 10;
FlicDecoder *flc = g_vm->resMgr()->getFlic(_resourceId);
flc->setFrame(1);
initSurface();
processMessage(QMessage(_id, kAnimate, 0, 0, 0, nullptr, 0));
_heroReaction->messages.push_back(msg);
_heroReaction->messages.push_back(QMessage(_id, kAnimate, 1, 0, 0, nullptr, 0));
break;
}
}
if (i == _heroReaction->messages.size())
processMessage(msg);
} else {
processMessage(msg);
}
_holdMessages = false;
g_vm->videoSystem()->makeAllDirty();
break;
}
case 1:
sub_408940();
break;
case 2: {
_resourceId = _walk->getSpriteId() + _imageId + 10;
FlicDecoder *flc = g_vm->resMgr()->getFlic(_resourceId);
flc->setFrame(1);
_time = flc->getDelay();
initSurface();
g_vm->videoSystem()->makeAllDirty();
break;
}
default:
break;
}
}
void QObjectPetka::setReactionAfterWalk(uint index, QReaction *reaction, QMessageObject *sender, bool deleteReaction) {
_heroReaction = nullptr;
stopWalk();
QMessage msg(_id, kWalked, 0, 0, 0, sender, 0);
g_vm->getQSystem()->addMessage(msg);
_heroReaction = new QReaction();
_sender = sender;
for (uint i = index + 1; i < reaction->messages.size(); ++i) {
_heroReaction->messages.push_back(reaction->messages[i]);
}
if (deleteReaction) {
delete reaction;
}
}
void QObjectPetka::stopWalk() {
_isWalking = false;
_holdMessages = false;
Common::List<QMessage> &list = g_vm->getQSystem()->_messages;
for (Common::List<QMessage>::iterator it = list.begin(); it != list.end(); ++it) {
if (it->opcode == kWalked && it->objId == _id) {
it->objId = (uint16)-1;
}
}
delete _heroReaction;
_heroReaction = nullptr;
if (!_field7C) {
Common::Point p = _walk->sub_4234B0();
_x = p.x;
_y = p.y;
QMessage msg(_id, kSet, (uint16)_imageId, 1, 0, nullptr, 0);
processMessage(msg);
}
}
void QObjectPetka::update(int time) {
if (!_animate || !_isShown)
return;
if (_isWalking)
_time += time * (g_vm->getQSystem()->_panelInterface->getHeroSpeed() + 50) / 50;
else
_time += time;
FlicDecoder *flc = g_vm->resMgr()->getFlic(_resourceId);
if (flc && flc->getFrameCount() != 1) {
if (_sound) {
Common::Rect bounds = flc->getBounds();
_sound->setBalance(bounds.left + bounds.width() / 2 - g_vm->getQSystem()->_xOffset, 640);
}
while (_time >= (int)flc->getDelay()) {
if (_sound && flc->getCurFrame() == 0) {
_startSound = true;
}
flc->setFrame(-1);
if (flc->getCurFrame() == (int32)flc->getFrameCount() - 1) {
g_vm->getQSystem()->addMessage(_id, kEnd, _resourceId, 0, 0, 0, nullptr);
}
if (flc->getCurFrame() + 1 == (int32)flc->getFrameCount() / 2) {
g_vm->getQSystem()->addMessage(_id, kHalf, _resourceId, 0, 0, 0, nullptr);
}
if (_field7C && flc->getCurFrame() == 0)
_time = -10000;
updateWalk();
flc = g_vm->resMgr()->getFlic(_resourceId);
_surfH = flc->getHeight() * _k;
_surfW = flc->getWidth() * _k;
_time -= flc->getDelay();
g_vm->videoSystem()->addDirtyRect(Common::Rect(_x, _y, _surfW + _x, _surfH + _y));
}
}
}
bool QObjectPetka::isInPoint(Common::Point p) {
if (!_isActive)
return false;
FlicDecoder *flc = g_vm->resMgr()->getFlic(_resourceId);
const Graphics::Surface *flcSurface = flc->getCurrentFrame();
Common::Rect bounds(_surfW, _surfH);
Graphics::ManagedSurface s(_surfW, _surfH, flcSurface->format);
s.transBlitFrom(*flcSurface, Common::Rect(0, 0, flcSurface->w, flcSurface->h), bounds);
p.x -= _x;
p.y -= _y;
if (!bounds.contains(p.x, p.y))
return false;
return *(uint16 *)s.getBasePtr(p.x, p.y) != 0;
}
void QObjectPetka::updateZ() {
if (_animate && _isShown && _updateZ) {
FlicDecoder *flc = g_vm->resMgr()->getFlic(_resourceId);
if (_isWalking) {
_z = _walk->currPos().y;
} else {
_z = _y + flc->getHeight() * _k;
}
}
}
void QObjectPetka::sub_408940() {
FlicDecoder *flc = g_vm->resMgr()->getFlic(_resourceId);
QSystem *sys = g_vm->getQSystem();
int xOff = sys->_xOffset;
Common::Rect dirty(_x - xOff, _y, _surfW + _x - xOff, _surfH + _y);
g_vm->videoSystem()->addDirtyRect(dirty);
Common::Point currPos = _walk->currPos();
_k = calcPerspective(currPos.y);
_surfW = flc->getWidth() * _k;
_surfH = flc->getHeight() * _k;
Common::Point p = _walk->sub_4234B0();
_x = p.x;
_y = p.y;
_x_ = currPos.x;
_y_ = currPos.y;
recalcOffset();
g_vm->videoSystem()->addDirtyRect(Common::Rect(_x - xOff, _y, _surfW + _x - xOff, _surfH + _y));
}
void QObjectPetka::recalcOffset() {
QSystem *sys = g_vm->getQSystem();
int xOff = sys->_xOffset;
if (_x_ < xOff + 160 || _x_ > xOff + 480) {
sys->_reqOffset = _x_ - 320;
}
sys->_reqOffset = CLIP<int>(sys->_reqOffset, 0, sys->_sceneWidth - 640);
}
QObjectChapayev::QObjectChapayev() {
_x = 477;
_y = 350;
// _surfId = -6;
}
}