mirror of
https://github.com/libretro/scummvm.git
synced 2025-01-27 05:32:45 +00:00
374 lines
9.6 KiB
C++
374 lines
9.6 KiB
C++
/* 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 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/stream.h"
|
|
#include "math/vector3d.h"
|
|
#include "math/quat.h"
|
|
#include "engines/grim/resource.h"
|
|
#include "engines/grim/emi/animationemi.h"
|
|
#include "common/textconsole.h"
|
|
|
|
namespace Grim {
|
|
|
|
AnimationEmi::AnimationEmi(const Common::String &filename, Common::SeekableReadStream *data) :
|
|
_name(""), _duration(0.0f), _numBones(0), _bones(nullptr) {
|
|
_fname = filename;
|
|
loadAnimation(data);
|
|
}
|
|
|
|
// Use modelemi's solution for the LA-strings.
|
|
void AnimationEmi::loadAnimation(Common::SeekableReadStream *data) {
|
|
int len = data->readUint32LE();
|
|
char *inString = new char[len];
|
|
data->read(inString, len);
|
|
_name = inString;
|
|
delete[] inString;
|
|
|
|
char temp[4];
|
|
data->read(temp, 4);
|
|
_duration = 1000 * get_float(temp);
|
|
_numBones = data->readUint32LE();
|
|
|
|
_bones = new Bone[_numBones];
|
|
for (int i = 0; i < _numBones; i++) {
|
|
_bones[i].loadBinary(data);
|
|
}
|
|
}
|
|
|
|
AnimationEmi::~AnimationEmi() {
|
|
g_resourceloader->uncacheAnimationEmi(this);
|
|
delete[] _bones;
|
|
}
|
|
|
|
void Bone::loadBinary(Common::SeekableReadStream *data) {
|
|
uint32 len = data->readUint32LE();
|
|
char *inString = new char[len];
|
|
data->read(inString, len);
|
|
_boneName = inString;
|
|
delete[] inString;
|
|
_operation = data->readUint32LE();
|
|
_priority = data->readUint32LE();
|
|
_c = data->readUint32LE();
|
|
_count = data->readUint32LE();
|
|
|
|
char temp[4];
|
|
if (_operation == 3) { // Translation
|
|
_translations = new AnimTranslation[_count];
|
|
for (int j = 0; j < _count; j++) {
|
|
_translations[j]._vec.readFromStream(data);
|
|
data->read(temp, 4);
|
|
_translations[j]._time = 1000 * get_float(temp);
|
|
}
|
|
} else if (_operation == 4) { // Rotation
|
|
_rotations = new AnimRotation[_count];
|
|
for (int j = 0; j < _count; j++) {
|
|
_rotations[j]._quat.readFromStream(data);
|
|
data->read(temp, 4);
|
|
_rotations[j]._time = 1000 * get_float(temp);
|
|
}
|
|
} else {
|
|
error("Unknown animation-operation %d", _operation);
|
|
}
|
|
}
|
|
|
|
Bone::~Bone() {
|
|
if (_operation == 3) {
|
|
delete[] _translations;
|
|
} else if (_operation == 4) {
|
|
delete[] _rotations;
|
|
}
|
|
}
|
|
|
|
AnimationStateEmi::AnimationStateEmi(const Common::String &anim) :
|
|
_skel(nullptr), _looping(false), _active(false),
|
|
_fadeMode(Animation::None), _fade(1.0f), _fadeLength(0), _time(0), _startFade(1.0f),
|
|
_boneJoints(nullptr) {
|
|
_anim = g_resourceloader->getAnimationEmi(anim);
|
|
if (_anim)
|
|
_boneJoints = new int[_anim->_numBones];
|
|
}
|
|
|
|
AnimationStateEmi::~AnimationStateEmi() {
|
|
deactivate();
|
|
delete[] _boneJoints;
|
|
}
|
|
|
|
void AnimationStateEmi::activate() {
|
|
if (!_active) {
|
|
_active = true;
|
|
if (_skel)
|
|
_skel->addAnimation(this);
|
|
}
|
|
}
|
|
|
|
void AnimationStateEmi::deactivate() {
|
|
if (_active) {
|
|
_active = false;
|
|
if (_skel)
|
|
_skel->removeAnimation(this);
|
|
}
|
|
}
|
|
|
|
void AnimationStateEmi::update(uint time) {
|
|
if (!_active)
|
|
return;
|
|
|
|
if (!_anim) {
|
|
deactivate();
|
|
return;
|
|
}
|
|
|
|
if (!_paused) {
|
|
uint durationMs = (uint)_anim->_duration;
|
|
if (_time >= durationMs) {
|
|
if (_looping) {
|
|
_time = _time % durationMs;
|
|
} else {
|
|
if (_fadeMode != Animation::FadeOut)
|
|
deactivate();
|
|
}
|
|
}
|
|
_time += time;
|
|
}
|
|
|
|
if (_fadeMode != Animation::None) {
|
|
if (_fadeMode == Animation::FadeIn) {
|
|
_fade += (float)time * (1.0f - _startFade) / _fadeLength;
|
|
if (_fade >= 1.f) {
|
|
_fade = 1.f;
|
|
_fadeMode = Animation::None;
|
|
}
|
|
} else {
|
|
_fade -= (float)time * _startFade / _fadeLength;
|
|
if (_fade <= 0.f) {
|
|
_fade = 0.f;
|
|
// Don't reset the _fadeMode here. This way if fadeOut() was called
|
|
// on a looping chore its keyframe animations will remain faded out
|
|
// when it calls play() again.
|
|
deactivate();
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void AnimationStateEmi::computeWeights() {
|
|
if (_fade <= 0.0f)
|
|
return;
|
|
|
|
for (int bone = 0; bone < _anim->_numBones; ++bone) {
|
|
Bone &curBone = _anim->_bones[bone];
|
|
int jointIndex = _boneJoints[bone];
|
|
if (jointIndex == -1)
|
|
continue;
|
|
|
|
AnimationLayer *layer = _skel->getLayer(curBone._priority);
|
|
JointAnimation &jointAnim = layer->_jointAnims[jointIndex];
|
|
|
|
if (curBone._rotations) {
|
|
jointAnim._rotWeight += _fade;
|
|
}
|
|
if (curBone._translations) {
|
|
jointAnim._transWeight += _fade;
|
|
}
|
|
}
|
|
}
|
|
|
|
void AnimationStateEmi::animate() {
|
|
if (_fade <= 0.0f)
|
|
return;
|
|
|
|
for (int bone = 0; bone < _anim->_numBones; ++bone) {
|
|
Bone &curBone = _anim->_bones[bone];
|
|
int jointIndex = _boneJoints[bone];
|
|
if (jointIndex == -1)
|
|
continue;
|
|
|
|
Joint *target = &_skel->_joints[jointIndex];
|
|
AnimationLayer *layer = _skel->getLayer(curBone._priority);
|
|
JointAnimation &jointAnim = layer->_jointAnims[jointIndex];
|
|
|
|
if (curBone._rotations) {
|
|
int keyfIdx = -1;
|
|
Math::Quaternion quat;
|
|
|
|
// Normalize the weight so that the sum of applied weights will equal 1.
|
|
float normalizedRotWeight = _fade;
|
|
if (jointAnim._rotWeight > 1.0f) {
|
|
// Note: Division by unnormalized sum of weights.
|
|
normalizedRotWeight = _fade / jointAnim._rotWeight;
|
|
}
|
|
|
|
for (int curKeyFrame = 0; curKeyFrame < curBone._count; curKeyFrame++) {
|
|
if (curBone._rotations[curKeyFrame]._time >= _time) {
|
|
keyfIdx = curKeyFrame;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (keyfIdx == 0) {
|
|
quat = curBone._rotations[0]._quat;
|
|
}
|
|
else if (keyfIdx != -1) {
|
|
float timeDelta = curBone._rotations[keyfIdx]._time - curBone._rotations[keyfIdx - 1]._time;
|
|
float interpVal = (_time - curBone._rotations[keyfIdx - 1]._time) / timeDelta;
|
|
|
|
quat = curBone._rotations[keyfIdx - 1]._quat.slerpQuat(curBone._rotations[keyfIdx]._quat, interpVal);
|
|
}
|
|
else {
|
|
quat = curBone._rotations[curBone._count - 1]._quat;
|
|
}
|
|
|
|
Math::Quaternion &quatFinal = jointAnim._quat;
|
|
quat = target->_quat.inverse() * quat;
|
|
quat = quatFinal * quat;
|
|
quatFinal = quatFinal.slerpQuat(quat, normalizedRotWeight);
|
|
}
|
|
|
|
if (curBone._translations) {
|
|
int keyfIdx = -1;
|
|
Math::Vector3d vec;
|
|
|
|
// Normalize the weight so that the sum of applied weights will equal 1.
|
|
float normalizedTransWeight = _fade;
|
|
if (jointAnim._transWeight > 1.0f) {
|
|
// Note: Division by unnormalized sum of weights.
|
|
normalizedTransWeight = _fade / jointAnim._transWeight;
|
|
}
|
|
|
|
for (int curKeyFrame = 0; curKeyFrame < curBone._count; curKeyFrame++) {
|
|
if (curBone._translations[curKeyFrame]._time >= _time) {
|
|
keyfIdx = curKeyFrame;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (keyfIdx == 0) {
|
|
vec = curBone._translations[0]._vec;
|
|
}
|
|
else if (keyfIdx != -1) {
|
|
float timeDelta = curBone._translations[keyfIdx]._time - curBone._translations[keyfIdx - 1]._time;
|
|
float interpVal = (_time - curBone._translations[keyfIdx - 1]._time) / timeDelta;
|
|
|
|
vec = curBone._translations[keyfIdx - 1]._vec +
|
|
(curBone._translations[keyfIdx]._vec - curBone._translations[keyfIdx - 1]._vec) * interpVal;
|
|
}
|
|
else {
|
|
vec = curBone._translations[curBone._count - 1]._vec;
|
|
}
|
|
|
|
Math::Vector3d &posFinal = jointAnim._pos;
|
|
vec = vec - target->_relMatrix.getPosition();
|
|
posFinal = posFinal + vec * normalizedTransWeight;
|
|
}
|
|
}
|
|
}
|
|
|
|
void AnimationStateEmi::play() {
|
|
if (!_active) {
|
|
_time = 0;
|
|
if (_fadeMode == Animation::FadeOut)
|
|
_fadeMode = Animation::None;
|
|
if (_fadeMode == Animation::FadeIn || _fade > 0.f)
|
|
activate();
|
|
}
|
|
_paused = false;
|
|
}
|
|
|
|
void AnimationStateEmi::stop() {
|
|
_fadeMode = Animation::None;
|
|
_time = 0;
|
|
deactivate();
|
|
}
|
|
|
|
void AnimationStateEmi::setPaused(bool paused) {
|
|
_paused = paused;
|
|
}
|
|
|
|
void AnimationStateEmi::setLooping(bool loop) {
|
|
_looping = loop;
|
|
}
|
|
|
|
void AnimationStateEmi::setSkeleton(Skeleton *skel) {
|
|
if (skel != _skel) {
|
|
if (_skel)
|
|
_skel->removeAnimation(this);
|
|
_skel = skel;
|
|
if (_active)
|
|
skel->addAnimation(this);
|
|
|
|
if (_anim) {
|
|
for (int i = 0; i < _anim->_numBones; ++i) {
|
|
_boneJoints[i] = skel->findJointIndex(_anim->_bones[i]._boneName);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void AnimationStateEmi::fade(Animation::FadeMode mode, int fadeLength) {
|
|
if (mode == Animation::None) {
|
|
_fade = 1.f;
|
|
} else if (_fadeMode != Animation::FadeOut && mode == Animation::FadeIn) {
|
|
_fade = 0.f;
|
|
}
|
|
_startFade = _fade;
|
|
_fadeMode = mode;
|
|
_fadeLength = fadeLength;
|
|
}
|
|
|
|
void AnimationStateEmi::advance(uint msecs) {
|
|
_time += msecs;
|
|
}
|
|
|
|
void AnimationStateEmi::saveState(SaveGame *state) {
|
|
state->writeBool(_looping);
|
|
state->writeBool(_active);
|
|
state->writeBool(_paused);
|
|
state->writeLEUint32(_time);
|
|
state->writeFloat(_fade);
|
|
state->writeFloat(_startFade);
|
|
state->writeLESint32((int)_fadeMode);
|
|
state->writeLESint32(_fadeLength);
|
|
}
|
|
|
|
void AnimationStateEmi::restoreState(SaveGame *state) {
|
|
if (state->saveMinorVersion() >= 10) {
|
|
_looping = state->readBool();
|
|
bool active = state->readBool();
|
|
_paused = state->readBool();
|
|
if (state->saveMinorVersion() < 22) {
|
|
_time = (uint)state->readFloat();
|
|
} else {
|
|
_time = state->readLEUint32();
|
|
}
|
|
_fade = state->readFloat();
|
|
_startFade = state->readFloat();
|
|
_fadeMode = (Animation::FadeMode)state->readLESint32();
|
|
_fadeLength = state->readLESint32();
|
|
|
|
if (active)
|
|
activate();
|
|
}
|
|
}
|
|
|
|
} // end of namespace Grim
|