mirror of
https://github.com/libretro/scummvm.git
synced 2025-01-23 11:04:44 +00:00
f2c9137979
now that we are using c++11 we don't need this wrapper anymore
2639 lines
74 KiB
C++
2639 lines
74 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 "math/line3d.h"
|
|
#include "math/rect2d.h"
|
|
|
|
#include "engines/grim/debug.h"
|
|
#include "engines/grim/actor.h"
|
|
#include "engines/grim/grim.h"
|
|
#include "engines/grim/costume.h"
|
|
#include "engines/grim/lipsync.h"
|
|
#include "engines/grim/movie/movie.h"
|
|
#include "engines/grim/sound.h"
|
|
#include "engines/grim/lua.h"
|
|
#include "engines/grim/resource.h"
|
|
#include "engines/grim/savegame.h"
|
|
#include "engines/grim/set.h"
|
|
#include "engines/grim/gfx_base.h"
|
|
#include "engines/grim/model.h"
|
|
#include "engines/grim/emi/emi.h"
|
|
#include "engines/grim/emi/costumeemi.h"
|
|
#include "engines/grim/emi/skeleton.h"
|
|
#include "engines/grim/emi/costume/emiskel_component.h"
|
|
#include "engines/grim/emi/modelemi.h"
|
|
|
|
namespace Grim {
|
|
|
|
Shadow::Shadow() :
|
|
active(false), dontNegate(false), userData(nullptr) {
|
|
}
|
|
|
|
static int animTurn(float turnAmt, const Math::Angle &dest, Math::Angle *cur) {
|
|
Math::Angle d = dest - *cur;
|
|
d.normalize(-180);
|
|
// If the actor won't turn because the rate is set to zero then
|
|
// have the actor turn all the way to the destination yaw.
|
|
// Without this some actors will lock the interface on changing
|
|
// scenes, this affects the Bone Wagon in particular.
|
|
if (turnAmt == 0 || turnAmt >= fabsf(d.getDegrees())) {
|
|
*cur = dest;
|
|
} else if (d > 0) {
|
|
*cur += turnAmt;
|
|
} else {
|
|
*cur -= turnAmt;
|
|
}
|
|
if (d != 0) {
|
|
return (d > 0 ? 1 : -1);
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
bool Actor::_isTalkingBackground = false;
|
|
|
|
void Actor::saveStaticState(SaveGame *state) {
|
|
state->writeBool(_isTalkingBackground);
|
|
}
|
|
|
|
void Actor::restoreStaticState(SaveGame *state) {
|
|
_isTalkingBackground = state->readBool();
|
|
}
|
|
|
|
Actor::Actor() :
|
|
_talkColor(255, 255, 255), _pos(0, 0, 0),
|
|
_lookingMode(false), _followBoxes(false), _running(false),
|
|
_pitch(0), _yaw(0), _roll(0), _walkRate(0.3f),
|
|
_turnRateMultiplier(0.f), _talkAnim(0),
|
|
_reflectionAngle(80), _scale(1.f), _timeScale(1.f),
|
|
_visible(true), _lipSync(nullptr), _turning(false), _singleTurning(false), _walking(false),
|
|
_walkedLast(false), _walkedCur(false),
|
|
_collisionMode(CollisionOff), _collisionScale(1.f),
|
|
_lastTurnDir(0), _currTurnDir(0),
|
|
_sayLineText(0), _talkDelay(0),
|
|
_attachedActor(0), _attachedJoint(""),
|
|
_globalAlpha(1.f), _alphaMode(AlphaOff),
|
|
_mustPlaceText(false),
|
|
_puckOrient(false), _talking(false),
|
|
_inOverworld(false), _drawnToClean(false), _backgroundTalk(false),
|
|
_sortOrder(0), _useParentSortOrder(false),
|
|
_sectorSortOrder(-1), _fakeUnbound(false), _lightMode(LightFastDyn),
|
|
_hasFollowedBoxes(false), _lookAtActor(0) {
|
|
|
|
// Some actors don't set walk and turn rates, so we default the
|
|
// _turnRate so Doug at the cat races can turn and we set the
|
|
// _walkRate so Glottis at the demon beaver entrance can walk and
|
|
// so Chepito in su.set
|
|
_turnRate = 100.0f;
|
|
|
|
_activeShadowSlot = -1;
|
|
_shadowArray = new Shadow[MAX_SHADOWS];
|
|
}
|
|
|
|
Actor::~Actor() {
|
|
if (_shadowArray) {
|
|
clearShadowPlanes();
|
|
delete[] _shadowArray;
|
|
}
|
|
while (!_costumeStack.empty()) {
|
|
delete _costumeStack.back();
|
|
_costumeStack.pop_back();
|
|
}
|
|
g_grim->immediatelyRemoveActor(this);
|
|
}
|
|
|
|
void Actor::saveState(SaveGame *savedState) const {
|
|
// store actor name
|
|
savedState->writeString(_name);
|
|
savedState->writeString(_setName);
|
|
|
|
savedState->writeColor(_talkColor);
|
|
|
|
savedState->writeVector3d(_pos);
|
|
|
|
savedState->writeFloat(_pitch.getDegrees());
|
|
savedState->writeFloat(_yaw.getDegrees());
|
|
savedState->writeFloat(_roll.getDegrees());
|
|
savedState->writeFloat(_walkRate);
|
|
savedState->writeFloat(_turnRate);
|
|
savedState->writeFloat(_turnRateMultiplier);
|
|
savedState->writeBool(_followBoxes);
|
|
savedState->writeFloat(_reflectionAngle);
|
|
savedState->writeBool(_visible);
|
|
savedState->writeBool(_lookingMode);
|
|
savedState->writeFloat(_scale);
|
|
savedState->writeFloat(_timeScale);
|
|
savedState->writeBool(_puckOrient);
|
|
|
|
savedState->writeString(_talkSoundName);
|
|
savedState->writeBool(_talking);
|
|
savedState->writeBool(_backgroundTalk);
|
|
|
|
savedState->writeLEUint32((uint32)_collisionMode);
|
|
savedState->writeFloat(_collisionScale);
|
|
|
|
if (_lipSync) {
|
|
savedState->writeBool(true);
|
|
savedState->writeString(_lipSync->getFilename());
|
|
} else {
|
|
savedState->writeBool(false);
|
|
}
|
|
|
|
savedState->writeLEUint32(_costumeStack.size());
|
|
for (Common::List<Costume *>::const_iterator i = _costumeStack.begin(); i != _costumeStack.end(); ++i) {
|
|
Costume *c = *i;
|
|
savedState->writeString(c->getFilename());
|
|
Costume *pc = c->getPreviousCostume();
|
|
int depth = 0;
|
|
while (pc) {
|
|
++depth;
|
|
pc = pc->getPreviousCostume();
|
|
}
|
|
savedState->writeLESint32(depth);
|
|
pc = c->getPreviousCostume();
|
|
for (int j = 0; j < depth; ++j) { //save the previousCostume hierarchy
|
|
savedState->writeString(pc->getFilename());
|
|
pc = pc->getPreviousCostume();
|
|
}
|
|
c->saveState(savedState);
|
|
}
|
|
|
|
savedState->writeBool(_turning);
|
|
savedState->writeBool(_singleTurning);
|
|
savedState->writeFloat(_moveYaw.getDegrees());
|
|
savedState->writeFloat(_movePitch.getDegrees());
|
|
savedState->writeFloat(_moveRoll.getDegrees());
|
|
|
|
savedState->writeBool(_walking);
|
|
savedState->writeVector3d(_destPos);
|
|
|
|
_restChore.saveState(savedState);
|
|
|
|
_walkChore.saveState(savedState);
|
|
savedState->writeBool(_walkedLast);
|
|
savedState->writeBool(_walkedCur);
|
|
|
|
_leftTurnChore.saveState(savedState);
|
|
_rightTurnChore.saveState(savedState);
|
|
savedState->writeLESint32(_lastTurnDir);
|
|
savedState->writeLESint32(_currTurnDir);
|
|
|
|
for (int i = 0; i < 10; ++i) {
|
|
_talkChore[i].saveState(savedState);
|
|
}
|
|
savedState->writeLESint32(_talkAnim);
|
|
|
|
_mumbleChore.saveState(savedState);
|
|
|
|
for (int i = 0; i < MAX_SHADOWS; ++i) {
|
|
Shadow &shadow = _shadowArray[i];
|
|
savedState->writeString(shadow.name);
|
|
|
|
savedState->writeVector3d(shadow.pos);
|
|
|
|
savedState->writeLEUint32(shadow.planeList.size());
|
|
// Cannot use g_grim->getCurrSet() here because an actor can have walk planes
|
|
// from other scenes. It happens e.g. when Membrillo calls Velasco to tell him
|
|
// Naranja is dead.
|
|
for (SectorListType::iterator j = shadow.planeList.begin(); j != shadow.planeList.end(); ++j) {
|
|
Plane &p = *j;
|
|
savedState->writeString(p.setName);
|
|
savedState->writeString(p.sector->getName());
|
|
}
|
|
|
|
savedState->writeLESint32(0);
|
|
savedState->writeBool(shadow.active);
|
|
savedState->writeBool(shadow.dontNegate);
|
|
}
|
|
savedState->writeLESint32(_activeShadowSlot);
|
|
|
|
savedState->writeLESint32(_sayLineText);
|
|
|
|
savedState->writeVector3d(_lookAtVector);
|
|
|
|
savedState->writeLEUint32(_path.size());
|
|
for (Common::List<Math::Vector3d>::const_iterator i = _path.begin(); i != _path.end(); ++i) {
|
|
savedState->writeVector3d(*i);
|
|
}
|
|
|
|
if (g_grim->getGameType() == GType_MONKEY4) {
|
|
savedState->writeLEUint32(_alphaMode);
|
|
savedState->writeFloat(_globalAlpha);
|
|
|
|
savedState->writeBool(_inOverworld);
|
|
savedState->writeLESint32(_sortOrder);
|
|
savedState->writeBool(_useParentSortOrder);
|
|
|
|
savedState->writeLESint32(_attachedActor);
|
|
savedState->writeString(_attachedJoint);
|
|
|
|
Common::List<MaterialPtr>::const_iterator it = _materials.begin();
|
|
for (; it != _materials.end(); ++it) {
|
|
if (*it) {
|
|
warning("%s", (*it)->getFilename().c_str());
|
|
savedState->writeLESint32((*it)->getActiveTexture());
|
|
}
|
|
}
|
|
|
|
savedState->writeLESint32(_lookAtActor);
|
|
|
|
savedState->writeLEUint32(_localAlpha.size());
|
|
for (unsigned int i = 0; i < _localAlpha.size(); i++) {
|
|
savedState->writeFloat(_localAlpha[i]);
|
|
}
|
|
savedState->writeLEUint32(_localAlphaMode.size());
|
|
for (unsigned int i = 0; i < _localAlphaMode.size(); i++) {
|
|
savedState->writeLESint32(_localAlphaMode[i]);
|
|
}
|
|
}
|
|
|
|
savedState->writeBool(_drawnToClean);
|
|
savedState->writeBool(_fakeUnbound);
|
|
}
|
|
|
|
bool Actor::restoreState(SaveGame *savedState) {
|
|
for (Common::List<Costume *>::const_iterator i = _costumeStack.begin(); i != _costumeStack.end(); ++i) {
|
|
delete *i;
|
|
}
|
|
_costumeStack.clear();
|
|
_materials.clear();
|
|
|
|
// load actor name
|
|
_name = savedState->readString();
|
|
_setName = savedState->readString();
|
|
|
|
_talkColor = savedState->readColor();
|
|
|
|
_pos = savedState->readVector3d();
|
|
_pitch = savedState->readFloat();
|
|
_yaw = savedState->readFloat();
|
|
_roll = savedState->readFloat();
|
|
_walkRate = savedState->readFloat();
|
|
_turnRate = savedState->readFloat();
|
|
if (savedState->saveMinorVersion() > 6) {
|
|
_turnRateMultiplier = savedState->readFloat();
|
|
} else {
|
|
_turnRateMultiplier = 1.f;
|
|
}
|
|
_followBoxes = savedState->readBool();
|
|
_reflectionAngle = savedState->readFloat();
|
|
_visible = savedState->readBool();
|
|
_lookingMode = savedState->readBool();
|
|
_scale = savedState->readFloat();
|
|
_timeScale = savedState->readFloat();
|
|
_puckOrient = savedState->readBool();
|
|
|
|
_talkSoundName = savedState->readString();
|
|
_talking = savedState->readBool();
|
|
_backgroundTalk = savedState->readBool();
|
|
if (isTalking()) {
|
|
g_grim->addTalkingActor(this);
|
|
}
|
|
|
|
_collisionMode = (CollisionMode)savedState->readLEUint32();
|
|
_collisionScale = savedState->readFloat();
|
|
|
|
if (savedState->readBool()) {
|
|
Common::String fn = savedState->readString();
|
|
_lipSync = g_resourceloader->getLipSync(fn);
|
|
} else {
|
|
_lipSync = nullptr;
|
|
}
|
|
|
|
uint32 size = savedState->readLEUint32();
|
|
for (uint32 i = 0; i < size; ++i) {
|
|
Common::String fname = savedState->readString();
|
|
const int depth = savedState->readLESint32();
|
|
Costume *pc = nullptr;
|
|
if (depth > 0) { //build all the previousCostume hierarchy
|
|
Common::String *names = new Common::String[depth];
|
|
for (int j = 0; j < depth; ++j) {
|
|
names[j] = savedState->readString();
|
|
}
|
|
for (int j = depth - 1; j >= 0; --j) {
|
|
pc = findCostume(names[j]);
|
|
if (!pc) {
|
|
pc = g_resourceloader->loadCostume(names[j], this, pc);
|
|
}
|
|
}
|
|
delete[] names;
|
|
}
|
|
|
|
Costume *c = g_resourceloader->loadCostume(fname, this, pc);
|
|
c->restoreState(savedState);
|
|
_costumeStack.push_back(c);
|
|
}
|
|
|
|
_turning = savedState->readBool();
|
|
if (savedState->saveMinorVersion() > 25) {
|
|
_singleTurning = savedState->readBool();
|
|
}
|
|
_moveYaw = savedState->readFloat();
|
|
if (savedState->saveMinorVersion() > 6) {
|
|
_movePitch = savedState->readFloat();
|
|
_moveRoll = savedState->readFloat();
|
|
} else {
|
|
_movePitch = _pitch;
|
|
_moveRoll = _roll;
|
|
}
|
|
|
|
_walking = savedState->readBool();
|
|
_destPos = savedState->readVector3d();
|
|
|
|
_restChore.restoreState(savedState, this);
|
|
|
|
_walkChore.restoreState(savedState, this);
|
|
_walkedLast = savedState->readBool();
|
|
_walkedCur = savedState->readBool();
|
|
|
|
_leftTurnChore.restoreState(savedState, this);
|
|
_rightTurnChore.restoreState(savedState, this);
|
|
_lastTurnDir = savedState->readLESint32();
|
|
_currTurnDir = savedState->readLESint32();
|
|
|
|
for (int i = 0; i < 10; ++i) {
|
|
_talkChore[i].restoreState(savedState, this);
|
|
}
|
|
_talkAnim = savedState->readLESint32();
|
|
|
|
_mumbleChore.restoreState(savedState, this);
|
|
|
|
clearShadowPlanes();
|
|
|
|
int maxShadows = MAX_SHADOWS;
|
|
if (savedState->saveMinorVersion() < 24)
|
|
maxShadows = 5;
|
|
for (int i = 0; i < maxShadows; ++i) {
|
|
Shadow &shadow = _shadowArray[i];
|
|
shadow.name = savedState->readString();
|
|
|
|
shadow.pos = savedState->readVector3d();
|
|
|
|
size = savedState->readLEUint32();
|
|
Set *scene = nullptr;
|
|
for (uint32 j = 0; j < size; ++j) {
|
|
Common::String actorSetName = savedState->readString();
|
|
Common::String secName = savedState->readString();
|
|
if (!scene || scene->getName() != actorSetName) {
|
|
scene = g_grim->findSet(actorSetName);
|
|
}
|
|
if (scene) {
|
|
addShadowPlane(secName.c_str(), scene, i);
|
|
} else {
|
|
warning("%s: No scene \"%s\" found, cannot restore shadow on sector \"%s\"", getName().c_str(), actorSetName.c_str(), secName.c_str());
|
|
}
|
|
}
|
|
|
|
int shadowMaskSize = savedState->readLESint32();
|
|
for (int s = 0; s < shadowMaskSize; s++) {
|
|
savedState->readByte();
|
|
}
|
|
shadow.active = savedState->readBool();
|
|
shadow.dontNegate = savedState->readBool();
|
|
}
|
|
_activeShadowSlot = savedState->readLESint32();
|
|
|
|
_sayLineText = savedState->readLESint32();
|
|
|
|
_lookAtVector = savedState->readVector3d();
|
|
|
|
size = savedState->readLEUint32();
|
|
_path.clear();
|
|
for (uint32 i = 0; i < size; ++i) {
|
|
_path.push_back(savedState->readVector3d());
|
|
}
|
|
|
|
if (g_grim->getGameType() == GType_MONKEY4) {
|
|
_alphaMode = (AlphaMode) savedState->readLEUint32();
|
|
_globalAlpha = savedState->readFloat();
|
|
|
|
_inOverworld = savedState->readBool();
|
|
_sortOrder = savedState->readLESint32();
|
|
if (savedState->saveMinorVersion() >= 22)
|
|
_useParentSortOrder = savedState->readBool();
|
|
|
|
if (savedState->saveMinorVersion() < 18)
|
|
savedState->readBool(); // Used to be _shadowActive.
|
|
|
|
_attachedActor = savedState->readLESint32();
|
|
_attachedJoint = savedState->readString();
|
|
|
|
// will be recalculated in next update()
|
|
_sectorSortOrder = -1;
|
|
|
|
if (savedState->saveMinorVersion() < 24) {
|
|
// Used to be the wear chore.
|
|
if (savedState->readBool()) {
|
|
savedState->readString();
|
|
}
|
|
savedState->readLESint32();
|
|
}
|
|
|
|
if (savedState->saveMinorVersion() >= 13) {
|
|
Common::List<MaterialPtr>::const_iterator it = _materials.begin();
|
|
for (; it != _materials.end(); ++it) {
|
|
if (*it) {
|
|
(*it)->setActiveTexture(savedState->readLESint32());
|
|
}
|
|
}
|
|
}
|
|
|
|
if (savedState->saveMinorVersion() >= 17)
|
|
_lookAtActor = savedState->readLESint32();
|
|
|
|
if (savedState->saveMinorVersion() >= 25) {
|
|
_localAlpha.resize(savedState->readLEUint32());
|
|
for (unsigned int i = 0; i < _localAlpha.size(); i++) {
|
|
_localAlpha[i] = savedState->readFloat();
|
|
}
|
|
|
|
_localAlphaMode.resize(savedState->readLEUint32());
|
|
for (unsigned int i = 0; i < _localAlphaMode.size(); i++) {
|
|
_localAlphaMode[i] = savedState->readLESint32();
|
|
}
|
|
}
|
|
}
|
|
|
|
if (savedState->saveMinorVersion() >= 4) {
|
|
_drawnToClean = savedState->readBool();
|
|
} else {
|
|
_drawnToClean = false;
|
|
}
|
|
if (savedState->saveMinorVersion() >= 27) {
|
|
_fakeUnbound = savedState->readBool();
|
|
} else {
|
|
// Note: actor will stay invisible until re-bound to set
|
|
// by game scripts.
|
|
_fakeUnbound = false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
void Actor::setRot(const Math::Vector3d &pos) {
|
|
Math::Angle y, p, r;
|
|
calculateOrientation(pos, &p, &y, &r);
|
|
setRot(p, y, r);
|
|
}
|
|
|
|
void Actor::setRot(const Math::Angle &pitchParam, const Math::Angle &yawParam, const Math::Angle &rollParam) {
|
|
_pitch = pitchParam;
|
|
_yaw = yawParam;
|
|
_moveYaw = _yaw;
|
|
_roll = rollParam;
|
|
_turning = false;
|
|
}
|
|
|
|
void Actor::setPos(const Math::Vector3d &position) {
|
|
_walking = false;
|
|
_pos = position;
|
|
|
|
// Don't allow positions outside the sectors.
|
|
// This is necessary after solving the tree pump puzzle, when the bone
|
|
// wagon returns to the signopost set.
|
|
if (_followBoxes) {
|
|
g_grim->getCurrSet()->findClosestSector(_pos, nullptr, &_pos);
|
|
}
|
|
|
|
if (g_grim->getGameType() == GType_MONKEY4) {
|
|
Math::Vector3d moveVec = position - _pos;
|
|
for (Actor *a : g_grim->getActiveActors()) {
|
|
handleCollisionWith(a, _collisionMode, &moveVec);
|
|
}
|
|
}
|
|
}
|
|
|
|
void Actor::calculateOrientation(const Math::Vector3d &pos, Math::Angle *pitch, Math::Angle *yaw, Math::Angle *roll) {
|
|
Math::Vector3d actorForward(0.f, 1.f, 0.f);
|
|
Math::Vector3d actorUp(0.f, 0.f, 1.f);
|
|
Math::Vector3d lookVector = pos - _pos;
|
|
lookVector.normalize();
|
|
|
|
// EMI: Y is up-down, actors use an X-Z plane for movement
|
|
if (g_grim->getGameType() == GType_MONKEY4) {
|
|
float temp = lookVector.z();
|
|
lookVector.x() = -lookVector.x();
|
|
lookVector.z() = lookVector.y();
|
|
lookVector.y() = temp;
|
|
}
|
|
|
|
Math::Vector3d up = actorUp;
|
|
if (_puckOrient) {
|
|
Sector *s = nullptr;
|
|
g_grim->getCurrSet()->findClosestSector(_pos, &s, nullptr);
|
|
if (s) {
|
|
up = s->getNormal();
|
|
}
|
|
}
|
|
|
|
Math::Matrix3 m;
|
|
m.buildFromTargetDir(actorForward, lookVector, actorUp, up);
|
|
|
|
if (_puckOrient) {
|
|
m.getEuler(yaw, pitch, roll, Math::EO_ZXY);
|
|
} else {
|
|
*pitch = _movePitch;
|
|
m.getEuler(yaw, nullptr, nullptr, Math::EO_ZXY);
|
|
*roll = _moveRoll;
|
|
}
|
|
}
|
|
|
|
void Actor::turnTo(const Math::Vector3d &pos, bool snap) {
|
|
Math::Angle y, p, r;
|
|
calculateOrientation(pos, &p, &y, &r);
|
|
turnTo(p, y, r, snap);
|
|
}
|
|
|
|
bool Actor::singleTurnTo(const Math::Vector3d &pos) {
|
|
Math::Angle y, p, r;
|
|
calculateOrientation(pos, &p, &y, &r);
|
|
|
|
float turnAmt = g_grim->getPerSecond(_turnRate);
|
|
bool done = animTurn(turnAmt, y, &_yaw) == 0;
|
|
done = animTurn(turnAmt, p, &_pitch) == 0 && done;
|
|
done = animTurn(turnAmt, r, &_roll) == 0 && done;
|
|
_moveYaw = _yaw;
|
|
_movePitch = _pitch;
|
|
_moveRoll = _roll;
|
|
_singleTurning = !done;
|
|
|
|
return done;
|
|
}
|
|
|
|
void Actor::turnTo(const Math::Angle &pitchParam, const Math::Angle &yawParam, const Math::Angle &rollParam, bool snap) {
|
|
_movePitch = pitchParam;
|
|
_moveRoll = rollParam;
|
|
_moveYaw = yawParam;
|
|
_turnRateMultiplier = (snap ? 5.f : 1.f);
|
|
if (_yaw != yawParam || _pitch != pitchParam || _roll != rollParam) {
|
|
_turning = true;
|
|
} else
|
|
_turning = false;
|
|
}
|
|
|
|
void Actor::walkTo(const Math::Vector3d &p) {
|
|
if (p == _pos)
|
|
_walking = false;
|
|
else {
|
|
_walking = true;
|
|
_destPos = p;
|
|
_path.clear();
|
|
|
|
if (_followBoxes) {
|
|
bool pathFound = false;
|
|
|
|
Set *currSet = g_grim->getCurrSet();
|
|
currSet->findClosestSector(p, nullptr, &_destPos);
|
|
|
|
Common::List<PathNode *> openList;
|
|
Common::List<PathNode *> closedList;
|
|
|
|
PathNode *start = new PathNode;
|
|
start->parent = nullptr;
|
|
start->pos = _pos;
|
|
start->dist = 0.f;
|
|
start->cost = 0.f;
|
|
openList.push_back(start);
|
|
currSet->findClosestSector(_pos, &start->sect, nullptr);
|
|
|
|
Common::List<Sector *> sectors;
|
|
for (int i = 0; i < currSet->getSectorCount(); ++i) {
|
|
Sector *s = currSet->getSectorBase(i);
|
|
int type = s->getType();
|
|
if ((type == Sector::WalkType || type == Sector::HotType || type == Sector::FunnelType) && s->isVisible()) {
|
|
sectors.push_back(s);
|
|
}
|
|
}
|
|
|
|
do {
|
|
PathNode *node = nullptr;
|
|
float cost = -1.f;
|
|
for (Common::List<PathNode *>::iterator j = openList.begin(); j != openList.end(); ++j) {
|
|
PathNode *n = *j;
|
|
float c = n->dist + n->cost;
|
|
if (cost < 0.f || c < cost) {
|
|
cost = c;
|
|
node = n;
|
|
}
|
|
}
|
|
closedList.push_back(node);
|
|
openList.remove(node);
|
|
Sector *sector = node->sect;
|
|
|
|
if (sector && sector->isPointInSector(_destPos)) {
|
|
PathNode *n = closedList.back();
|
|
// Don't put the start position in the list, or else
|
|
// the first angle calculated in updateWalk() will be
|
|
// meaningless. The only node without parent is the start
|
|
// one.
|
|
while (n->parent) {
|
|
_path.push_back(n->pos);
|
|
n = n->parent;
|
|
}
|
|
|
|
pathFound = true;
|
|
break;
|
|
}
|
|
|
|
for (Common::List<Sector *>::iterator i = sectors.begin(); i != sectors.end(); ++i) {
|
|
Sector *s = *i;
|
|
bool inClosed = false;
|
|
for (Common::List<PathNode *>::iterator j = closedList.begin(); j != closedList.end(); ++j) {
|
|
if ((*j)->sect == s) {
|
|
inClosed = true;
|
|
break;
|
|
}
|
|
}
|
|
if (inClosed)
|
|
continue;
|
|
|
|
// Get "bridges" from the current sector to the other.
|
|
Common::List<Math::Line3d> bridges = sector->getBridgesTo(s);
|
|
if (bridges.empty())
|
|
continue; // The sectors are not adjacent.
|
|
|
|
Math::Vector3d closestPoint;
|
|
if (g_grim->getGameType() == GType_GRIM)
|
|
closestPoint = s->getClosestPoint(_destPos);
|
|
else
|
|
closestPoint = _destPos;
|
|
Math::Vector3d best;
|
|
float bestDist = 1e6f;
|
|
Math::Line3d l(node->pos, closestPoint);
|
|
|
|
// Pick a point on the boundary of the two sectors to walk towards.
|
|
while (!bridges.empty()) {
|
|
Math::Line3d bridge = bridges.back();
|
|
Math::Vector3d pos;
|
|
const bool useXZ = (g_grim->getGameType() == GType_MONKEY4);
|
|
|
|
// Prefer points on the straight line from this node towards
|
|
// the destination. Otherwise pick the middle point of a bridge
|
|
// that is closest to the destination.
|
|
if (!bridge.intersectLine2d(l, &pos, useXZ)) {
|
|
pos = bridge.middle();
|
|
} else {
|
|
best = pos;
|
|
break;
|
|
}
|
|
float dist = (pos - closestPoint).getMagnitude();
|
|
if (dist < bestDist) {
|
|
bestDist = dist;
|
|
best = pos;
|
|
}
|
|
bridges.pop_back();
|
|
}
|
|
best = handleCollisionTo(node->pos, best);
|
|
|
|
PathNode *n = nullptr;
|
|
for (Common::List<PathNode *>::iterator j = openList.begin(); j != openList.end(); ++j) {
|
|
if ((*j)->sect == s) {
|
|
n = *j;
|
|
break;
|
|
}
|
|
}
|
|
if (n) {
|
|
float newCost = node->cost + (best - node->pos).getMagnitude();
|
|
if (newCost < n->cost) {
|
|
n->cost = newCost;
|
|
n->parent = node;
|
|
n->pos = best;
|
|
n->dist = (n->pos - _destPos).getMagnitude();
|
|
}
|
|
} else {
|
|
n = new PathNode;
|
|
n->parent = node;
|
|
n->sect = s;
|
|
n->pos = best;
|
|
n->dist = (n->pos - _destPos).getMagnitude();
|
|
n->cost = node->cost + (n->pos - node->pos).getMagnitude();
|
|
openList.push_back(n);
|
|
}
|
|
}
|
|
} while (!openList.empty());
|
|
|
|
for (Common::List<PathNode *>::iterator j = closedList.begin(); j != closedList.end(); ++j) {
|
|
delete *j;
|
|
}
|
|
for (Common::List<PathNode *>::iterator j = openList.begin(); j != openList.end(); ++j) {
|
|
delete *j;
|
|
}
|
|
|
|
if (!pathFound) {
|
|
warning("Actor::walkTo(): No path found for %s", _name.c_str());
|
|
if (g_grim->getGameType() == GType_MONKEY4) {
|
|
_walking = false;
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
_path.push_front(_destPos);
|
|
}
|
|
}
|
|
|
|
bool Actor::isWalking() const {
|
|
return _walkedLast || _walkedCur || _walking;
|
|
}
|
|
|
|
bool Actor::isTurning() const {
|
|
if (g_grim->getGameType() == GType_MONKEY4)
|
|
if (_singleTurning)
|
|
return true;
|
|
if (_turning)
|
|
return true;
|
|
|
|
if (_lastTurnDir != 0 || _currTurnDir != 0)
|
|
return true;
|
|
|
|
return false;
|
|
}
|
|
|
|
void Actor::stopTurning() {
|
|
_turning = false;
|
|
if (_lastTurnDir != 0)
|
|
getTurnChore(_lastTurnDir)->stop(true);
|
|
|
|
_lastTurnDir = 0;
|
|
_currTurnDir = 0;
|
|
}
|
|
|
|
void Actor::moveTo(const Math::Vector3d &pos) {
|
|
// The walking actor doesn't always have the collision mode set, but it must however check
|
|
// the collisions. E.g. the set hl doesn't set Manny's mode, but it must check for
|
|
// collisions with Raoul.
|
|
// On the other hand, it must not *set* the sphere mode, it must just use it for this call:
|
|
// the set xb sets Manny's collision scale as 1 and mode as Off. If you then go to set tb
|
|
// and talk to Doug at the photo window, Manny's mode *must be* Off, otherwise Doug will
|
|
// collide, miss the target point and will walk away, leaving Manny waiting for him at
|
|
// the window forever.
|
|
CollisionMode mode = _collisionMode;
|
|
if (_collisionMode == CollisionOff) {
|
|
mode = CollisionSphere;
|
|
}
|
|
|
|
Math::Vector3d moveVec = pos - _pos;
|
|
for (Actor *a : g_grim->getActiveActors()) {
|
|
handleCollisionWith(a, mode, &moveVec);
|
|
}
|
|
_pos += moveVec;
|
|
}
|
|
|
|
void Actor::walkForward() {
|
|
float dist = g_grim->getPerSecond(_walkRate);
|
|
// Limit the amount of the movement per frame, otherwise with low fps
|
|
// scripts that use WalkActorForward and proximity may break.
|
|
if ((dist > 0 && dist > _walkRate / 5.f) || (dist < 0 && dist < _walkRate / 5.f))
|
|
dist = _walkRate / 5.f;
|
|
|
|
// Handle special case where actor is trying to walk but _walkRate is
|
|
// currently set to 0.0f by _walkChore to simulate roboter-like walk style:
|
|
// set _walkedCur to true to keep _walkChore playing in Actor::update()
|
|
if (_walkRate == 0.0f) {
|
|
_walkedCur = true;
|
|
}
|
|
|
|
_walking = false;
|
|
|
|
if (!_followBoxes) {
|
|
Math::Vector3d forwardVec(-_moveYaw.getSine() * _pitch.getCosine(),
|
|
_moveYaw.getCosine() * _pitch.getCosine(), _pitch.getSine());
|
|
|
|
// EMI: Y is up-down, actors use an X-Z plane for movement
|
|
if (g_grim->getGameType() == GType_MONKEY4) {
|
|
float temp = forwardVec.z();
|
|
forwardVec.x() = -forwardVec.x();
|
|
forwardVec.z() = forwardVec.y();
|
|
forwardVec.y() = temp;
|
|
}
|
|
|
|
_pos += forwardVec * dist;
|
|
_walkedCur = true;
|
|
return;
|
|
}
|
|
|
|
bool backwards = false;
|
|
if (dist < 0) {
|
|
dist = -dist;
|
|
backwards = true;
|
|
}
|
|
|
|
int tries = 0;
|
|
while (dist > 0.0f) {
|
|
Sector *currSector = nullptr, *prevSector = nullptr, *startSector = nullptr;
|
|
Sector::ExitInfo ei;
|
|
|
|
g_grim->getCurrSet()->findClosestSector(_pos, &currSector, &_pos);
|
|
if (!currSector) { // Shouldn't happen...
|
|
Math::Vector3d forwardVec(-_moveYaw.getSine() * _pitch.getCosine(),
|
|
_moveYaw.getCosine() * _pitch.getCosine(), _pitch.getSine());
|
|
|
|
// EMI: Y is up-down, actors use an X-Z plane for movement
|
|
if (g_grim->getGameType() == GType_MONKEY4) {
|
|
float temp = forwardVec.z();
|
|
forwardVec.x() = -forwardVec.x();
|
|
forwardVec.z() = forwardVec.y();
|
|
forwardVec.y() = temp;
|
|
}
|
|
|
|
if (backwards)
|
|
forwardVec = -forwardVec;
|
|
|
|
moveTo(_pos + forwardVec * dist);
|
|
_walkedCur = true;
|
|
return;
|
|
}
|
|
startSector = currSector;
|
|
|
|
float oldDist = dist;
|
|
while (currSector) {
|
|
prevSector = currSector;
|
|
Math::Vector3d forwardVec;
|
|
const Math::Vector3d &normal = currSector->getNormal();
|
|
if (g_grim->getGameType() == GType_GRIM) {
|
|
Math::Angle ax = Math::Vector2d(normal.x(), normal.z()).getAngle();
|
|
Math::Angle ay = Math::Vector2d(normal.y(), normal.z()).getAngle();
|
|
|
|
float z1 = -_moveYaw.getCosine() * (ay - _pitch).getCosine();
|
|
float z2 = _moveYaw.getSine() * (ax - _pitch).getCosine();
|
|
forwardVec = Math::Vector3d(-_moveYaw.getSine() * ax.getSine() * _pitch.getCosine(),
|
|
_moveYaw.getCosine() * ay.getSine() * _pitch.getCosine(), z1 + z2);
|
|
} else {
|
|
Math::Angle ax = Math::Vector2d(normal.x(), normal.y()).getAngle();
|
|
Math::Angle az = Math::Vector2d(normal.z(), normal.y()).getAngle();
|
|
|
|
float y1 = _moveYaw.getCosine() * (az - _pitch).getCosine();
|
|
float y2 = _moveYaw.getSine() * (ax - _pitch).getCosine();
|
|
forwardVec = Math::Vector3d(-_moveYaw.getSine() * ax.getSine() * _pitch.getCosine(), y1 + y2,
|
|
-_moveYaw.getCosine() * az.getSine() * _pitch.getCosine());
|
|
}
|
|
|
|
if (backwards)
|
|
forwardVec = -forwardVec;
|
|
|
|
Math::Vector3d puckVec = currSector->getProjectionToPuckVector(forwardVec);
|
|
puckVec /= puckVec.getMagnitude();
|
|
currSector->getExitInfo(_pos, puckVec, &ei);
|
|
float exitDist = (ei.exitPoint - _pos).getMagnitude();
|
|
if (dist < exitDist) {
|
|
moveTo(_pos + puckVec * dist);
|
|
_walkedCur = true;
|
|
return;
|
|
}
|
|
_pos = ei.exitPoint;
|
|
dist -= exitDist;
|
|
if (exitDist > 0.0001)
|
|
_walkedCur = true;
|
|
|
|
// Check for an adjacent sector which can continue
|
|
// the path
|
|
currSector = g_grim->getCurrSet()->findPointSector(ei.exitPoint + (float)0.0001 * puckVec, Sector::WalkType);
|
|
|
|
// EMI: some sectors are significantly higher/lower than others.
|
|
if (currSector && g_grim->getGameType() == GType_MONKEY4) {
|
|
float planeDist = currSector->distanceToPoint(_pos);
|
|
if (fabs(planeDist) < 1.f)
|
|
_pos -= planeDist * currSector->getNormal();
|
|
}
|
|
|
|
if (currSector == prevSector || currSector == startSector)
|
|
break;
|
|
}
|
|
|
|
int turnDir = 1;
|
|
if (ei.angleWithEdge > 90) {
|
|
ei.angleWithEdge = 180 - ei.angleWithEdge;
|
|
ei.edgeDir = -ei.edgeDir;
|
|
turnDir = -1;
|
|
}
|
|
if (ei.angleWithEdge > _reflectionAngle)
|
|
return;
|
|
|
|
ei.angleWithEdge += (float)1.0f;
|
|
|
|
if (g_grim->getGameType() == GType_MONKEY4) {
|
|
ei.angleWithEdge = -ei.angleWithEdge;
|
|
}
|
|
turnTo(0, _moveYaw + ei.angleWithEdge * turnDir, 0, true);
|
|
|
|
if (oldDist <= dist + 0.001f) {
|
|
// If we didn't move at all, keep trying a couple more times
|
|
// in case we can move in the new direction.
|
|
tries++;
|
|
if (tries > 3)
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
Math::Vector3d Actor::getSimplePuckVector() const {
|
|
if (g_grim->getGameType() == GType_MONKEY4) {
|
|
Math::Angle yaw = 0;
|
|
const Actor *a = this;
|
|
while (a) {
|
|
yaw += a->_yaw;
|
|
if (!a->isAttached())
|
|
break;
|
|
a = Actor::getPool().getObject(a->_attachedActor);
|
|
}
|
|
return Math::Vector3d(yaw.getSine(), 0, yaw.getCosine());
|
|
} else {
|
|
return Math::Vector3d(-_yaw.getSine(), _yaw.getCosine(), 0);
|
|
}
|
|
}
|
|
|
|
Math::Vector3d Actor::getPuckVector() const {
|
|
Math::Vector3d forwardVec = getSimplePuckVector();
|
|
|
|
Set *currSet = g_grim->getCurrSet();
|
|
if (!currSet)
|
|
return forwardVec;
|
|
|
|
Sector *sector = currSet->findPointSector(_pos, Sector::WalkType);
|
|
if (!sector)
|
|
return forwardVec;
|
|
else
|
|
return sector->getProjectionToPuckVector(forwardVec);
|
|
}
|
|
|
|
void Actor::setPuckOrient(bool orient) {
|
|
_puckOrient = orient;
|
|
warning("Actor::setPuckOrient() not implemented");
|
|
}
|
|
|
|
void Actor::setRestChore(int chore, Costume *cost) {
|
|
if (_restChore.equals(cost, chore))
|
|
return;
|
|
|
|
_restChore.stop(g_grim->getGameType() == GType_GRIM);
|
|
|
|
if (!cost) {
|
|
cost = _restChore._costume;
|
|
}
|
|
if (!cost) {
|
|
cost = getCurrentCostume();
|
|
}
|
|
|
|
_restChore = ActionChore(cost, chore);
|
|
|
|
_restChore.playLooping(g_grim->getGameType() == GType_GRIM);
|
|
}
|
|
|
|
int Actor::getRestChore() const {
|
|
return _restChore._chore;
|
|
}
|
|
|
|
void Actor::setWalkChore(int chore, Costume *cost) {
|
|
if (_walkChore.equals(cost, chore))
|
|
return;
|
|
|
|
if (_walkedLast && _walkChore.isPlaying()) {
|
|
_walkChore.stop(true);
|
|
|
|
if (g_grim->getGameType() == GType_GRIM) {
|
|
_restChore.playLooping(true);
|
|
}
|
|
}
|
|
|
|
if (!cost) {
|
|
cost = _walkChore._costume;
|
|
}
|
|
if (!cost) {
|
|
cost = getCurrentCostume();
|
|
}
|
|
|
|
_walkChore = ActionChore(cost, chore);
|
|
}
|
|
|
|
void Actor::setTurnChores(int left_chore, int right_chore, Costume *cost) {
|
|
if (_leftTurnChore.equals(cost, left_chore) && _rightTurnChore.equals(cost, right_chore))
|
|
return;
|
|
|
|
if (!cost) {
|
|
cost = _leftTurnChore._costume;
|
|
}
|
|
if (!cost) {
|
|
cost = getCurrentCostume();
|
|
}
|
|
|
|
_leftTurnChore.stop(true);
|
|
_rightTurnChore.stop(true);
|
|
_lastTurnDir = 0;
|
|
|
|
_leftTurnChore = ActionChore(cost, left_chore);
|
|
_rightTurnChore = ActionChore(cost, right_chore);
|
|
|
|
if ((left_chore >= 0 && right_chore < 0) || (left_chore < 0 && right_chore >= 0))
|
|
error("Unexpectedly got only one turn chore");
|
|
}
|
|
|
|
void Actor::setTalkChore(int index, int chore, Costume *cost) {
|
|
if (index < 1 || index > 10)
|
|
error("Got talk chore index out of range (%d)", index);
|
|
|
|
index--;
|
|
|
|
if (!cost) {
|
|
cost = _talkChore[index]._costume;
|
|
}
|
|
if (!cost) {
|
|
cost = getCurrentCostume();
|
|
}
|
|
|
|
if (_talkChore[index].equals(cost, chore))
|
|
return;
|
|
|
|
_talkChore[index].stop();
|
|
|
|
_talkChore[index] = ActionChore(cost, chore);
|
|
}
|
|
|
|
int Actor::getTalkChore(int index) const {
|
|
return _talkChore[index]._chore;
|
|
}
|
|
|
|
Costume *Actor::getTalkCostume(int index) const {
|
|
return _talkChore[index]._costume;
|
|
}
|
|
|
|
void Actor::setMumbleChore(int chore, Costume *cost) {
|
|
if (_mumbleChore.isPlaying()) {
|
|
_mumbleChore.stop();
|
|
}
|
|
|
|
if (!cost) {
|
|
cost = _mumbleChore._costume;
|
|
}
|
|
if (!cost) {
|
|
cost = getCurrentCostume();
|
|
}
|
|
|
|
_mumbleChore = ActionChore(cost, chore);
|
|
}
|
|
|
|
void Actor::stopAllChores(bool ignoreLoopingChores) {
|
|
for (Common::List<Costume *>::iterator i = _costumeStack.begin(); i != _costumeStack.end(); ++i) {
|
|
Costume *costume = *i;
|
|
costume->stopChores(ignoreLoopingChores);
|
|
if (costume->isChoring(false) == -1) {
|
|
freeCostume(costume);
|
|
i = _costumeStack.erase(i);
|
|
--i;
|
|
}
|
|
}
|
|
}
|
|
|
|
void Actor::turn(int dir) {
|
|
_walking = false;
|
|
float delta = g_grim->getPerSecond(_turnRate) * dir;
|
|
if (g_grim->getGameType() == GType_MONKEY4) {
|
|
delta = -delta;
|
|
}
|
|
_moveYaw = _moveYaw + delta;
|
|
_turning = true;
|
|
_turnRateMultiplier = 5.f;
|
|
_currTurnDir = dir;
|
|
}
|
|
|
|
Math::Angle Actor::getYawTo(const Actor *a) const {
|
|
Math::Vector3d forwardVec = getSimplePuckVector();
|
|
Math::Vector3d delta = a->getWorldPos() - getWorldPos();
|
|
|
|
if (g_grim->getGameType() == GType_MONKEY4) {
|
|
delta.y() = 0;
|
|
} else {
|
|
delta.z() = 0;
|
|
}
|
|
if (delta.getMagnitude() < Math::epsilon)
|
|
return Math::Angle(0);
|
|
return Math::Vector3d::angle(forwardVec, delta);
|
|
}
|
|
|
|
Math::Angle Actor::getYawTo(const Math::Vector3d &p) const {
|
|
Math::Vector3d dpos = p - _pos;
|
|
|
|
if (g_grim->getGameType() == GType_MONKEY4) {
|
|
dpos.y() = dpos.z();
|
|
}
|
|
if (dpos.x() == 0 && dpos.y() == 0)
|
|
return 0;
|
|
else
|
|
return Math::Angle::arcTangent2(-dpos.x(), dpos.y());
|
|
}
|
|
|
|
void Actor::sayLine(const char *msgId, bool background, float x, float y) {
|
|
assert(msgId);
|
|
|
|
if (msgId[0] == 0) {
|
|
warning("Actor::sayLine: Empty message");
|
|
return;
|
|
}
|
|
|
|
char id[50];
|
|
Common::String msg = LuaBase::instance()->parseMsgText(msgId, id);
|
|
|
|
if (id[0] == 0) {
|
|
error("Actor::sayLine: No message ID for text");
|
|
return;
|
|
}
|
|
|
|
// During Fullscreen movies SayLine is usually called for text display only.
|
|
// The movie with Charlie screaming after Manny put the sheet on him instead
|
|
// uses sayLine for the voice too.
|
|
// However, normal SMUSH movies may call SayLine, for example:
|
|
// When Domino yells at Manny (a SMUSH movie) he does it with
|
|
// a SayLine request rather than as part of the movie!
|
|
|
|
Common::String soundName = id;
|
|
|
|
if (g_grim->getGameType() == GType_GRIM) {
|
|
if (g_grim->isRemastered()) {
|
|
soundName = g_grim->getLanguagePrefix() + "_" + soundName;
|
|
}
|
|
soundName += ".wav";
|
|
} else if (g_grim->getGameType() == GType_MONKEY4 && g_grim->getGamePlatform() == Common::kPlatformPS2) {
|
|
soundName += ".scx";
|
|
} else {
|
|
soundName += ".wVC";
|
|
}
|
|
if (_talkSoundName == soundName)
|
|
return;
|
|
|
|
if (_talking || msg.empty())
|
|
shutUp();
|
|
|
|
_talkSoundName = soundName;
|
|
|
|
Set *currSet = g_grim->getCurrSet();
|
|
|
|
if (g_grim->getSpeechMode() != GrimEngine::TextOnly) {
|
|
// if there is no costume probably the character is drawn by a smush movie, so
|
|
// we don't want to go out of sync with it.
|
|
if (g_grim->getGameType() == GType_GRIM && getCurrentCostume()) {
|
|
_talkDelay = 500;
|
|
}
|
|
g_sound->startVoice(_talkSoundName.c_str());
|
|
}
|
|
|
|
// If the actor is clearly not visible then don't try to play the lip sync
|
|
if (_visible && (!g_movie->isPlaying() || g_grim->getMode() == GrimEngine::NormalMode)) {
|
|
Common::String soundLip = id;
|
|
soundLip += ".lip";
|
|
|
|
if (!_talkChore[0].isPlaying()) {
|
|
// _talkChore[0] is *_stop_talk
|
|
_talkChore[0].setLastFrame();
|
|
}
|
|
// Sometimes actors speak offscreen before they, including their
|
|
// talk chores are initialized.
|
|
// For example, when reading the work order (a LIP file exists for no reason).
|
|
// Also, some lip sync files have no entries
|
|
// In these cases, revert to using the mumble chore.
|
|
if (g_grim->getSpeechMode() != GrimEngine::TextOnly)
|
|
_lipSync = g_resourceloader->getLipSync(soundLip);
|
|
// If there's no lip sync file then load the mumble chore if it exists
|
|
// (the mumble chore doesn't exist with the cat races announcer)
|
|
if (!_lipSync)
|
|
_mumbleChore.playLooping();
|
|
|
|
_talkAnim = -1;
|
|
}
|
|
|
|
_talking = true;
|
|
g_grim->addTalkingActor(this);
|
|
|
|
_backgroundTalk = background;
|
|
if (background)
|
|
_isTalkingBackground = true;
|
|
|
|
if (_sayLineText && g_grim->getMode() != GrimEngine::SmushMode) {
|
|
delete TextObject::getPool().getObject(_sayLineText);
|
|
_sayLineText = 0;
|
|
}
|
|
|
|
if (!msg.empty()) {
|
|
GrimEngine::SpeechMode m = g_grim->getSpeechMode();
|
|
if (!g_grim->_sayLineDefaults.getFont() || m == GrimEngine::VoiceOnly)
|
|
return;
|
|
|
|
if (background) {
|
|
// if we're talking background draw the text object only if there are no no-background
|
|
// talking actors. This prevents glottis and nick subtitles overlapping in the high roller lounge,
|
|
// where glottis is background and nick isn't. (https://github.com/residualvm/residualvm/issues/685)
|
|
for (Actor *a : g_grim->getTalkingActors()) {
|
|
if (!a->_backgroundTalk && a->_sayLineText) {
|
|
return;
|
|
}
|
|
}
|
|
} else {
|
|
// if we're not background then delete the TextObject of any talking background actor.
|
|
for (Actor *a : g_grim->getTalkingActors()) {
|
|
if (a->_backgroundTalk && a->_sayLineText) {
|
|
delete TextObject::getPool().getObject(a->_sayLineText);
|
|
a->_sayLineText = 0;
|
|
}
|
|
}
|
|
}
|
|
|
|
TextObject *textObject = new TextObject();
|
|
textObject->setDefaults(&g_grim->_sayLineDefaults);
|
|
textObject->setFGColor(_talkColor);
|
|
textObject->setIsSpeech();
|
|
if (m == GrimEngine::TextOnly || g_grim->getMode() == GrimEngine::SmushMode) {
|
|
textObject->setDuration(500 + msg.size() * 15 * (11 - g_grim->getTextSpeed()));
|
|
}
|
|
if (g_grim->getGameType() == GType_MONKEY4 && (x != -1 || y != -1)) {
|
|
textObject->setX(320 * (x + 1));
|
|
textObject->setY(240 * (y + 1));
|
|
} else if (g_grim->getMode() == GrimEngine::SmushMode) {
|
|
textObject->setX(640 / 2);
|
|
textObject->setY(456);
|
|
g_grim->setMovieSubtitle(textObject);
|
|
} else {
|
|
if (_visible && isInSet(currSet->getName())) {
|
|
_mustPlaceText = true;
|
|
} else {
|
|
_mustPlaceText = false;
|
|
textObject->setX(640 / 2);
|
|
textObject->setY(463);
|
|
}
|
|
}
|
|
textObject->setText(msgId, _mustPlaceText);
|
|
if (g_grim->getMode() != GrimEngine::SmushMode)
|
|
_sayLineText = textObject->getId();
|
|
}
|
|
}
|
|
|
|
void Actor::lineCleanup() {
|
|
if (_sayLineText) {
|
|
delete TextObject::getPool().getObject(_sayLineText);
|
|
_sayLineText = 0;
|
|
}
|
|
}
|
|
|
|
bool Actor::isTalking() {
|
|
// This should return if the actor is actually talking, disregarding of the background status,
|
|
// otherwise when Naranja is asleep Toto's lines will be cut sometimes. Naranja and Toto both
|
|
// talk in background, and when Naranja lines stop toto:wait_for_message() should not return
|
|
// until he actually stops talking.
|
|
return _talking;
|
|
}
|
|
|
|
bool Actor::isTalkingForeground() const {
|
|
if (!_talking) {
|
|
return false;
|
|
}
|
|
|
|
if (_backgroundTalk)
|
|
return _isTalkingBackground;
|
|
|
|
return true;
|
|
}
|
|
|
|
void Actor::shutUp() {
|
|
// While the call to stop the sound is usually made by the game,
|
|
// we also need to handle when the user terminates the dialog.
|
|
// Some warning messages will occur when the user terminates the
|
|
// actor dialog but the game will continue alright.
|
|
if (_talkSoundName != "") {
|
|
g_sound->stopSound(_talkSoundName.c_str());
|
|
_talkSoundName = "";
|
|
}
|
|
|
|
if (_lipSync) {
|
|
if (_talkAnim != -1)
|
|
_talkChore[_talkAnim].stop(g_grim->getGameType() == GType_MONKEY4, ActionChore::talkFadeTime);
|
|
_lipSync = nullptr;
|
|
}
|
|
// having a lipsync is no guarantee the mumble chore is no running. the talk chores may be -1 (domino in do)
|
|
stopMumbleChore();
|
|
stopTalking();
|
|
|
|
if (_sayLineText) {
|
|
delete TextObject::getPool().getObject(_sayLineText);
|
|
_sayLineText = 0;
|
|
}
|
|
|
|
// The actors talking in background have a special behaviour: if there are two or more of them
|
|
// talking at the same time, after one of them finishes talking calling isTalking() an *all*
|
|
// of them must return false. This is necessary for the angelitos in set fo: when they start crying
|
|
// they talk in background, and the lua script that hangs on IsMessageGoing() must return before they
|
|
// stop, since they can go on forever.
|
|
if (_backgroundTalk)
|
|
_isTalkingBackground = false;
|
|
|
|
_talking = false;
|
|
}
|
|
|
|
void Actor::pushCostume(const char *n) {
|
|
Costume *c = findCostume(n);
|
|
if (c) {
|
|
Debug::debug(Debug::Actors, "Trying to push a costume already in the stack. %s, %s", getName().c_str(), n);
|
|
return;
|
|
}
|
|
|
|
Costume *newCost = g_resourceloader->loadCostume(n, this, getCurrentCostume());
|
|
|
|
_costumeStack.push_back(newCost);
|
|
}
|
|
|
|
void Actor::setColormap(const char *map) {
|
|
if (!_costumeStack.empty()) {
|
|
Costume *cost = _costumeStack.back();
|
|
cost->setColormap(map);
|
|
} else {
|
|
warning("Actor::setColormap: No costumes");
|
|
}
|
|
}
|
|
|
|
void Actor::setCostume(const char *n) {
|
|
if (!_costumeStack.empty())
|
|
popCostume();
|
|
|
|
pushCostume(n);
|
|
}
|
|
|
|
void Actor::popCostume() {
|
|
if (!_costumeStack.empty()) {
|
|
freeCostume(_costumeStack.back());
|
|
_costumeStack.pop_back();
|
|
|
|
if (_costumeStack.empty()) {
|
|
Debug::debug(Debug::Actors, "Popped (freed) the last costume for an actor.\n");
|
|
}
|
|
} else {
|
|
Debug::warning(Debug::Actors, "Attempted to pop (free) a costume when the stack is empty!");
|
|
}
|
|
}
|
|
|
|
void Actor::clearCostumes() {
|
|
// Make sure to destroy costume copies in reverse order
|
|
while (!_costumeStack.empty())
|
|
popCostume();
|
|
}
|
|
|
|
Costume *Actor::getCurrentCostume() const {
|
|
if (g_grim->getGameType() == GType_MONKEY4) {
|
|
// Return the first costume that has a model component.
|
|
for (Common::List<Costume *>::const_iterator i = _costumeStack.begin(); i != _costumeStack.end(); ++i) {
|
|
EMICostume *costume = static_cast<EMICostume *>(*i);
|
|
if (costume->getEMIModel()) {
|
|
return costume;
|
|
}
|
|
}
|
|
return nullptr;
|
|
} else {
|
|
if (_costumeStack.empty())
|
|
return nullptr;
|
|
else
|
|
return _costumeStack.back();
|
|
}
|
|
}
|
|
|
|
void Actor::setHead(int joint1, int joint2, int joint3, float maxRoll, float maxPitch, float maxYaw) {
|
|
Costume *curCostume = getCurrentCostume();
|
|
if (curCostume) {
|
|
curCostume->setHead(joint1, joint2, joint3, maxRoll, maxPitch, maxYaw);
|
|
}
|
|
}
|
|
|
|
void Actor::setHead(const char *joint, const Math::Vector3d &offset) {
|
|
Costume *curCostume = getCurrentCostume();
|
|
if (curCostume) {
|
|
EMICostume *costume = static_cast<EMICostume *>(curCostume);
|
|
costume->setHead(joint, offset);
|
|
}
|
|
}
|
|
|
|
void Actor::setHeadLimits(float yawRange, float maxPitch, float minPitch) {
|
|
Costume *curCostume = getCurrentCostume();
|
|
if (curCostume) {
|
|
EMICostume *costume = static_cast<EMICostume *>(curCostume);
|
|
costume->setHeadLimits(yawRange, maxPitch, minPitch);
|
|
}
|
|
}
|
|
|
|
void Actor::setLookAtRate(float rate) {
|
|
getCurrentCostume()->setLookAtRate(rate);
|
|
}
|
|
|
|
float Actor::getLookAtRate() const {
|
|
return getCurrentCostume()->getLookAtRate();
|
|
}
|
|
|
|
EMIModel *Actor::findModelWithMesh(const Common::String &mesh) {
|
|
for (Common::List<Costume *>::iterator i = _costumeStack.begin(); i != _costumeStack.end(); ++i) {
|
|
EMICostume *costume = static_cast<EMICostume *>(*i);
|
|
if (!costume) {
|
|
continue;
|
|
}
|
|
for (int j = 0; j < costume->getNumChores(); j++) {
|
|
EMIModel *model = costume->getEMIModel(j);
|
|
if (!model) {
|
|
continue;
|
|
}
|
|
if (mesh == model->_meshName) {
|
|
return model;
|
|
}
|
|
}
|
|
}
|
|
return nullptr;
|
|
}
|
|
|
|
void Actor::setGlobalAlpha(float alpha, const Common::String &mesh) {
|
|
if (mesh.empty()) {
|
|
_globalAlpha = alpha;
|
|
} else {
|
|
EMIModel *model = findModelWithMesh(mesh);
|
|
if (model != nullptr) {
|
|
model->_meshAlpha = alpha;
|
|
}
|
|
}
|
|
}
|
|
|
|
void Actor::setAlphaMode(AlphaMode mode, const Common::String &mesh) {
|
|
if (mesh.empty()) {
|
|
_alphaMode = mode;
|
|
} else {
|
|
EMIModel *model = findModelWithMesh(mesh);
|
|
if (model != nullptr) {
|
|
model->_meshAlphaMode = mode;
|
|
}
|
|
}
|
|
}
|
|
|
|
Costume *Actor::findCostume(const Common::String &n) {
|
|
for (Common::List<Costume *>::iterator i = _costumeStack.begin(); i != _costumeStack.end(); ++i) {
|
|
if ((*i)->getFilename().compareToIgnoreCase(n) == 0)
|
|
return *i;
|
|
}
|
|
|
|
return nullptr;
|
|
}
|
|
|
|
void Actor::setFollowBoxes(bool follow) {
|
|
_followBoxes = follow;
|
|
if (follow)
|
|
_hasFollowedBoxes = true;
|
|
}
|
|
|
|
void Actor::updateWalk() {
|
|
if (_path.empty()) {
|
|
return;
|
|
}
|
|
|
|
Math::Vector3d destPos = _path.back();
|
|
Math::Vector3d dir = destPos - _pos;
|
|
float dist = dir.getMagnitude();
|
|
|
|
_walkedCur = true;
|
|
float walkAmt = g_grim->getPerSecond(_walkRate);
|
|
if (walkAmt >= dist) {
|
|
walkAmt = dist;
|
|
_path.pop_back();
|
|
if (_path.empty()) {
|
|
_walking = false;
|
|
_pos = destPos;
|
|
// It seems that we need to allow an already active turning motion to
|
|
// continue or else turning actors away from barriers won't work right
|
|
_turning = false;
|
|
return;
|
|
}
|
|
}
|
|
|
|
turnTo(destPos, true);
|
|
|
|
dir = destPos - _pos;
|
|
dir.normalize();
|
|
_pos += dir * walkAmt;
|
|
}
|
|
|
|
void Actor::update(uint frameTime) {
|
|
Set *set = g_grim->getCurrSet();
|
|
// Snap actor to walkboxes if following them. This might be
|
|
// necessary for example after activating/deactivating
|
|
// walkboxes, etc.
|
|
if (_followBoxes && !_walking) {
|
|
set->findClosestSector(_pos, nullptr, &_pos);
|
|
}
|
|
|
|
if (g_grim->getGameType() == GType_MONKEY4) {
|
|
if (_followBoxes) {
|
|
// Check for sort order information in the current sector
|
|
int oldSortOrder = getEffectiveSortOrder();
|
|
_sectorSortOrder = set->findSectorSortOrder(_pos, Sector::WalkType);
|
|
|
|
if (oldSortOrder != getEffectiveSortOrder())
|
|
g_emi->invalidateSortOrder();
|
|
} else if (_sectorSortOrder >= 0) {
|
|
_sectorSortOrder = -1;
|
|
g_emi->invalidateSortOrder();
|
|
}
|
|
}
|
|
|
|
if (_turning) {
|
|
float turnAmt = g_grim->getPerSecond(_turnRate) * _turnRateMultiplier;
|
|
_currTurnDir = animTurn(turnAmt, _moveYaw, &_yaw);
|
|
if (g_grim->getGameType() == GType_MONKEY4) {
|
|
_currTurnDir = -_currTurnDir;
|
|
}
|
|
int p = animTurn(turnAmt, _movePitch, &_pitch);
|
|
int r = animTurn(turnAmt, _moveRoll, &_roll);
|
|
if (_currTurnDir == 0 && p == 0 && r == 0) {
|
|
_turning = false;
|
|
_turnRateMultiplier = 1.f;
|
|
}
|
|
}
|
|
|
|
if (_walking) {
|
|
updateWalk();
|
|
}
|
|
|
|
if (_walkChore.isValid()) {
|
|
if (_walkedCur) {
|
|
if (!_walkChore.isPlaying()) {
|
|
_walkChore.playLooping(true);
|
|
}
|
|
if (g_grim->getGameType() == GType_GRIM && _restChore.isPlaying()) {
|
|
_restChore.stop(true);
|
|
}
|
|
} else {
|
|
if (_walkedLast && _walkChore.isPlaying()) {
|
|
_walkChore.stop(true);
|
|
if (!_restChore.isPlaying()) {
|
|
_restChore.playLooping(true);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if (_leftTurnChore.isValid()) {
|
|
if (_walkedCur || _walkedLast)
|
|
_currTurnDir = 0;
|
|
|
|
if (g_grim->getGameType() == GType_GRIM && _restChore.isValid()) {
|
|
if (_currTurnDir != 0) {
|
|
if (getTurnChore(_currTurnDir)->isPlaying() && _restChore.isPlaying()) {
|
|
_restChore.stop(true, 500);
|
|
}
|
|
} else if (_lastTurnDir != 0) {
|
|
if (!_walkedCur && getTurnChore(_lastTurnDir)->isPlaying()) {
|
|
_restChore.playLooping(true);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (_lastTurnDir != 0 && _lastTurnDir != _currTurnDir) {
|
|
getTurnChore(_lastTurnDir)->stop(true);
|
|
}
|
|
if (_currTurnDir != 0 && _currTurnDir != _lastTurnDir) {
|
|
getTurnChore(_currTurnDir)->playLooping(true, 500);
|
|
if (_currTurnDir == 1) {
|
|
// When turning to the left, ensure that the components of the right turn chore
|
|
// are fading out (or stopped).
|
|
// This is necessary because the left turn chore typically contains both the
|
|
// left turn and right turn keyframe components. The above call to playLooping
|
|
// will thus start fading in both of the components, overriding the right turn's
|
|
// fade out that was started before.
|
|
// The left turn chore's keys will eventually stop the right turn keyframe from
|
|
// playing, but the stopping will be instantaneous. To get a smooth transition,
|
|
// we want to keep fading out the right turn. The chore's "stop" key will be
|
|
// ignored when the keyframe is fading out (see KeyframeComponent::stop()).
|
|
_rightTurnChore.stop(true);
|
|
}
|
|
}
|
|
} else {
|
|
_currTurnDir = 0;
|
|
}
|
|
|
|
// The rest chore might have been stopped because of a
|
|
// StopActorChore(nil). Restart it if so.
|
|
if (!_walkedCur && _currTurnDir == 0 && !_restChore.isPlaying()) {
|
|
_restChore.playLooping(g_grim->getGameType() == GType_GRIM);
|
|
}
|
|
|
|
_walkedLast = _walkedCur;
|
|
_walkedCur = false;
|
|
_lastTurnDir = _currTurnDir;
|
|
_currTurnDir = 0;
|
|
|
|
// Update lip syncing
|
|
if (_lipSync) {
|
|
int posSound;
|
|
|
|
// While getPosIn16msTicks will return "-1" to indicate that the
|
|
// sound is no longer playing, it is more appropriate to check first
|
|
if (g_grim->getSpeechMode() != GrimEngine::TextOnly && g_sound->getSoundStatus(_talkSoundName.c_str()))
|
|
posSound = g_sound->getPosIn16msTicks(_talkSoundName.c_str());
|
|
else
|
|
posSound = -1;
|
|
if (posSound != -1) {
|
|
int anim = _lipSync->getAnim(posSound);
|
|
if (_talkAnim != anim) {
|
|
if (anim != -1) {
|
|
if (_talkChore[anim].isValid()) {
|
|
stopMumbleChore();
|
|
if (_talkAnim != -1) {
|
|
_talkChore[_talkAnim].stop(g_grim->getGameType() == GType_MONKEY4, ActionChore::talkFadeTime);
|
|
}
|
|
if (g_grim->getGameType() == GType_GRIM) {
|
|
// Run the stop_talk chore so that it resets the components
|
|
// to the right visibility.
|
|
stopTalking();
|
|
} else {
|
|
// Make sure the talk rest chore isn't playing.
|
|
_talkChore[0].stop();
|
|
}
|
|
_talkAnim = anim;
|
|
_talkChore[_talkAnim].play(g_grim->getGameType() == GType_MONKEY4, ActionChore::talkFadeTime);
|
|
} else if (_mumbleChore.isValid() && !_mumbleChore.isPlaying()) {
|
|
_mumbleChore.playLooping();
|
|
_talkAnim = -1;
|
|
}
|
|
} else {
|
|
stopMumbleChore();
|
|
if (_talkAnim != -1)
|
|
_talkChore[_talkAnim].stop(true, ActionChore::talkFadeTime);
|
|
|
|
_talkAnim = 0;
|
|
stopTalking();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
frameTime = (uint)(frameTime * _timeScale);
|
|
for (Common::List<Costume *>::iterator i = _costumeStack.begin(); i != _costumeStack.end(); ++i) {
|
|
Costume *c = *i;
|
|
c->setPosRotate(_pos, _pitch, _yaw, _roll);
|
|
int marker = c->update(frameTime);
|
|
if (marker > 0) {
|
|
costumeMarkerCallback(marker);
|
|
}
|
|
if (g_grim->getGameType() == GType_MONKEY4 && c->isChoring(false) == -1) {
|
|
freeCostume(c);
|
|
i = _costumeStack.erase(i);
|
|
--i;
|
|
}
|
|
}
|
|
|
|
Costume *c = getCurrentCostume();
|
|
if (c) {
|
|
c->animate();
|
|
}
|
|
|
|
if (_lookingMode && _lookAtActor != 0) {
|
|
Actor *actor = Actor::getPool().getObject(_lookAtActor);
|
|
if (actor)
|
|
_lookAtVector = actor->getHeadPos();
|
|
}
|
|
|
|
for (Common::List<Costume *>::iterator i = _costumeStack.begin(); i != _costumeStack.end(); ++i) {
|
|
(*i)->moveHead(_lookingMode, _lookAtVector);
|
|
}
|
|
}
|
|
|
|
// Not all the talking actors are in the current set, and so on not all the talking actors
|
|
// update() is called. For example, Don when he comes out of his office after reaping Meche.
|
|
bool Actor::updateTalk(uint frameTime) {
|
|
if (_talking) {
|
|
// If there's no sound file then we're obviously not talking
|
|
GrimEngine::SpeechMode m = g_grim->getSpeechMode();
|
|
TextObject *textObject = nullptr;
|
|
if (_sayLineText)
|
|
textObject = TextObject::getPool().getObject(_sayLineText);
|
|
if (m == GrimEngine::TextOnly && !textObject) {
|
|
shutUp();
|
|
return false;
|
|
} else if (m != GrimEngine::TextOnly && (_talkSoundName.empty() || !g_sound->getSoundStatus(_talkSoundName.c_str()))) {
|
|
_talkDelay -= frameTime;
|
|
if (_talkDelay <= 0) {
|
|
_talkDelay = 0;
|
|
shutUp();
|
|
return false;
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
void Actor::draw() {
|
|
// FIXME: if isAttached(), factor in the joint rotation as well.
|
|
const Math::Vector3d &absPos = getWorldPos();
|
|
if (!_costumeStack.empty()) {
|
|
g_grim->getCurrSet()->setupLights(absPos, _inOverworld);
|
|
if (g_grim->getGameType() == GType_GRIM) {
|
|
Costume *costume = _costumeStack.back();
|
|
drawCostume(costume);
|
|
} else {
|
|
for (Common::List<Costume *>::iterator it = _costumeStack.begin(); it != _costumeStack.end(); ++it) {
|
|
Costume *costume = *it;
|
|
drawCostume(costume);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (_mustPlaceText) {
|
|
Common::Point p1, p2;
|
|
if (g_grim->getGameType() == GType_GRIM) {
|
|
if (!_costumeStack.empty()) {
|
|
int x1 = 1000, y1 = 1000, x2 = -1000, y2 = -1000;
|
|
g_driver->startActorDraw(this);
|
|
_costumeStack.back()->getBoundingBox(&x1, &y1, &x2, &y2);
|
|
g_driver->finishActorDraw();
|
|
p1.x = x1;
|
|
p1.y = y1;
|
|
p2.x = x2;
|
|
p2.y = y2;
|
|
}
|
|
} else {
|
|
g_driver->getActorScreenBBox(this, p1, p2);
|
|
}
|
|
|
|
TextObject *textObject = TextObject::getPool().getObject(_sayLineText);
|
|
if (textObject) {
|
|
if (p1.x == 1000 || p2.x == -1000 || p2.x == -1000) {
|
|
textObject->setX(640 / 2);
|
|
textObject->setY(463);
|
|
} else {
|
|
textObject->setX((p1.x + p2.x) / 2);
|
|
textObject->setY(p1.y);
|
|
}
|
|
// Deletes the original text and rebuilds it with the newly placed text
|
|
textObject->reset();
|
|
}
|
|
_mustPlaceText = false;
|
|
}
|
|
}
|
|
|
|
void Actor::drawCostume(Costume *costume) {
|
|
for (int l = 0; l < MAX_SHADOWS; l++) {
|
|
if (!shouldDrawShadow(l))
|
|
continue;
|
|
g_driver->setShadow(&_shadowArray[l]);
|
|
g_driver->setShadowMode();
|
|
g_driver->drawShadowPlanes();
|
|
g_driver->startActorDraw(this);
|
|
costume->draw();
|
|
g_driver->finishActorDraw();
|
|
g_driver->clearShadowMode();
|
|
g_driver->setShadow(nullptr);
|
|
}
|
|
|
|
// normal draw actor
|
|
g_driver->startActorDraw(this);
|
|
costume->draw();
|
|
g_driver->finishActorDraw();
|
|
}
|
|
|
|
void Actor::setShadowPlane(const char *n) {
|
|
assert(_activeShadowSlot != -1);
|
|
|
|
_shadowArray[_activeShadowSlot].name = n;
|
|
}
|
|
|
|
void Actor::addShadowPlane(const char *n, Set *scene, int shadowId) {
|
|
assert(shadowId != -1);
|
|
|
|
// This needs to be an exact match, because with a substring search a search for sector
|
|
// "shadow1" would return sector "shadow10" in set st, causing shadows not to be cast
|
|
// on the street.
|
|
Sector *sector = scene->getSectorByName(n);
|
|
if (sector) {
|
|
// Create a copy so we are sure it will not be deleted by the Set destructor
|
|
// behind our back. This is important when Membrillo phones Velasco to tell him
|
|
// Naranja is dead, because the scene changes back and forth few times and so
|
|
// the scenes' sectors are deleted while they are still keeped by the actors.
|
|
Plane p = { scene->getName(), new Sector(*sector) };
|
|
_shadowArray[shadowId].planeList.push_back(p);
|
|
g_grim->flagRefreshShadowMask(true);
|
|
}
|
|
}
|
|
|
|
bool Actor::shouldDrawShadow(int shadowId) {
|
|
Shadow *shadow = &_shadowArray[shadowId];
|
|
if (!shadow->active)
|
|
return false;
|
|
|
|
// Don't draw a shadow if the shadow caster and the actor are on different sides
|
|
// of the shadow plane.
|
|
|
|
if (shadow->planeList.size() == 0)
|
|
return false;
|
|
Sector *sector = shadow->planeList.front().sector;
|
|
Math::Vector3d n = sector->getNormal();
|
|
Math::Vector3d p = sector->getVertices()[0];
|
|
float d = -(n.x() * p.x() + n.y() * p.y() + n.z() * p.z());
|
|
|
|
Math::Vector3d bboxPos, bboxSize;
|
|
getBBoxInfo(bboxPos, bboxSize);
|
|
Math::Vector3d centerOffset = bboxPos + bboxSize * 0.5f;
|
|
p = getPos() + centerOffset;
|
|
|
|
bool actorSide = n.x() * p.x() + n.y() * p.y() + n.z() * p.z() + d < 0.f;
|
|
p = shadow->pos;
|
|
bool shadowSide = n.x() * p.x() + n.y() * p.y() + n.z() * p.z() + d < 0.f;
|
|
|
|
if (actorSide == shadowSide)
|
|
return true;
|
|
return false;
|
|
}
|
|
|
|
void Actor::addShadowPlane(const char *n) {
|
|
addShadowPlane(n, g_grim->getCurrSet(), _activeShadowSlot);
|
|
}
|
|
|
|
void Actor::setActiveShadow(int shadowId) {
|
|
assert(shadowId >= 0 && shadowId < MAX_SHADOWS);
|
|
|
|
_activeShadowSlot = shadowId;
|
|
_shadowArray[_activeShadowSlot].active = true;
|
|
}
|
|
|
|
void Actor::setShadowValid(int valid) {
|
|
if (valid == -1)
|
|
_shadowArray[_activeShadowSlot].dontNegate = true;
|
|
else
|
|
_shadowArray[_activeShadowSlot].dontNegate = false;
|
|
}
|
|
|
|
void Actor::setActivateShadow(int shadowId, bool state) {
|
|
assert(shadowId >= 0 && shadowId < MAX_SHADOWS);
|
|
|
|
_shadowArray[shadowId].active = state;
|
|
}
|
|
|
|
void Actor::setShadowPoint(const Math::Vector3d &p) {
|
|
assert(_activeShadowSlot != -1);
|
|
|
|
_shadowArray[_activeShadowSlot].pos = p;
|
|
}
|
|
|
|
void Actor::setShadowColor(const Color &color) {
|
|
assert(_activeShadowSlot != -1);
|
|
|
|
_shadowArray[_activeShadowSlot].color = color;
|
|
}
|
|
|
|
void Actor::clearShadowPlanes() {
|
|
for (int i = 0; i < MAX_SHADOWS; i++) {
|
|
clearShadowPlane(i);
|
|
}
|
|
}
|
|
|
|
void Actor::clearShadowPlane(int i) {
|
|
Shadow *shadow = &_shadowArray[i];
|
|
while (!shadow->planeList.empty()) {
|
|
delete shadow->planeList.back().sector;
|
|
shadow->planeList.pop_back();
|
|
}
|
|
shadow->active = false;
|
|
shadow->dontNegate = false;
|
|
|
|
g_driver->destroyShadow(shadow);
|
|
}
|
|
|
|
void Actor::putInSet(const Common::String &set) {
|
|
if (_drawnToClean) {
|
|
// actor was frozen and...
|
|
if (set == "") {
|
|
// ...is getting unbound from set.
|
|
// This is done in game scripts to reduce the drawing
|
|
// workload in original engine, and we should not need
|
|
// it here. And implementing off-screen buffers is
|
|
// highly non-trivial. Disobey so it is still drawn.
|
|
_fakeUnbound = true;
|
|
return;
|
|
} else {
|
|
// ...is getting bound to a set. Thaw and continue.
|
|
_drawnToClean = false;
|
|
}
|
|
}
|
|
_fakeUnbound = false;
|
|
// The set should change immediately, otherwise a very rapid set change
|
|
// for an actor will be recognized incorrectly and the actor will be lost.
|
|
_setName = set;
|
|
|
|
g_grim->invalidateActiveActorsList();
|
|
}
|
|
|
|
bool Actor::isDrawableInSet(const Common::String &set) const {
|
|
return _setName == set;
|
|
}
|
|
|
|
bool Actor::isInSet(const Common::String &set) const {
|
|
return !_fakeUnbound && _setName == set;
|
|
}
|
|
|
|
void Actor::freeCostume(Costume *costume) {
|
|
Debug::debug(Debug::Actors, "Freeing costume %s", costume->getFilename().c_str());
|
|
freeCostumeChore(costume, &_restChore);
|
|
freeCostumeChore(costume, &_walkChore);
|
|
freeCostumeChore(costume, &_leftTurnChore);
|
|
freeCostumeChore(costume, &_rightTurnChore);
|
|
freeCostumeChore(costume, &_mumbleChore);
|
|
for (int i = 0; i < 10; i++)
|
|
freeCostumeChore(costume, &_talkChore[i]);
|
|
delete costume;
|
|
}
|
|
|
|
void Actor::freeCostumeChore(const Costume *toFree, ActionChore *chore) {
|
|
if (chore->_costume == toFree) {
|
|
*chore = ActionChore();
|
|
}
|
|
}
|
|
|
|
void Actor::stopTalking() {
|
|
// _talkChore[0] is *_stop_talk
|
|
// Don't playLooping it, or else manny's mouth will flicker when he smokes.
|
|
_talkChore[0].setLastFrame();
|
|
}
|
|
|
|
bool Actor::stopMumbleChore() {
|
|
if (_mumbleChore.isPlaying()) {
|
|
_mumbleChore.stop();
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
void Actor::setCollisionMode(CollisionMode mode) {
|
|
_collisionMode = mode;
|
|
}
|
|
|
|
void Actor::setCollisionScale(float scale) {
|
|
_collisionScale = scale;
|
|
}
|
|
|
|
Math::Vector3d Actor::handleCollisionTo(const Math::Vector3d &from, const Math::Vector3d &pos) const {
|
|
if (_collisionMode == CollisionOff) {
|
|
return pos;
|
|
}
|
|
|
|
Math::Vector3d p = pos;
|
|
Math::Vector3d moveVec = pos - _pos;
|
|
for (Actor *a : Actor::getPool()) {
|
|
if (a != this && a->isInSet(_setName) && a->isVisible()) {
|
|
p = a->getTangentPos(from, p);
|
|
handleCollisionWith(a, _collisionMode, &moveVec);
|
|
}
|
|
}
|
|
return p;
|
|
}
|
|
|
|
Math::Vector3d Actor::getTangentPos(const Math::Vector3d &pos, const Math::Vector3d &dest) const {
|
|
if (_collisionMode == CollisionOff) {
|
|
return dest;
|
|
}
|
|
if (pos.getDistanceTo(dest) < Math::epsilon) {
|
|
return dest;
|
|
}
|
|
Math::Vector3d p;
|
|
float size;
|
|
if (!getSphereInfo(false, size, p))
|
|
return dest;
|
|
Math::Vector2d p1(pos.x(), pos.y());
|
|
Math::Vector2d p2(dest.x(), dest.y());
|
|
if (p1.getDistanceTo(p2) < Math::epsilon) {
|
|
return dest;
|
|
}
|
|
Math::Segment2d segment(p1, p2);
|
|
|
|
// TODO: collision with Box
|
|
// if (_collisionMode == CollisionSphere) {
|
|
Math::Vector2d center(p.x(), p.y());
|
|
|
|
Math::Vector2d inter;
|
|
float distance = segment.getLine().getDistanceTo(center, &inter);
|
|
|
|
if (distance < size && segment.containsPoint(inter)) {
|
|
Math::Vector2d v(inter - center);
|
|
v.normalize();
|
|
v *= size;
|
|
v += center;
|
|
|
|
return Math::Vector3d(v.getX(), v.getY(), dest.z());
|
|
}
|
|
// } else {
|
|
// }
|
|
|
|
return dest;
|
|
}
|
|
|
|
void Actor::setLocalAlpha(unsigned int vertex, float alpha) {
|
|
if (vertex >= _localAlpha.size()) {
|
|
_localAlpha.resize(MAX(MAX_LOCAL_ALPHA_VERTICES, vertex + 1));
|
|
}
|
|
_localAlpha[vertex] = alpha;
|
|
}
|
|
|
|
void Actor::setLocalAlphaMode(unsigned int vertex, AlphaMode alphaMode) {
|
|
if (vertex >= _localAlphaMode.size()) {
|
|
_localAlphaMode.resize(MAX(MAX_LOCAL_ALPHA_VERTICES, vertex + 1));
|
|
}
|
|
_localAlphaMode[vertex] = alphaMode;
|
|
}
|
|
|
|
bool Actor::hasLocalAlpha() const {
|
|
return !_localAlphaMode.empty();
|
|
}
|
|
|
|
float Actor::getLocalAlpha(unsigned int vertex) const {
|
|
if (vertex < _localAlphaMode.size() && vertex < _localAlpha.size() && _localAlphaMode[vertex] == Actor::AlphaReplace) {
|
|
return _localAlpha[vertex];
|
|
} else {
|
|
return 1.0f;
|
|
}
|
|
}
|
|
|
|
void Actor::getBBoxInfo(Math::Vector3d &bboxPos, Math::Vector3d &bboxSize) const {
|
|
if (g_grim->getGameType() == GType_MONKEY4) {
|
|
EMICostume *costume = static_cast<EMICostume *>(getCurrentCostume());
|
|
if (!costume) {
|
|
bboxPos = Math::Vector3d(0, 0, 0);
|
|
bboxSize = Math::Vector3d(0, 0, 0);
|
|
return;
|
|
}
|
|
EMIModel *model = costume->getEMIModel();
|
|
bboxPos = *model->_center;
|
|
bboxSize = *model->_boxData2 - *model->_boxData;
|
|
} else {
|
|
Model *model = getCurrentCostume()->getModel();
|
|
bboxPos = model->_bboxPos;
|
|
bboxSize = model->_bboxSize;
|
|
}
|
|
}
|
|
|
|
bool Actor::getSphereInfo(bool adjustZ, float &size, Math::Vector3d &p) const {
|
|
if (g_grim->getGameType() == GType_MONKEY4) {
|
|
EMICostume *costume = static_cast<EMICostume *>(getCurrentCostume());
|
|
if (!costume) {
|
|
Debug::warning(Debug::Actors, "Actor::getSphereInfo: actor \"%s\" has no costume", getName().c_str());
|
|
return false;
|
|
}
|
|
EMIModel *model = costume->getEMIModel();
|
|
assert(model);
|
|
p = _pos + *(model->_center);
|
|
// pre-scale factor of 0.8 was guessed by comparing with the original game
|
|
size = model->_radius * _collisionScale * 0.8f;
|
|
} else {
|
|
Model *model = getCurrentCostume()->getModel();
|
|
assert(model);
|
|
|
|
p = _pos + model->_insertOffset;
|
|
// center the sphere on the model center.
|
|
if (adjustZ) {
|
|
p.z() += model->_bboxSize.z() / 2.f;
|
|
}
|
|
size = model->_radius * _collisionScale;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
bool Actor::handleCollisionWith(Actor *actor, CollisionMode mode, Math::Vector3d *vec) const {
|
|
if (actor->_collisionMode == CollisionOff || actor == this) {
|
|
return false;
|
|
}
|
|
|
|
if (!actor->getCurrentCostume()) {
|
|
return false;
|
|
}
|
|
|
|
Math::Vector3d p1, p2;
|
|
float size1, size2;
|
|
// you may ask: why center the sphere of the first actor only (by setting adjustZ to true)?
|
|
// because it seems the original does so.
|
|
// if you change this code test this places: the rocks in lb and bv (both when booting directly in the
|
|
// set and when coming in from another one) and the poles in xb.
|
|
if (!this->getSphereInfo(true, size1, p1) || !actor->getSphereInfo(false, size2, p2)) {
|
|
return false;
|
|
}
|
|
|
|
CollisionMode mode1 = mode;
|
|
CollisionMode mode2 = actor->_collisionMode;
|
|
|
|
if (mode1 == CollisionSphere && mode2 == CollisionSphere) {
|
|
Math::Vector3d pos = p1 + *vec;
|
|
float distance = (pos - p2).getMagnitude();
|
|
if (distance < size1 + size2) {
|
|
// Move the destination point so that its distance from the
|
|
// center of the circle is size1+size2.
|
|
Math::Vector3d v = pos - p2;
|
|
v.normalize();
|
|
v *= size1 + size2;
|
|
*vec = v + p2 - p1;
|
|
|
|
collisionHandlerCallback(actor);
|
|
return true;
|
|
}
|
|
} else if (mode1 == CollisionBox && mode2 == CollisionBox) {
|
|
warning("Collision between box and box not implemented!");
|
|
return false;
|
|
} else {
|
|
Math::Vector3d bboxSize1, bboxSize2;
|
|
Math::Vector3d bboxPos1, bboxPos2;
|
|
|
|
// get bboxSize and bboxPos for the current and the colliding actor
|
|
this->getBBoxInfo(bboxPos1, bboxSize1);
|
|
actor->getBBoxInfo(bboxPos2, bboxSize2);
|
|
|
|
Math::Rect2d rect;
|
|
|
|
Math::Vector3d bboxPos;
|
|
Math::Vector3d size;
|
|
float scale;
|
|
Math::Vector3d pos;
|
|
Math::Vector3d circlePos;
|
|
Math::Angle yaw;
|
|
|
|
Math::Vector2d circle;
|
|
float radius;
|
|
|
|
if (mode1 == CollisionBox) {
|
|
pos = p1 + *vec;
|
|
bboxPos = pos + bboxPos1;
|
|
size = bboxSize1;
|
|
scale = _collisionScale;
|
|
yaw = _yaw;
|
|
|
|
circle.setX(p2.x());
|
|
circle.setY(p2.y());
|
|
circlePos = p2;
|
|
radius = size2;
|
|
} else {
|
|
pos = p2;
|
|
bboxPos = p2 + bboxPos2;
|
|
size = bboxSize2;
|
|
scale = actor->_collisionScale;
|
|
yaw = actor->_yaw;
|
|
|
|
circle.setX(p1.x() + vec->x());
|
|
circle.setY(p1.y() + vec->y());
|
|
circlePos = p1;
|
|
radius = size1;
|
|
}
|
|
|
|
rect._topLeft = Math::Vector2d(bboxPos.x(), bboxPos.y() + size.y());
|
|
rect._topRight = Math::Vector2d(bboxPos.x() + size.x(), bboxPos.y() + size.y());
|
|
rect._bottomLeft = Math::Vector2d(bboxPos.x(), bboxPos.y());
|
|
rect._bottomRight = Math::Vector2d(bboxPos.x() + size.x(), bboxPos.y());
|
|
|
|
rect.scale(scale);
|
|
rect.rotateAround(Math::Vector2d(pos.x(), pos.y()), yaw);
|
|
|
|
if (rect.intersectsCircle(circle, radius)) {
|
|
Math::Vector2d center = rect.getCenter();
|
|
// Draw a line from the center of the rect to the place the character
|
|
// would go to.
|
|
Math::Vector2d v = circle - center;
|
|
v.normalize();
|
|
|
|
Math::Segment2d edge;
|
|
// That line intersects (usually) an edge
|
|
rect.getIntersection(center, v, &edge);
|
|
// Take the perpendicular of that edge
|
|
Math::Line2d perpendicular = edge.getPerpendicular(circle);
|
|
|
|
Math::Vector3d point;
|
|
Math::Vector2d p;
|
|
// If that perpendicular intersects the edge
|
|
if (edge.intersectsLine(perpendicular, &p)) {
|
|
Math::Vector2d direction = perpendicular.getDirection();
|
|
direction.normalize();
|
|
|
|
// Move from the intersection until we are at a safe distance
|
|
Math::Vector2d point1(p - direction * radius);
|
|
Math::Vector2d point2(p + direction * radius);
|
|
|
|
if (center.getDistanceTo(point1) < center.getDistanceTo(point2)) {
|
|
point = point2.toVector3d();
|
|
} else {
|
|
point = point1.toVector3d();
|
|
}
|
|
} else { //if not we're around a corner
|
|
// Find the nearest vertex of the rect
|
|
Math::Vector2d vertex = rect.getTopLeft();
|
|
float distance = vertex.getDistanceTo(circle);
|
|
|
|
Math::Vector2d other = rect.getTopRight();
|
|
float otherDist = other.getDistanceTo(circle);
|
|
if (otherDist < distance) {
|
|
distance = otherDist;
|
|
vertex = other;
|
|
}
|
|
|
|
other = rect.getBottomLeft();
|
|
otherDist = other.getDistanceTo(circle);
|
|
if (otherDist < distance) {
|
|
distance = otherDist;
|
|
vertex = other;
|
|
}
|
|
|
|
other = rect.getBottomRight();
|
|
if (other.getDistanceTo(circle) < distance) {
|
|
vertex = other;
|
|
}
|
|
|
|
// and move on a circle around it
|
|
Math::Vector2d dst = circle - vertex;
|
|
dst.normalize();
|
|
dst = dst * radius;
|
|
point = (vertex + dst).toVector3d();
|
|
}
|
|
|
|
float z = vec->z();
|
|
*vec = point - circlePos;
|
|
vec->z() = z;
|
|
collisionHandlerCallback(actor);
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
void Actor::costumeMarkerCallback(int marker) {
|
|
LuaObjects objects;
|
|
objects.add(this);
|
|
objects.add(marker);
|
|
|
|
LuaBase::instance()->callback("costumeMarkerHandler", objects);
|
|
}
|
|
|
|
void Actor::collisionHandlerCallback(Actor *other) const {
|
|
LuaObjects objects;
|
|
objects.add(this);
|
|
objects.add(other);
|
|
|
|
LuaBase::instance()->callback("collisionHandler", objects);
|
|
|
|
LuaObjects objects2;
|
|
objects2.add(other);
|
|
objects2.add(this);
|
|
LuaBase::instance()->callback("collisionHandler", objects2);
|
|
}
|
|
|
|
const Math::Matrix4 Actor::getFinalMatrix() const {
|
|
// Defaults to identity (no rotation)
|
|
Math::Matrix4 m;
|
|
|
|
// Add the rotation and translation from the parent actor, if this actor is attached
|
|
if (isAttached()) {
|
|
Actor *attachedActor = Actor::getPool().getObject(_attachedActor);
|
|
m = attachedActor->getFinalMatrix();
|
|
|
|
// If this actor is attached to a joint, add that rotation
|
|
EMICostume *cost = static_cast<EMICostume *>(attachedActor->getCurrentCostume());
|
|
if (cost && cost->_emiSkel && cost->_emiSkel->_obj) {
|
|
Joint *j = cost->_emiSkel->_obj->getJointNamed(_attachedJoint);
|
|
m = m * j->_finalMatrix;
|
|
}
|
|
}
|
|
|
|
// Translate by this actor's position
|
|
Math::Vector3d pp = getPos();
|
|
Math::Matrix4 t;
|
|
t.setToIdentity();
|
|
t.setPosition(pp);
|
|
m = m * t;
|
|
|
|
// Scaling could be applied here as follows:
|
|
//
|
|
// const float &scale = getScale();
|
|
// t.setToIdentity();
|
|
// t.setValue(3, 3, 1.0 / scale);
|
|
// m = m * t;
|
|
//
|
|
// However, actor's _scale is only changed via the lua call SetActorScale
|
|
// which is not used in EMI. Actor::getFinalMatrix() is only used for EMI
|
|
// so the additional scaling can be omitted.
|
|
|
|
// Finish with this actor's rotation
|
|
Math::Matrix4 rotMat(getRoll(), getYaw(), getPitch(), Math::EO_ZYX);
|
|
m = m * rotMat;
|
|
return m;
|
|
}
|
|
|
|
Math::Vector3d Actor::getWorldPos() const {
|
|
if (! isAttached())
|
|
return getPos();
|
|
|
|
return getFinalMatrix().getPosition();
|
|
}
|
|
|
|
Math::Quaternion Actor::getRotationQuat() const {
|
|
if (g_grim->getGameType() == GType_MONKEY4) {
|
|
const Math::Matrix4 m = getFinalMatrix();
|
|
return Math::Quaternion(m).inverse();
|
|
} else {
|
|
return Math::Quaternion::fromEuler(_yaw, _pitch, _roll, Math::EO_ZXY).inverse();
|
|
}
|
|
}
|
|
|
|
Math::Vector3d Actor::getHeadPos() const {
|
|
if (g_grim->getGameType() == GType_GRIM) {
|
|
for (Common::List<Costume *>::const_iterator i = _costumeStack.begin(); i != _costumeStack.end(); ++i) {
|
|
int headJoint = (*i)->getHeadJoint();
|
|
if (headJoint == -1)
|
|
continue;
|
|
|
|
ModelNode *allNodes = (*i)->getModelNodes();
|
|
ModelNode *node = allNodes + headJoint;
|
|
|
|
node->_needsUpdate = true;
|
|
ModelNode *root = node;
|
|
while (root->_parent) {
|
|
root = root->_parent;
|
|
root->_needsUpdate = true;
|
|
}
|
|
|
|
Math::Matrix4 matrix;
|
|
matrix.setPosition(_pos);
|
|
matrix.buildFromEuler(_yaw, _pitch, _roll, Math::EO_ZXY);
|
|
root->setMatrix(matrix);
|
|
root->update();
|
|
|
|
return node->_pivotMatrix.getPosition();
|
|
}
|
|
}
|
|
|
|
return getWorldPos();
|
|
}
|
|
|
|
void Actor::setSortOrder(const int order) {
|
|
_sortOrder = order;
|
|
|
|
// If this actor is attached to another actor, we'll use the
|
|
// explicitly specified sort order instead of the parent actor's
|
|
// sort order from now on.
|
|
if (_useParentSortOrder)
|
|
_useParentSortOrder = false;
|
|
}
|
|
|
|
int Actor::getSortOrder() const {
|
|
return _sortOrder;
|
|
}
|
|
|
|
int Actor::getEffectiveSortOrder() const {
|
|
if (_useParentSortOrder && _attachedActor != 0) {
|
|
Actor *attachedActor = Actor::getPool().getObject(_attachedActor);
|
|
return attachedActor->getEffectiveSortOrder();
|
|
}
|
|
return _sectorSortOrder >= 0 ? _sectorSortOrder : getSortOrder();
|
|
}
|
|
|
|
void Actor::activateShadow(bool active, const char *shadowName) {
|
|
Set *set = g_grim->getCurrSet();
|
|
if (!set) {
|
|
warning("Actor %s trying to activate shadow to null Set", getName().c_str());
|
|
return;
|
|
}
|
|
if (!shadowName) {
|
|
for (int i = 0; i < set->getShadowCount(); ++i) {
|
|
activateShadow(active, set->getShadow(i));
|
|
}
|
|
} else {
|
|
SetShadow *shadow = set->getShadowByName(shadowName);
|
|
if (shadow)
|
|
activateShadow(active, shadow);
|
|
}
|
|
}
|
|
|
|
void Actor::activateShadow(bool active, SetShadow *setShadow) {
|
|
int shadowId = -1;
|
|
for (int i = 0; i < MAX_SHADOWS; i++) {
|
|
if (setShadow->_name.equals(_shadowArray[i].name)) {
|
|
shadowId = i;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (shadowId == -1) {
|
|
for (int i = 0; i < MAX_SHADOWS; i++) {
|
|
if (!_shadowArray[i].active) {
|
|
shadowId = i;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (shadowId == -1) {
|
|
warning("Actor %s trying to activate shadow %s, but all shadow slots are in use", getName().c_str(), setShadow->_name.c_str());
|
|
return;
|
|
}
|
|
|
|
clearShadowPlane(shadowId);
|
|
setActivateShadow(shadowId, active);
|
|
|
|
if (active) {
|
|
setActiveShadow(shadowId);
|
|
setShadowPoint(setShadow->_shadowPoint);
|
|
setShadowPlane(setShadow->_name.c_str());
|
|
setShadowColor(setShadow->_color);
|
|
setShadowValid(-1); // Don't negate the normal.
|
|
|
|
Common::List<Common::String>::iterator it;
|
|
for (it = setShadow->_sectorNames.begin(); it != setShadow->_sectorNames.end(); ++it) {
|
|
addShadowPlane((*it).c_str(), g_grim->getCurrSet(), shadowId);
|
|
}
|
|
}
|
|
}
|
|
|
|
void Actor::attachToActor(Actor *parent, const char *joint) {
|
|
assert(parent != nullptr);
|
|
// No need to attach if we're already attached to this parent
|
|
if (parent->getId() == _attachedActor)
|
|
return;
|
|
// If we're attached to a different parent, detach first
|
|
if (_attachedActor != 0)
|
|
detach();
|
|
|
|
// Find the new rotation relative to the parent actor's rotation
|
|
// Note: Any joint rotation is a part of the parent actor's rotation Quat
|
|
Math::Quaternion newRot = getRotationQuat().inverse() * parent->getRotationQuat();
|
|
|
|
// Find the new position coordinates
|
|
Math::Matrix4 parentMatrix = parent->getFinalMatrix();
|
|
|
|
// If the parent has a skeleton, check if it has the requested joint
|
|
// Some models (pile o' boulders) don't have a skeleton
|
|
Common::String jointStr = joint ? joint : "";
|
|
EMICostume *cost = static_cast<EMICostume *>(parent->getCurrentCostume());
|
|
if (cost && cost->_emiSkel && cost->_emiSkel->_obj) {
|
|
assert(cost->_emiSkel->_obj->hasJoint(jointStr));
|
|
|
|
// Add the rotation from the attached actor's joint
|
|
Joint *j = cost->_emiSkel->_obj->getJointNamed(_attachedJoint);
|
|
newRot = newRot.inverse() * j->_finalQuat;
|
|
|
|
// Get the final position coordinates
|
|
_pos = _pos - j->_finalMatrix.getPosition();
|
|
j->_finalMatrix.transpose();
|
|
j->_finalMatrix.transform(&_pos, true);
|
|
}
|
|
|
|
// Get the final rotation euler coordinates
|
|
newRot.getEuler(&_roll, &_yaw, &_pitch, Math::EO_ZYX);
|
|
|
|
// Get the final position coordinates
|
|
_pos = _pos - parentMatrix.getPosition();
|
|
parentMatrix.transpose();
|
|
parentMatrix.transform(&_pos, true);
|
|
|
|
// Save the attachement info
|
|
_attachedActor = parent->getId();
|
|
_attachedJoint = jointStr;
|
|
|
|
// Use the parent actor's sort order.
|
|
_useParentSortOrder = true;
|
|
}
|
|
|
|
void Actor::detach() {
|
|
if (!isAttached())
|
|
return;
|
|
|
|
// Replace our sort order with the parent actor's sort order. Note
|
|
// that we do this even if a sort order was explicitly specified for
|
|
// the actor during the time it was attached (in which case the actor
|
|
// doesn't respect the parent actor's sort order). This seems weird,
|
|
// but matches the behavior of the original engine.
|
|
Actor *attachedActor = Actor::getPool().getObject(_attachedActor);
|
|
_sortOrder = attachedActor->getEffectiveSortOrder();
|
|
_useParentSortOrder = false;
|
|
|
|
// Position and rotate the actor in relation to the world coords
|
|
setPos(getWorldPos());
|
|
Math::Quaternion q = getRotationQuat();
|
|
q.inverse().getEuler(&_roll, &_yaw, &_pitch, Math::EO_ZYX);
|
|
|
|
// Remove the attached actor
|
|
_attachedActor = 0;
|
|
_attachedJoint = "";
|
|
}
|
|
|
|
void Actor::drawToCleanBuffer() {
|
|
_drawnToClean = true;
|
|
}
|
|
|
|
MaterialPtr Actor::findMaterial(const Common::String &name) {
|
|
Common::String fixedName = g_resourceloader->fixFilename(name, false);
|
|
Common::List<MaterialPtr>::iterator it = _materials.begin();
|
|
for (; it != _materials.end(); ++it) {
|
|
if (*it) {
|
|
if ((*it)->getFilename() == fixedName) {
|
|
return *it;
|
|
}
|
|
} else {
|
|
it = _materials.erase(it);
|
|
--it;
|
|
}
|
|
}
|
|
return (MaterialPtr)nullptr;
|
|
}
|
|
|
|
MaterialPtr Actor::loadMaterial(const Common::String &name, bool clamp) {
|
|
MaterialPtr mat = findMaterial(name);
|
|
if (!mat) {
|
|
mat = g_resourceloader->loadMaterial(name.c_str(), nullptr, clamp);
|
|
// Note: We store a weak reference.
|
|
_materials.push_back(mat);
|
|
mat->dereference();
|
|
}
|
|
return mat;
|
|
}
|
|
|
|
unsigned const int Actor::ActionChore::fadeTime = 150;
|
|
unsigned const int Actor::ActionChore::talkFadeTime = 50;
|
|
|
|
Actor::ActionChore::ActionChore() :
|
|
_costume(nullptr),
|
|
_chore(-1) {
|
|
}
|
|
|
|
Actor::ActionChore::ActionChore(Costume *cost, int chore) :
|
|
_costume(cost),
|
|
_chore(chore) {
|
|
}
|
|
|
|
void Actor::ActionChore::play(bool fade, unsigned int time) {
|
|
if (isValid()) {
|
|
if (fade) {
|
|
_costume->playChore(_chore, time);
|
|
} else {
|
|
_costume->playChore(_chore);
|
|
}
|
|
}
|
|
}
|
|
|
|
void Actor::ActionChore::playLooping(bool fade, unsigned int time) {
|
|
if (isValid()) {
|
|
if (fade) {
|
|
_costume->playChoreLooping(_chore, time);
|
|
} else {
|
|
_costume->playChoreLooping(_chore);
|
|
}
|
|
}
|
|
}
|
|
|
|
void Actor::ActionChore::stop(bool fade, unsigned int time) {
|
|
if (isValid()) {
|
|
if (fade) {
|
|
_costume->stopChore(_chore, time);
|
|
} else {
|
|
_costume->stopChore(_chore);
|
|
}
|
|
}
|
|
}
|
|
|
|
void Actor::ActionChore::setLastFrame() {
|
|
if (isValid()) {
|
|
_costume->setChoreLastFrame(_chore);
|
|
}
|
|
}
|
|
|
|
bool Actor::ActionChore::isPlaying() const {
|
|
return (isValid() && _costume->isChoring(_chore, false) >= 0);
|
|
}
|
|
|
|
void Actor::ActionChore::saveState(SaveGame *savedState) const {
|
|
if (_costume) {
|
|
savedState->writeBool(true);
|
|
savedState->writeString(_costume->getFilename());
|
|
} else {
|
|
savedState->writeBool(false);
|
|
}
|
|
savedState->writeLESint32(_chore);
|
|
}
|
|
|
|
void Actor::ActionChore::restoreState(SaveGame *savedState, Actor *actor) {
|
|
if (savedState->readBool()) {
|
|
Common::String fname = savedState->readString();
|
|
_costume = actor->findCostume(fname);
|
|
} else {
|
|
_costume = nullptr;
|
|
}
|
|
_chore = savedState->readLESint32();
|
|
}
|
|
|
|
} // end of namespace Grim
|