scummvm/engines/grim/costume.cpp
Einar Johan Trøan Sømåen 109dbdb5b7
GRIM: Initialize Costume::_components in Costume::load
Presumably the indata always defines the parents before
the children, but nevertheless doesn't hurt to fix this,
if nothing else to make tools like Coverity happy.

Fixes COVERITY: 1470594
2022-10-02 00:50:42 +02:00

575 lines
18 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/endian.h"
#include "engines/grim/debug.h"
#include "engines/grim/colormap.h"
#include "engines/grim/costume.h"
#include "engines/grim/textsplit.h"
#include "engines/grim/resource.h"
#include "engines/grim/model.h"
#include "engines/grim/savegame.h"
#include "engines/grim/emi/modelemi.h"
#include "engines/grim/costume/chore.h"
#include "engines/grim/costume/head.h"
#include "engines/grim/emi/costume/emianim_component.h"
#include "engines/grim/emi/costume/emimesh_component.h"
#include "engines/grim/emi/costume/emiskel_component.h"
#include "engines/grim/costume/main_model_component.h"
#include "engines/grim/costume/colormap_component.h"
#include "engines/grim/costume/keyframe_component.h"
#include "engines/grim/costume/mesh_component.h"
#include "engines/grim/costume/lua_var_component.h"
#include "engines/grim/costume/sound_component.h"
#include "engines/grim/costume/bitmap_component.h"
#include "engines/grim/costume/material_component.h"
#include "engines/grim/costume/sprite_component.h"
#include "engines/grim/costume/anim_component.h"
namespace Grim {
// A costume in the Residual/GrimE engine consists of a set of
// components, and a set of chores. Each component represents an
// on-screen object, or a keyframe animation, or a sound effect; each
// chore gives a set of instructions for how to move and/or activate
// or deactivate each component at certain times.
//
// Each actor contains a stack of costumes, on which a new costume can
// be pushed or from which an old costume can be popped at any time.
// For the most part, these costumes are independent. The exception
// is the main model component ('MMDL'), for which multiple costumes
// share the same base 3D object (if they refer to the same file).
//
// This is complicated by the fact that multiple keyframe animations
// can have an effect on the positions of the 3D objects. Each
// keyframe animation has certain nodes internally "tagged", and the
// keyframe components specify precedences for the tagged nodes and
// for the non-tagged nodes. If the highest precedence for a given
// node is given by multiple keyframe animations, their contributions
// are averaged.
//
// Each component can implement several virtual methods which are
// called by the costume:
//
// init() -- allows the component to initialize itself. This is
// separate from the constructor since there are cases where
// information from child components may be needed before
// the object can be fully constructed. This is particularly
// the case with colormaps, which are needed before even
// starting to load a 3D model.
// setKey(val) -- notifies the component of a change in the "state"
// given by a playing chore
// update() -- gives the component a chance to update its internal
// state once every frame
// draw() -- actually draws the component onto the screen
// reset() -- notifies the component that a chore controlling it
// has stopped
//
// For the 3D objects, a model's component first initializes internal
// state for the model's nodes in its update() method. Then the
// keyframes' update() methods work with this data to implement the
// precedence and add up all contributions for the highest precedence.
// Then the model's draw() method does the averaging and draws the
// polygons accordingly. (Actually, this is a lie -- the top-level
// 3D objects draw themselves and all their children. This makes it
// easier to move objects Manny is holding when his hands move, for
// example.)
//
// For bitmaps, the actual drawing is handled by the Set class. The
// bitmaps to be drawn are associated to the needed camera setups
// using NewObjectState; bitmaps marked OBJSTATE_UNDERLAY and
// OBJSTATE_STATE are drawn first, then the 3D objects, then bitmaps
// marked OBJSTATE_OVERLAY. So the BitmapComponent just needs to pass
// along setKey requests to the actual bitmap object.
Costume::Costume(const Common::String &fname, Actor *owner, Costume *prevCost) :
Object(), _head(nullptr), _chores(nullptr), _components(nullptr),
_numComponents(0), _numChores(0), _fname(fname), _owner(owner) {
_lookAtRate = 200;
_prevCostume = prevCost;
}
void Costume::load(Common::SeekableReadStream *data) {
TextSplitter ts(_fname, data);
ts.expectString("costume v0.1");
ts.expectString("section tags");
int numTags;
ts.scanString(" numtags %d", 1, &numTags);
tag32 *tags = new tag32[numTags];
for (int i = 0; i < numTags; i++) {
unsigned char t[4];
int which;
// Obtain a tag ID from the file
ts.scanString(" %d '%c%c%c%c'", 5, &which, &t[0], &t[1], &t[2], &t[3]);
// Force characters to upper case
for (int j = 0; j < 4; j++)
t[j] = toupper(t[j]);
memcpy(&tags[which], t, sizeof(tag32));
tags[which] = FROM_BE_32(tags[which]);
}
ts.expectString("section components");
ts.scanString(" numcomponents %d", 1, &_numComponents);
_components = new Component *[_numComponents]{};
for (int i = 0; i < _numComponents; i++) {
int id, tagID, hash, parentID, namePos;
const char *line = ts.getCurrentLine();
Component *prevComponent = nullptr;
if (sscanf(line, " %d %d %d %d %n", &id, &tagID, &hash, &parentID, &namePos) < 4)
error("Bad component specification line: `%s'", line);
ts.nextLine();
// A Parent ID of "-1" indicates that the component should
// use the properties of the previous costume as a base
if (parentID == -1) {
if (_prevCostume) {
// However, only the first item can actually share the
// node hierarchy with the previous costume, so flag
// that component so it knows what to do
if (i == 0)
parentID = -2;
prevComponent = _prevCostume->_components[0];
// Make sure that the component is valid
if (!prevComponent->isComponentType('M','M','D','L'))
prevComponent = nullptr;
} else if (id > 0) {
// Use the MainModelComponent of this costume as prevComponent,
// so that the component can use its colormap.
prevComponent = _components[0];
}
}
// Actually load the appropriate component
_components[id] = loadComponent(tags[tagID], parentID < 0 ? nullptr : _components[parentID], parentID, line + namePos, prevComponent);
_components[id]->setCostume(this);
}
delete[] tags;
for (int i = 0; i < _numComponents; i++) {
if (_components[i]) {
_components[i]->init();
}
}
ts.expectString("section chores");
ts.scanString(" numchores %d", 1, &_numChores);
_chores = new Chore *[_numChores];
for (int i = 0; i < _numChores; i++) {
int id, length, tracks;
char name[32];
ts.scanString(" %d %d %d %32s", 4, &id, &length, &tracks, name);
_chores[id] = new Chore(name, i, this, length, tracks);
Debug::debug(Debug::Chores, "Loaded chore: %s\n", name);
}
ts.expectString("section keys");
for (int i = 0; i < _numChores; i++) {
int which;
ts.scanString("chore %d", 1, &which);
_chores[which]->load(ts);
}
_head = new Head();
}
Costume::~Costume() {
stopChores();
for (int i = _numComponents - 1; i >= 0; i--) {
delete _components[i];
}
delete[] _components;
for (int i = 0; i < _numChores; ++i) {
delete _chores[i];
}
delete[] _chores;
delete _head;
}
Component *Costume::loadComponent (tag32 tag, Component *parent, int parentID, const char *name, Component *prevComponent) {
if (tag == MKTAG('M','M','D','L'))
return new MainModelComponent(parent, parentID, name, prevComponent, tag);
else if (tag == MKTAG('M','O','D','L'))
return new ModelComponent(parent, parentID, name, prevComponent, tag);
else if (tag == MKTAG('C','M','A','P'))
return new ColormapComponent(parent, parentID, name, tag);
else if (tag == MKTAG('K','E','Y','F'))
return new KeyframeComponent(parent, parentID, name, tag);
else if (tag == MKTAG('M','E','S','H'))
return new MeshComponent(parent, parentID, name, tag);
else if (tag == MKTAG('L','U','A','V'))
return new LuaVarComponent(parent, parentID, name, tag);
else if (tag == MKTAG('I','M','L','S'))
return new SoundComponent(parent, parentID, name, tag);
else if (tag == MKTAG('B','K','N','D'))
return new BitmapComponent(parent, parentID, name, tag);
else if (tag == MKTAG('M','A','T',' '))
return new MaterialComponent(parent, parentID, name, tag);
else if (tag == MKTAG('S','P','R','T'))
return new SpriteComponent(parent, parentID, name, tag);
else if (tag == MKTAG('A','N','I','M')) //Used in the demo
return new AnimComponent(parent, parentID, name, tag);
char t[4];
memcpy(t, &tag, sizeof(tag32));
warning("loadComponent: Unknown tag '%c%c%c%c', name '%s'", t[0], t[1], t[2], t[3], name);
return nullptr;
}
ModelComponent *Costume::getMainModelComponent() const {
for (int i = 0; i < _numComponents; i++) {
// Needs to handle Main Models (pigeons) and normal Models
// (when Manny climbs the rope)
if (_components[i] && _components[i]->isComponentType('M','M','D','L'))
return static_cast<ModelComponent *>(_components[i]);
}
return nullptr;
}
ModelNode *Costume::getModelNodes() {
ModelComponent *comp = getMainModelComponent();
if (comp) {
return comp->getHierarchy();
}
return nullptr;
}
Model *Costume::getModel() {
ModelComponent *comp = getMainModelComponent();
if (comp) {
return comp->getModel();
}
return nullptr;
}
void Costume::setChoreLastFrame(int num) {
if (num < 0 || num >= _numChores) {
Debug::warning(Debug::Chores, "Requested chore number %d is outside the range of chores (0-%d)", num, _numChores);
return;
}
_chores[num]->setLastFrame();
}
void Costume::setChoreLooping(int num, bool val) {
if (num < 0 || num >= _numChores) {
Debug::warning(Debug::Chores, "Requested chore number %d is outside the range of chores (0-%d)", num, _numChores);
return;
}
_chores[num]->setLooping(val);
}
void Costume::playChoreLooping(const char *name, uint msecs) {
for (int i = 0; i < _numChores; ++i) {
if (strcmp(_chores[i]->getName(), name) == 0) {
playChoreLooping(i, msecs);
return;
}
}
warning("Costume::playChoreLooping: Could not find chore: %s", name);
return;
}
void Costume::playChoreLooping(int num, uint msecs) {
if (num < 0 || num >= _numChores) {
Debug::warning(Debug::Chores, "Requested chore number %d is outside the range of chores (0-%d)", num, _numChores);
return;
}
_chores[num]->playLooping(msecs);
if (Common::find(_playingChores.begin(), _playingChores.end(), _chores[num]) == _playingChores.end())
_playingChores.push_back(_chores[num]);
}
Chore *Costume::getChore(const char *name) {
for (int i = 0; i < _numChores; ++i) {
if (strcmp(_chores[i]->getName(), name) == 0) {
return _chores[i];
}
}
return nullptr;
}
int Costume::getChoreId(const char *name) {
if (name == nullptr) {
return -1;
}
for (int i = 0; i < _numChores; ++i) {
if (strcmp(_chores[i]->getName(), name) == 0) {
return i;
}
}
return -1;
}
void Costume::playChore(const char *name, uint msecs) {
for (int i = 0; i < _numChores; ++i) {
if (strcmp(_chores[i]->getName(), name) == 0) {
playChore(i, msecs);
return;
}
}
warning("Costume::playChore: Could not find chore: %s", name);
return;
}
void Costume::playChore(int num, uint msecs) {
if (num < 0 || num >= _numChores) {
Debug::warning(Debug::Chores, "Requested chore number %d is outside the range of chores (0-%d)", num, _numChores);
return;
}
_chores[num]->play(msecs);
if (Common::find(_playingChores.begin(), _playingChores.end(), _chores[num]) == _playingChores.end())
_playingChores.push_back(_chores[num]);
}
void Costume::stopChore(int num, uint msecs) {
if (num < 0 || num >= _numChores) {
Debug::warning(Debug::Chores, "Requested chore number %d is outside the range of chores (0-%d)", num, _numChores);
return;
}
_chores[num]->stop(msecs);
}
void Costume::setColormap(const Common::String &map) {
// Sometimes setColormap is called on a null costume,
// see where raoul is gone in hh.set
if (!map.size())
return;
_cmap = g_resourceloader->getColormap(map);
for (int i = 0; i < _numComponents; i++)
if (_components[i])
_components[i]->setColormap(nullptr);
}
void Costume::stopChores(bool ignoreLoopingChores, int msecs) {
for (int i = 0; i < _numChores; i++) {
if (ignoreLoopingChores && _chores[i]->isLooping()) {
continue;
}
_chores[i]->stop(msecs);
}
}
void Costume::fadeChoreIn(int chore, uint msecs) {
if (chore < 0 || chore >= _numChores) {
Debug::warning(Debug::Chores, "Requested chore number %d is outside the range of chores (0-%d)", chore, _numChores);
return;
}
_chores[chore]->fadeIn(msecs);
if (Common::find(_playingChores.begin(), _playingChores.end(), _chores[chore]) == _playingChores.end())
_playingChores.push_back(_chores[chore]);
}
void Costume::fadeChoreOut(int chore, uint msecs) {
if (chore < 0 || chore >= _numChores) {
Debug::warning(Debug::Chores, "Requested chore number %d is outside the range of chores (0-%d)", chore, _numChores);
return;
}
_chores[chore]->fadeOut(msecs);
}
int Costume::isChoring(const char *name, bool excludeLooping) {
for (int i = 0; i < _numChores; i++) {
if (!strcmp(_chores[i]->getName(), name) && _chores[i]->isPlaying() && !(excludeLooping && _chores[i]->isLooping()))
return i;
}
return -1;
}
int Costume::isChoring(int num, bool excludeLooping) {
if (num < 0 || num >= _numChores) {
Debug::warning(Debug::Chores, "Requested chore number %d is outside the range of chores (0-%d)", num, _numChores);
return -1;
}
if (_chores[num]->isPlaying() && !(excludeLooping && _chores[num]->isLooping()))
return num;
else
return -1;
}
int Costume::isChoring(bool excludeLooping) {
for (int i = 0; i < _numChores; i++) {
if (_chores[i]->isPlaying() && !(excludeLooping && _chores[i]->isLooping()))
return i;
}
return -1;
}
void Costume::draw() {
for (int i = 0; i < _numComponents; i++)
if (_components[i])
_components[i]->draw();
}
void Costume::getBoundingBox(int *x1, int *y1, int *x2, int *y2) {
for (int i = 0; i < _numComponents; i++) {
if (_components[i] &&(_components[i]->isComponentType('M','M','D','L') ||
_components[i]->isComponentType('M','O','D','L'))) {
ModelComponent *c = static_cast<ModelComponent *>(_components[i]);
c->getBoundingBox(x1, y1, x2, y2);
}
if (_components[i] &&(_components[i]->isComponentType('m','e','s','h'))) {
EMIMeshComponent *c = static_cast<EMIMeshComponent *>(_components[i]);
c->getBoundingBox(x1, y1, x2, y2);
}
}
}
int Costume::update(uint time) {
for (Common::List<Chore*>::iterator i = _playingChores.begin(); i != _playingChores.end(); ++i) {
(*i)->update(time);
if (!(*i)->isPlaying()) {
i = _playingChores.erase(i);
--i;
}
}
int marker = 0;
for (int i = 0; i < _numComponents; i++) {
if (_components[i]) {
_components[i]->setMatrix(_matrix);
int m = _components[i]->update(time);
if (m > 0) {
marker = m;
}
}
}
return marker;
}
void Costume::animate() {
for (int i = 0; i < _numComponents; i++) {
if (_components[i]) {
_components[i]->animate();
}
}
}
void Costume::moveHead(bool entering, const Math::Vector3d &lookAt) {
_head->lookAt(entering, lookAt, _lookAtRate, _matrix);
}
int Costume::getHeadJoint() const {
return static_cast<Head *>(_head)->getJoint3();
}
void Costume::setHead(int joint1, int joint2, int joint3, float maxRoll, float maxPitch, float maxYaw) {
Head *head = static_cast<Head *>(_head);
head->setJoints(joint1, joint2, joint3);
head->loadJoints(getModelNodes());
head->setMaxAngles(maxPitch, maxYaw, maxRoll);
}
void Costume::setLookAtRate(float rate) {
_lookAtRate = rate;
}
float Costume::getLookAtRate() const {
return _lookAtRate;
}
void Costume::setPosRotate(const Math::Vector3d &pos, const Math::Angle &pitch,
const Math::Angle &yaw, const Math::Angle &roll) {
_matrix.setPosition(pos);
_matrix.buildFromEuler(yaw, pitch, roll, Math::EO_ZXY);
}
Math::Matrix4 Costume::getMatrix() const {
return _matrix;
}
Costume *Costume::getPreviousCostume() const {
return _prevCostume;
}
void Costume::saveState(SaveGame *state) const {
if (_cmap) {
state->writeBool(true);
state->writeString(_cmap->getFilename());
} else {
state->writeBool(false);
}
for (int i = 0; i < _numChores; ++i) {
_chores[i]->saveState(state);
}
for (int i = 0; i < _numComponents; ++i) {
Component *c = _components[i];
if (c) {
state->writeBool(c->_visible);
c->saveState(state);
}
}
state->writeLEUint32(_playingChores.size());
for (Common::List<Chore*>::const_iterator i = _playingChores.begin(); i != _playingChores.end(); ++i) {
state->writeLESint32((*i)->getChoreId());
}
state->writeFloat(_lookAtRate);
_head->saveState(state);
}
bool Costume::restoreState(SaveGame *state) {
if (state->readBool()) {
Common::String str = state->readString();
setColormap(str);
}
for (int i = 0; i < _numChores; ++i) {
_chores[i]->restoreState(state);
}
for (int i = 0; i < _numComponents; ++i) {
Component *c = _components[i];
if (c) {
c->_visible = state->readBool();
if (state->saveMinorVersion() < 14) {
// skip the old _matrix vector
state->readVector3d();
}
c->restoreState(state);
}
}
int numPlayingChores = state->readLEUint32();
for (int i = 0; i < numPlayingChores; ++i) {
int id = state->readLESint32();
_playingChores.push_back(_chores[id]);
}
_lookAtRate = state->readFloat();
_head->restoreState(state);
_head->loadJoints(getModelNodes());
return true;
}
} // end of namespace Grim