mirror of
https://github.com/scummvm/scummvm.git
synced 2025-04-02 10:52:32 -04:00
428 lines
10 KiB
C++
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;
|
|
}
|
|
|
|
}
|