mirror of
https://github.com/libretro/scummvm.git
synced 2025-04-03 07:11:49 +00:00
EMI: Initial work on getting costumes to draw animations.
Also, an initial split-off for EMI from the costume-class, now uses it as a base class. (somaen)
This commit is contained in:
parent
b79aa00944
commit
1226e45ac3
@ -29,7 +29,7 @@
|
||||
#include "engines/grim/grim.h"
|
||||
#include "engines/grim/resource.h"
|
||||
#include "engines/grim/model.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"
|
||||
@ -105,21 +105,16 @@ namespace Grim {
|
||||
// marked OBJSTATE_OVERLAY. So the BitmapComponent just needs to pass
|
||||
// along setKey requests to the actual bitmap object.
|
||||
|
||||
Costume::Costume(const Common::String &fname, Common::SeekableReadStream *data, Costume *prevCost) :
|
||||
Costume::Costume(const Common::String &fname, Costume *prevCost) :
|
||||
Object(), _head(new Head()), _chores(NULL) {
|
||||
|
||||
_fname = fname;
|
||||
_lookAtRate = 200;
|
||||
_prevCostume = prevCost;
|
||||
if (g_grim->getGameType() == GType_MONKEY4) {
|
||||
loadEMI(data, prevCost);
|
||||
} else {
|
||||
TextSplitter ts(data);
|
||||
loadGRIM(ts, prevCost);
|
||||
}
|
||||
}
|
||||
|
||||
void Costume::loadGRIM(TextSplitter &ts, Costume *prevCost) {
|
||||
void Costume::load(Common::SeekableReadStream *data) {
|
||||
TextSplitter ts(data);
|
||||
ts.expectString("costume v0.1");
|
||||
ts.expectString("section tags");
|
||||
int numTags;
|
||||
@ -152,7 +147,7 @@ void Costume::loadGRIM(TextSplitter &ts, Costume *prevCost) {
|
||||
// A Parent ID of "-1" indicates that the component should
|
||||
// use the properties of the previous costume as a base
|
||||
if (parentID == -1) {
|
||||
if (prevCost) {
|
||||
if (_prevCostume) {
|
||||
MainModelComponent *mmc;
|
||||
|
||||
// However, only the first item can actually share the
|
||||
@ -160,7 +155,7 @@ void Costume::loadGRIM(TextSplitter &ts, Costume *prevCost) {
|
||||
// that component so it knows what to do
|
||||
if (i == 0)
|
||||
parentID = -2;
|
||||
prevComponent = prevCost->_components[0];
|
||||
prevComponent = _prevCostume->_components[0];
|
||||
mmc = dynamic_cast<MainModelComponent *>(prevComponent);
|
||||
// Make sure that the component is valid
|
||||
if (!mmc)
|
||||
@ -205,85 +200,6 @@ void Costume::loadGRIM(TextSplitter &ts, Costume *prevCost) {
|
||||
}
|
||||
}
|
||||
|
||||
void Costume::loadEMI(Common::SeekableReadStream *data, Costume *prevCost) {
|
||||
Common::List<Component *>components;
|
||||
|
||||
_numChores = data->readUint32LE();
|
||||
_chores = new Chore *[_numChores];
|
||||
for (int i = 0; i < _numChores; i++) {
|
||||
_chores[i] = new PoolChore();
|
||||
uint32 nameLength;
|
||||
Component *prevComponent = NULL;
|
||||
nameLength = data->readUint32LE();
|
||||
data->read(_chores[i]->_name, nameLength);
|
||||
float length;
|
||||
data->read(&length, 4);
|
||||
_chores[i]->_length = (int)length;
|
||||
|
||||
_chores[i]->_owner = this;
|
||||
_chores[i]->_numTracks = data->readUint32LE();
|
||||
_chores[i]->_tracks = new ChoreTrack[_chores[i]->_numTracks];
|
||||
|
||||
for (int k = 0; k < _chores[i]->_numTracks; k++) {
|
||||
int componentNameLength = data->readUint32LE();
|
||||
|
||||
char *name = new char[componentNameLength];
|
||||
data->read(name, componentNameLength);
|
||||
|
||||
data->readUint32LE();
|
||||
int parentID = data->readUint32LE();
|
||||
if (parentID == -1 && prevCost) {
|
||||
MainModelComponent *mmc;
|
||||
|
||||
// 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 = prevCost->_components[0];
|
||||
mmc = dynamic_cast<MainModelComponent *>(prevComponent);
|
||||
// Make sure that the component is valid
|
||||
if (!mmc)
|
||||
prevComponent = NULL;
|
||||
}
|
||||
// Actually load the appropriate component
|
||||
Component *component = loadComponentEMI(parentID < 0 ? NULL : _components[parentID], parentID, name, prevComponent);
|
||||
|
||||
|
||||
//Component *component = loadComponentEMI(name, parent);
|
||||
|
||||
components.push_back(component);
|
||||
|
||||
ChoreTrack &track = _chores[i]->_tracks[k];
|
||||
track.numKeys = data->readUint32LE();
|
||||
track.keys = new TrackKey[track.numKeys];
|
||||
|
||||
// this is probably wrong
|
||||
track.compID = 0;
|
||||
for (int j = 0; j < track.numKeys; j++) {
|
||||
float time, value;
|
||||
data->read(&time, 4);
|
||||
data->read(&value, 4);
|
||||
track.keys[j].time = (int)time;
|
||||
track.keys[j].value = (int)value;
|
||||
}
|
||||
delete[] name;
|
||||
}
|
||||
//_chores[i]._tracks->compID;
|
||||
}
|
||||
|
||||
_numComponents = components.size();
|
||||
_components = new Component *[_numComponents];
|
||||
int i = 0;
|
||||
for (Common::List<Component *>::iterator it = components.begin(); it != components.end(); ++it, ++i) {
|
||||
_components[i] = *it;
|
||||
if (!_components[i])
|
||||
continue;
|
||||
_components[i]->setCostume(this);
|
||||
_components[i]->init();
|
||||
}
|
||||
}
|
||||
|
||||
Costume::~Costume() {
|
||||
if (_chores) {
|
||||
stopChores();
|
||||
@ -332,54 +248,6 @@ Component *Costume::loadComponent (tag32 tag, Component *parent, int parentID, c
|
||||
return NULL;
|
||||
}
|
||||
|
||||
Component *Costume::loadComponentEMI(Component *parent, int parentID, const char *name, Component *prevComponent) {
|
||||
// some have an exclimation mark, this could mean something.
|
||||
// for now, return 0 otherwise it will just crash in some other part.
|
||||
//return 0;
|
||||
|
||||
assert(name[0] == '!');
|
||||
++name;
|
||||
|
||||
char type[5];
|
||||
tag32 tag = 0;
|
||||
memcpy(&tag, name, 4);
|
||||
memcpy(&type, name, 4);
|
||||
type[4] = 0;
|
||||
|
||||
name += 4;
|
||||
|
||||
if (FROM_BE_32(tag) == MKTAG('m','e','s','h')) {
|
||||
//Debug::warning(Debug::Costumes, "Actor::loadComponentEMI Implement MESH-handling: %s" , name);
|
||||
return new EMIMeshComponent(parent, parentID, name, prevComponent, tag);
|
||||
} else if (FROM_BE_32(tag) == MKTAG('s','k','e','l')) {
|
||||
//Debug::warning(Debug::Costumes, "Actor::loadComponentEMI Implement SKEL-handling: %s" , name);
|
||||
return new EMISkelComponent(parent, parentID, name, prevComponent, tag);
|
||||
} else if (FROM_BE_32(tag) == MKTAG('t','e','x','i')) {
|
||||
Debug::warning(Debug::Costumes, "Actor::loadComponentEMI Implement TEXI-handling: %s" , name);
|
||||
//return new MaterialComponent(parent, parentID, name, tag);
|
||||
} else if (FROM_BE_32(tag) == MKTAG('a','n','i','m')) {
|
||||
//Debug::warning(Debug::Costumes, "Actor::loadComponentEMI Implement ANIM-handling: %s" , name);
|
||||
return new EMIAnimComponent(parent, parentID, name, prevComponent, tag);
|
||||
} else if (FROM_BE_32(tag) == MKTAG('l','u','a','c')) {
|
||||
Debug::warning(Debug::Costumes, "Actor::loadComponentEMI Implement LUAC-handling: %s" , name);
|
||||
} else if (FROM_BE_32(tag) == MKTAG('l','u','a','v')) {
|
||||
Debug::warning(Debug::Costumes, "Actor::loadComponentEMI Implement LUAV-handling: %s" , name);
|
||||
//return new LuaVarComponent(parent, parentID, name, tag);
|
||||
} else if (FROM_BE_32(tag) == MKTAG('s','p','r','t')) {
|
||||
Debug::warning(Debug::Costumes, "Actor::loadComponentEMI Implement SPRT-handling: %s" , name);
|
||||
//return new SpriteComponent(parent, parentID, name, tag);
|
||||
} else if (FROM_BE_32(tag) == MKTAG('s','h','a','d')) {
|
||||
Debug::warning(Debug::Costumes, "Actor::loadComponentEMI Implement SHAD-handling: %s" , name);
|
||||
} else {
|
||||
error("Actor::loadComponentEMI missing tag: %s for %s", name, type);
|
||||
}
|
||||
/*
|
||||
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 NULL;
|
||||
}
|
||||
|
||||
ModelComponent *Costume::getMainModelComponent() const {
|
||||
for (int i = 0; i < _numComponents; i++) {
|
||||
// Needs to handle Main Models (pigeons) and normal Models
|
||||
|
@ -44,12 +44,10 @@ class Head;
|
||||
|
||||
class Costume : public Object {
|
||||
public:
|
||||
Costume(const Common::String &filename, Common::SeekableReadStream *data, Costume *prevCost);
|
||||
|
||||
void loadGRIM(TextSplitter &ts, Costume *prevCost);
|
||||
void loadEMI(Common::SeekableReadStream *data, Costume *prevCost);
|
||||
Costume(const Common::String &filename, Costume *prevCost);
|
||||
|
||||
virtual ~Costume();
|
||||
virtual void load(Common::SeekableReadStream *data);
|
||||
|
||||
const Common::String &getFilename() const { return _fname; }
|
||||
void playChore(const char *name);
|
||||
@ -77,10 +75,10 @@ public:
|
||||
|
||||
CMap *getCMap() { return _cmap; }
|
||||
|
||||
int update(uint frameTime);
|
||||
virtual int update(uint frameTime);
|
||||
void animate();
|
||||
void setupTextures();
|
||||
void draw();
|
||||
virtual void draw();
|
||||
void getBoundingBox(int *x1, int *y1, int *x2, int *y2);
|
||||
void setPosRotate(Math::Vector3d pos, const Math::Angle &pitch,
|
||||
const Math::Angle &yaw, const Math::Angle &roll);
|
||||
@ -88,12 +86,15 @@ public:
|
||||
|
||||
Costume *getPreviousCostume() const;
|
||||
|
||||
void saveState(SaveGame *state) const;
|
||||
bool restoreState(SaveGame *state);
|
||||
virtual void saveState(SaveGame *state) const;
|
||||
virtual bool restoreState(SaveGame *state);
|
||||
|
||||
Component *getComponent(int num) { return _components[num]; }
|
||||
protected:
|
||||
virtual Component *loadComponent(tag32 tag, Component *parent, int parentID, const char *name, Component *prevComponent);
|
||||
|
||||
void load(TextSplitter &ts, Costume *prevCost);
|
||||
|
||||
private:
|
||||
Component *loadComponent(tag32 tag, Component *parent, int parentID, const char *name, Component *prevComponent);
|
||||
Component *loadComponentEMI(Component *parent, int parentID, const char *name, Component *prevComponent);
|
||||
ModelComponent *getMainModelComponent() const;
|
||||
|
||||
Common::String _fname;
|
||||
|
@ -174,4 +174,9 @@ void Chore::fadeOut(uint msecs) {
|
||||
fade(Animation::FadeOut, msecs);
|
||||
}
|
||||
|
||||
void Chore::createTracks(int num) {
|
||||
_numTracks = num;
|
||||
_tracks = new ChoreTrack[_numTracks];
|
||||
}
|
||||
|
||||
} // end of namespace Grim
|
||||
|
@ -32,6 +32,7 @@ namespace Grim {
|
||||
|
||||
class Costume;
|
||||
class Animation;
|
||||
class Component;
|
||||
|
||||
struct TrackKey {
|
||||
int time, value;
|
||||
@ -41,6 +42,7 @@ struct ChoreTrack {
|
||||
int compID;
|
||||
int numKeys;
|
||||
TrackKey *keys;
|
||||
Component *component;
|
||||
};
|
||||
|
||||
class Chore {
|
||||
@ -61,6 +63,9 @@ public:
|
||||
bool isPlaying() { return _playing; }
|
||||
|
||||
virtual int getId() { return _choreId; }
|
||||
|
||||
void setOwner(Costume *owner) { _owner = owner; }
|
||||
void createTracks(int num);
|
||||
|
||||
private:
|
||||
void setKeys(int startTime, int stopTime);
|
||||
@ -78,6 +83,7 @@ private:
|
||||
int _currTime;
|
||||
|
||||
friend class Costume;
|
||||
friend class EMICostume;
|
||||
};
|
||||
|
||||
class PoolChore : public PoolObject<PoolChore, MKTAG('C', 'H', 'O', 'R')>, public Chore {
|
||||
|
@ -71,6 +71,7 @@ protected:
|
||||
void resetHierCMap();
|
||||
|
||||
friend class Costume;
|
||||
friend class EMICostume;
|
||||
};
|
||||
|
||||
} // end of namespace Grim
|
||||
|
@ -8,12 +8,12 @@
|
||||
* modify it under the terms of the GNU Lesser General Public
|
||||
* License as published by the Free Software Foundation; either
|
||||
* version 2.1 of the License, or (at your option) any later version.
|
||||
|
||||
|
||||
* This library 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
|
||||
* Lesser General Public License for more details.
|
||||
|
||||
|
||||
* You should have received a copy of the GNU Lesser General Public
|
||||
* License along with this library; if not, write to the Free Software
|
||||
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
|
||||
|
@ -23,8 +23,12 @@
|
||||
#include "common/foreach.h"
|
||||
|
||||
#include "engines/grim/emi/costume/emianim_component.h"
|
||||
#include "engines/grim/emi/costume/emiskel_component.h"
|
||||
#include "engines/grim/resource.h"
|
||||
#include "engines/grim/costume.h"
|
||||
#include "engines/grim/emi/costumeemi.h"
|
||||
#include "engines/grim/emi/modelemi.h"
|
||||
#include "engines/grim/emi/skeleton.h"
|
||||
#include "engines/grim/emi/animationemi.h"
|
||||
|
||||
namespace Grim {
|
||||
@ -50,7 +54,10 @@ void EMIAnimComponent::reset() {
|
||||
}
|
||||
|
||||
void EMIAnimComponent::draw() {
|
||||
|
||||
EMISkelComponent *skel = ((EMICostume*) _cost)->_emiSkel;
|
||||
if (skel && skel->_obj) {
|
||||
skel->_obj->setAnim(_obj);
|
||||
}
|
||||
}
|
||||
|
||||
} // end of namespace Grim
|
||||
|
@ -25,6 +25,7 @@
|
||||
#include "engines/grim/emi/costume/emimesh_component.h"
|
||||
#include "engines/grim/emi/modelemi.h"
|
||||
#include "engines/grim/resource.h"
|
||||
#include "engines/grim/costume.h"
|
||||
|
||||
namespace Grim {
|
||||
|
||||
|
@ -38,7 +38,7 @@ public:
|
||||
void reset();
|
||||
void draw();
|
||||
|
||||
private:
|
||||
public:
|
||||
bool _hierShared;
|
||||
Common::List<EMIMeshComponent*> _children;
|
||||
EMIMeshComponent *_parentModel;
|
||||
|
@ -26,6 +26,7 @@
|
||||
#include "engines/grim/resource.h"
|
||||
#include "engines/grim/emi/modelemi.h"
|
||||
#include "engines/grim/emi/skeleton.h"
|
||||
#include "engines/grim/costume.h"
|
||||
|
||||
namespace Grim {
|
||||
|
||||
@ -42,6 +43,9 @@ void EMISkelComponent::init() {
|
||||
}
|
||||
|
||||
int EMISkelComponent::update(uint time) {
|
||||
if (_obj) {
|
||||
_obj->animate(time/1000.f);
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
|
@ -40,7 +40,7 @@ public:
|
||||
void reset();
|
||||
void draw();
|
||||
|
||||
private:
|
||||
public:
|
||||
bool _hierShared;
|
||||
Component *_parentModel;
|
||||
Common::String _filename;
|
||||
|
238
engines/grim/emi/costumeemi.cpp
Normal file
238
engines/grim/emi/costumeemi.cpp
Normal file
@ -0,0 +1,238 @@
|
||||
/* ResidualVM - A 3D game interpreter
|
||||
*
|
||||
* ResidualVM 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 library is free software; you can redistribute it and/or
|
||||
* modify it under the terms of the GNU Lesser General Public
|
||||
* License as published by the Free Software Foundation; either
|
||||
* version 2.1 of the License, or (at your option) any later version.
|
||||
|
||||
* This library 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
|
||||
* Lesser General Public License for more details.
|
||||
|
||||
* You should have received a copy of the GNU Lesser General Public
|
||||
* License along with this library; if not, write to the Free Software
|
||||
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
|
||||
*
|
||||
*/
|
||||
|
||||
#include "common/endian.h"
|
||||
|
||||
#include "engines/grim/debug.h"
|
||||
#include "engines/grim/costume.h"
|
||||
#include "engines/grim/grim.h"
|
||||
#include "engines/grim/resource.h"
|
||||
#include "engines/grim/emi/costumeemi.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"
|
||||
|
||||
namespace Grim {
|
||||
|
||||
EMICostume::EMICostume(const Common::String &fname, Costume *prevCost) :
|
||||
Costume(fname, prevCost), _emiSkel(NULL), _emiMesh(NULL) {
|
||||
}
|
||||
|
||||
void EMICostume::load(Common::SeekableReadStream *data) {
|
||||
Common::List<Component *>components;
|
||||
|
||||
_numChores = data->readUint32LE();
|
||||
_chores = new Chore *[_numChores];
|
||||
for (int i = 0; i < _numChores; i++) {
|
||||
_chores[i] = new PoolChore();
|
||||
uint32 nameLength;
|
||||
Component *prevComponent = NULL;
|
||||
nameLength = data->readUint32LE();
|
||||
data->read(_chores[i]->_name, nameLength);
|
||||
float length;
|
||||
data->read(&length, 4);
|
||||
_chores[i]->_length = (int)length;
|
||||
|
||||
_chores[i]->setOwner(this);
|
||||
_chores[i]->createTracks(data->readUint32LE());
|
||||
|
||||
for (int k = 0; k < _chores[i]->_numTracks; k++) {
|
||||
int componentNameLength = data->readUint32LE();
|
||||
|
||||
char *name = new char[componentNameLength];
|
||||
data->read(name, componentNameLength);
|
||||
|
||||
data->readUint32LE();
|
||||
int parentID = data->readUint32LE();
|
||||
if (parentID == -1 && _prevCostume) {
|
||||
MainModelComponent *mmc;
|
||||
|
||||
// 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->getComponent(0);
|
||||
mmc = dynamic_cast<MainModelComponent *>(prevComponent);
|
||||
// Make sure that the component is valid
|
||||
if (!mmc)
|
||||
prevComponent = NULL;
|
||||
}
|
||||
// Actually load the appropriate component
|
||||
Component *component = loadComponent(parentID < 0 ? NULL : _components[parentID], parentID, name, prevComponent);
|
||||
if (component) {
|
||||
component->setCostume(this);
|
||||
component->init();
|
||||
|
||||
if (strcmp(_chores[i]->_name, "wear_default") == 0) {
|
||||
EMIMeshComponent *m = dynamic_cast<EMIMeshComponent *>(component);
|
||||
EMISkelComponent *s = dynamic_cast<EMISkelComponent *>(component);
|
||||
if (m) {
|
||||
_emiMesh = m;
|
||||
if (_emiSkel) {
|
||||
_emiMesh->_obj->setSkeleton(_emiSkel->_obj);
|
||||
}
|
||||
} else if (s) {
|
||||
_emiSkel = s;
|
||||
if (_emiMesh) {
|
||||
_emiMesh->_obj->setSkeleton(_emiSkel->_obj);
|
||||
}
|
||||
}
|
||||
|
||||
}
|
||||
}
|
||||
|
||||
//Component *component = loadComponentEMI(name, parent);
|
||||
|
||||
components.push_back(component);
|
||||
|
||||
ChoreTrack &track = _chores[i]->_tracks[k];
|
||||
track.numKeys = data->readUint32LE();
|
||||
track.keys = new TrackKey[track.numKeys];
|
||||
track.component = component;
|
||||
|
||||
// this is probably wrong
|
||||
track.compID = 0;
|
||||
for (int j = 0; j < track.numKeys; j++) {
|
||||
float time, value;
|
||||
data->read(&time, 4);
|
||||
data->read(&value, 4);
|
||||
track.keys[j].time = (int)time;
|
||||
track.keys[j].value = (int)value;
|
||||
}
|
||||
delete[] name;
|
||||
}
|
||||
//_chores[i]._tracks->compID;
|
||||
}
|
||||
|
||||
_numComponents = components.size();
|
||||
_components = new Component *[_numComponents];
|
||||
int i = 0;
|
||||
for (Common::List<Component *>::iterator it = components.begin(); it != components.end(); ++it, ++i) {
|
||||
_components[i] = *it;
|
||||
if (!_components[i])
|
||||
continue;
|
||||
|
||||
}
|
||||
}
|
||||
|
||||
Component *EMICostume::loadComponent(Component *parent, int parentID, const char *name, Component *prevComponent) {
|
||||
// some have an exclimation mark, this could mean something.
|
||||
// for now, return 0 otherwise it will just crash in some other part.
|
||||
//return 0;
|
||||
|
||||
assert(name[0] == '!');
|
||||
++name;
|
||||
|
||||
char type[5];
|
||||
tag32 tag = 0;
|
||||
memcpy(&tag, name, 4);
|
||||
memcpy(&type, name, 4);
|
||||
type[4] = 0;
|
||||
|
||||
name += 4;
|
||||
|
||||
if (FROM_BE_32(tag) == MKTAG('m','e','s','h')) {
|
||||
//Debug::warning(Debug::Costumes, "Actor::loadComponentEMI Implement MESH-handling: %s" , name);
|
||||
return new EMIMeshComponent(parent, parentID, name, prevComponent, tag);
|
||||
} else if (FROM_BE_32(tag) == MKTAG('s','k','e','l')) {
|
||||
//Debug::warning(Debug::Costumes, "Actor::loadComponentEMI Implement SKEL-handling: %s" , name);
|
||||
return new EMISkelComponent(parent, parentID, name, prevComponent, tag);
|
||||
} else if (FROM_BE_32(tag) == MKTAG('t','e','x','i')) {
|
||||
Debug::warning(Debug::Costumes, "Actor::loadComponentEMI Implement TEXI-handling: %s" , name);
|
||||
//return new MaterialComponent(parent, parentID, name, tag);
|
||||
} else if (FROM_BE_32(tag) == MKTAG('a','n','i','m')) {
|
||||
//Debug::warning(Debug::Costumes, "Actor::loadComponentEMI Implement ANIM-handling: %s" , name);
|
||||
return new EMIAnimComponent(parent, parentID, name, prevComponent, tag);
|
||||
} else if (FROM_BE_32(tag) == MKTAG('l','u','a','c')) {
|
||||
Debug::warning(Debug::Costumes, "Actor::loadComponentEMI Implement LUAC-handling: %s" , name);
|
||||
} else if (FROM_BE_32(tag) == MKTAG('l','u','a','v')) {
|
||||
Debug::warning(Debug::Costumes, "Actor::loadComponentEMI Implement LUAV-handling: %s" , name);
|
||||
//return new LuaVarComponent(parent, parentID, name, tag);
|
||||
} else if (FROM_BE_32(tag) == MKTAG('s','p','r','t')) {
|
||||
Debug::warning(Debug::Costumes, "Actor::loadComponentEMI Implement SPRT-handling: %s" , name);
|
||||
//return new SpriteComponent(parent, parentID, name, tag);
|
||||
} else if (FROM_BE_32(tag) == MKTAG('s','h','a','d')) {
|
||||
Debug::warning(Debug::Costumes, "Actor::loadComponentEMI Implement SHAD-handling: %s" , name);
|
||||
} else {
|
||||
error("Actor::loadComponentEMI missing tag: %s for %s", name, type);
|
||||
}
|
||||
/*
|
||||
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 NULL;
|
||||
}
|
||||
|
||||
void EMICostume::draw() {
|
||||
for (Common::List<Chore*>::iterator it = _playingChores.begin(); it != _playingChores.end(); ++it) {
|
||||
Chore *c = (*it);
|
||||
for (int i = 0; i < c->_numTracks; ++i) {
|
||||
if (c->_tracks[i].component) {
|
||||
c->_tracks[i].component->draw();
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (_emiMesh) {
|
||||
_emiMesh->draw();
|
||||
}
|
||||
}
|
||||
|
||||
int EMICostume::update(uint time) {
|
||||
for (Common::List<Chore*>::iterator i = _playingChores.begin(); i != _playingChores.end(); ++i) {
|
||||
(*i)->update(time);
|
||||
if (!(*i)->_playing) {
|
||||
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 EMICostume::saveState(SaveGame *state) const {
|
||||
// TODO
|
||||
return;
|
||||
}
|
||||
|
||||
bool EMICostume::restoreState(SaveGame *state) {
|
||||
// TODO
|
||||
return true;
|
||||
}
|
||||
|
||||
} // end of namespace Grim
|
61
engines/grim/emi/costumeemi.h
Normal file
61
engines/grim/emi/costumeemi.h
Normal file
@ -0,0 +1,61 @@
|
||||
/* ResidualVM - A 3D game interpreter
|
||||
*
|
||||
* ResidualVM 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 library is free software; you can redistribute it and/or
|
||||
* modify it under the terms of the GNU Lesser General Public
|
||||
* License as published by the Free Software Foundation; either
|
||||
* version 2.1 of the License, or (at your option) any later version.
|
||||
|
||||
* This library 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
|
||||
* Lesser General Public License for more details.
|
||||
|
||||
* You should have received a copy of the GNU Lesser General Public
|
||||
* License along with this library; if not, write to the Free Software
|
||||
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
|
||||
*
|
||||
*/
|
||||
|
||||
#ifndef GRIM_COSTUMEEMI_H
|
||||
#define GRIM_COSTUMEEMI_H
|
||||
|
||||
#include "common/stream.h"
|
||||
|
||||
#include "engines/grim/object.h"
|
||||
#include "engines/grim/costume.h"
|
||||
|
||||
namespace Grim {
|
||||
|
||||
typedef uint32 tag32;
|
||||
|
||||
class EMISkelComponent;
|
||||
class EMIMeshComponent;
|
||||
|
||||
class EMICostume : public Costume {
|
||||
public:
|
||||
EMICostume(const Common::String &filename, Costume *prevCost);
|
||||
|
||||
void load(Common::SeekableReadStream *data);
|
||||
|
||||
int update(uint frameTime);
|
||||
void draw();
|
||||
|
||||
void saveState(SaveGame *state) const;
|
||||
bool restoreState(SaveGame *state);
|
||||
|
||||
public:
|
||||
EMISkelComponent *_emiSkel;
|
||||
EMIMeshComponent *_emiMesh;
|
||||
private:
|
||||
Component *loadComponent(Component *parent, int parentID, const char *name, Component *prevComponent);
|
||||
|
||||
friend class Chore;
|
||||
};
|
||||
|
||||
} // end of namespace Grim
|
||||
|
||||
#endif
|
@ -195,8 +195,13 @@ void EMIModel::loadMesh(Common::SeekableReadStream *data) {
|
||||
}
|
||||
|
||||
void EMIModel::setSkeleton(Skeleton *skel) {
|
||||
if (_skeleton == skel) {
|
||||
return;
|
||||
}
|
||||
_skeleton = skel;
|
||||
|
||||
if (!skel || !_numBoneInfos) {
|
||||
return;
|
||||
}
|
||||
int boneVert = 0;
|
||||
delete[] _vertexBoneInfo; _vertexBoneInfo = NULL;
|
||||
delete[] _vertexBone; _vertexBone = NULL;
|
||||
@ -226,7 +231,7 @@ void EMIModel::setSkeleton(Skeleton *skel) {
|
||||
}
|
||||
|
||||
void EMIModel::prepareForRender() {
|
||||
if (!_skeleton)
|
||||
if (!_skeleton || !_vertexBoneInfo)
|
||||
return;
|
||||
for (int i = 0; i < _numVertices; i++) {
|
||||
_drawVertices[i] = _vertices[i];
|
||||
|
@ -8,12 +8,12 @@
|
||||
* modify it under the terms of the GNU Lesser General Public
|
||||
* License as published by the Free Software Foundation; either
|
||||
* version 2.1 of the License, or (at your option) any later version.
|
||||
|
||||
|
||||
* This library 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
|
||||
* Lesser General Public License for more details.
|
||||
|
||||
|
||||
* You should have received a copy of the GNU Lesser General Public
|
||||
* License along with this library; if not, write to the Free Software
|
||||
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
|
||||
@ -32,7 +32,7 @@ namespace Grim {
|
||||
#define ROTATE_OP 4
|
||||
#define TRANSLATE_OP 3
|
||||
|
||||
Skeleton::Skeleton(const Common::String &filename, Common::SeekableReadStream *data) {
|
||||
Skeleton::Skeleton(const Common::String &filename, Common::SeekableReadStream *data) : _anim(NULL), _time(0) {
|
||||
loadSkeleton(data);
|
||||
}
|
||||
|
||||
@ -52,7 +52,7 @@ void Skeleton::loadSkeleton(Common::SeekableReadStream *data) {
|
||||
data->read(inString, 32);
|
||||
_joints[i]._parent = inString;
|
||||
|
||||
_joints[i]._trans.readFromStream(data);
|
||||
_joints[i]._trans.readFromStream(data);
|
||||
_joints[i]._quat.readFromStream(data);
|
||||
|
||||
_joints[i]._parentIndex = findJointIndex(_joints[i]._parent, i);
|
||||
@ -83,13 +83,20 @@ void Skeleton::initBones() {
|
||||
}
|
||||
|
||||
void Skeleton::resetAnim() {
|
||||
_time = 0;
|
||||
for (int i = 0; i < _numJoints; i++) {
|
||||
_joints[i]._finalMatrix = _joints[i]._absMatrix;
|
||||
}
|
||||
}
|
||||
|
||||
void Skeleton::setAnim(AnimationEmi *anim) {
|
||||
if (_anim == anim) {
|
||||
return;
|
||||
}
|
||||
_anim = anim;
|
||||
if (!_anim) {
|
||||
return;
|
||||
}
|
||||
for (int i = 0; i < _numJoints; i++) {
|
||||
_joints[i]._animIndex = -1;
|
||||
}
|
||||
@ -108,16 +115,15 @@ int Skeleton::findJointIndex(Common::String name, int max) {
|
||||
return i;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
return -1;
|
||||
}
|
||||
|
||||
void Skeleton::animate(float time) {
|
||||
void Skeleton::animate(float delta) {
|
||||
if (_anim == NULL)
|
||||
return;
|
||||
|
||||
if (time > _anim->_duration) {
|
||||
time = 0.0;
|
||||
_time += delta;
|
||||
if (_time > _anim->_duration) {
|
||||
resetAnim();
|
||||
}
|
||||
|
||||
@ -143,12 +149,12 @@ void Skeleton::animate(float time) {
|
||||
// Find the right keyframe
|
||||
for (curKeyFrame = 0; curKeyFrame < _curBone->_count; curKeyFrame++) {
|
||||
if (_curBone->_operation == ROTATE_OP) {
|
||||
if (_curBone->_rotations[curKeyFrame]->_time >= time) {
|
||||
if (_curBone->_rotations[curKeyFrame]->_time >= _time) {
|
||||
keyfIdx = curKeyFrame;
|
||||
break;
|
||||
}
|
||||
} else if (_curBone->_operation == TRANSLATE_OP) {
|
||||
if (_curBone->_translations[curKeyFrame]->_time >= time) {
|
||||
if (_curBone->_translations[curKeyFrame]->_time >= _time) {
|
||||
keyfIdx = curKeyFrame;
|
||||
break;
|
||||
}
|
||||
@ -166,7 +172,7 @@ void Skeleton::animate(float time) {
|
||||
quat = _curBone->_rotations[keyfIdx-1]->_quat;
|
||||
} else {
|
||||
timeDelta = _curBone->_rotations[keyfIdx-1]->_time - _curBone->_rotations[keyfIdx]->_time;
|
||||
interpVal = (time - _curBone->_rotations[keyfIdx]->_time) / timeDelta;
|
||||
interpVal = (_time - _curBone->_rotations[keyfIdx]->_time) / timeDelta;
|
||||
|
||||
// Might be the other way around (keyfIdx - 1 slerped against keyfIdx)
|
||||
quat = _curBone->_rotations[keyfIdx]->_quat.slerpQuat(_curBone->_rotations[keyfIdx - 1]->_quat, interpVal);
|
||||
@ -179,7 +185,7 @@ void Skeleton::animate(float time) {
|
||||
vec = _curBone->_translations[keyfIdx-1]->_vec;
|
||||
} else {
|
||||
timeDelta = _curBone->_translations[keyfIdx-1]->_time - _curBone->_translations[keyfIdx]->_time;
|
||||
interpVal = (time - _curBone->_translations[keyfIdx]->_time) / timeDelta;
|
||||
interpVal = (_time - _curBone->_translations[keyfIdx]->_time) / timeDelta;
|
||||
|
||||
vec.x() = _curBone->_translations[keyfIdx-1]->_vec.x() +
|
||||
(_curBone->_translations[keyfIdx]->_vec.x() - _curBone->_translations[keyfIdx-1]->_vec.x()) * interpVal;
|
||||
|
@ -8,12 +8,12 @@
|
||||
* modify it under the terms of the GNU Lesser General Public
|
||||
* License as published by the Free Software Foundation; either
|
||||
* version 2.1 of the License, or (at your option) any later version.
|
||||
|
||||
|
||||
* This library 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
|
||||
* Lesser General Public License for more details.
|
||||
|
||||
|
||||
* You should have received a copy of the GNU Lesser General Public
|
||||
* License along with this library; if not, write to the Free Software
|
||||
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
|
||||
@ -64,6 +64,7 @@ public:
|
||||
void setAnim(AnimationEmi *anim);
|
||||
void animate(float time);
|
||||
int findJointIndex(Common::String name, int max);
|
||||
float _time;
|
||||
};
|
||||
|
||||
} // end of namespace Grim
|
||||
|
@ -24,6 +24,7 @@ MODULE_OBJS := \
|
||||
emi/sound/emisound.o \
|
||||
emi/sound/codecs/scx.o \
|
||||
emi/animationemi.o \
|
||||
emi/costumeemi.o \
|
||||
emi/modelemi.o \
|
||||
emi/skeleton.o \
|
||||
emi/lua_v2.o \
|
||||
|
@ -35,6 +35,7 @@
|
||||
#include "engines/grim/inputdialog.h"
|
||||
#include "engines/grim/debug.h"
|
||||
#include "engines/grim/emi/animationemi.h"
|
||||
#include "engines/grim/emi/costumeemi.h"
|
||||
#include "engines/grim/emi/modelemi.h"
|
||||
#include "engines/grim/emi/skeleton.h"
|
||||
#include "engines/grim/patchr.h"
|
||||
@ -330,10 +331,15 @@ Costume *ResourceLoader::loadCostume(const Common::String &filename, Costume *pr
|
||||
if (!stream) {
|
||||
error("Could not find costume \"%s\"", filename.c_str());
|
||||
}
|
||||
|
||||
Costume *result = new Costume(filename, stream, prevCost);
|
||||
Costume *result;
|
||||
if (g_grim->getGameType() == GType_MONKEY4) {
|
||||
result = new EMICostume(filename, prevCost);
|
||||
} else {
|
||||
result = new Costume(filename, prevCost);
|
||||
}
|
||||
result->load(stream);
|
||||
delete stream;
|
||||
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
@ -461,7 +467,7 @@ AnimationEmi *ResourceLoader::loadAnimationEmi(const Common::String &filename) {
|
||||
|
||||
AnimationEmi *result = new AnimationEmi(filename, stream);
|
||||
delete stream;
|
||||
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
|
Loading…
x
Reference in New Issue
Block a user