scummvm/engines/grim/actor.cpp

1428 lines
39 KiB
C++
Raw Normal View History

2009-05-26 14:13:08 +00:00
/* Residual - A 3D game interpreter
*
* Residual is the legal property of its developers, whose names
2011-04-16 14:12:44 +02:00
* are too numerous to list here. Please refer to the COPYRIGHT
* file distributed with this source distribution.
2006-04-02 14:20:45 +00:00
*
* 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
*
*/
2003-08-15 18:00:22 +00:00
2011-05-08 15:38:26 +02:00
#define FORBIDDEN_SYMBOL_EXCEPTION_printf
2011-05-08 17:40:31 +02:00
#define FORBIDDEN_SYMBOL_EXCEPTION_chdir
#define FORBIDDEN_SYMBOL_EXCEPTION_getcwd
#define FORBIDDEN_SYMBOL_EXCEPTION_getwd
2011-05-24 13:03:59 -04:00
#define FORBIDDEN_SYMBOL_EXCEPTION_mkdir
2011-05-08 17:40:31 +02:00
#define FORBIDDEN_SYMBOL_EXCEPTION_unlink
2011-05-08 15:38:26 +02:00
#include "graphics/line3d.h"
#include "engines/grim/actor.h"
#include "engines/grim/grim.h"
#include "engines/grim/colormap.h"
#include "engines/grim/costume.h"
#include "engines/grim/lipsync.h"
#include "engines/grim/movie/movie.h"
#include "engines/grim/imuse/imuse.h"
#include "engines/grim/lua.h"
2011-05-13 17:55:14 -07:00
#include "engines/grim/resource.h"
#include "engines/grim/savegame.h"
2005-01-01 12:27:57 +00:00
2009-05-25 06:49:57 +00:00
namespace Grim {
2008-09-26 17:48:46 +00:00
int g_winX1, g_winY1, g_winX2, g_winY2;
Actor::Actor(const Common::String &actorName) :
2011-05-05 11:25:08 +02:00
Object(), _name(actorName), _setName(""), _talkColor(g_grim->getColor(2)), _pos(0, 0, 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
2011-04-24 12:02:33 +02:00
// _walkRate so Glottis at the demon beaver entrance can walk and
// so Chepito in su.set
_pitch(0), _yaw(0), _roll(0), _walkRate(0.3f), _turnRate(100.0f),
2005-08-10 08:33:45 +00:00
_reflectionAngle(80),
2009-04-21 18:04:24 +00:00
_visible(true), _lipSync(NULL), _turning(false), _walking(false),
2004-12-09 23:55:43 +00:00
_restCostume(NULL), _restChore(-1),
_walkCostume(NULL), _walkChore(-1), _walkedLast(false), _walkedCur(false),
_turnCostume(NULL), _leftTurnChore(-1), _rightTurnChore(-1),
_lastTurnDir(0), _currTurnDir(0),
_mumbleCostume(NULL), _mumbleChore(-1), _sayLineText(0) {
2004-12-09 23:55:43 +00:00
_lookingMode = false;
_lookAtRate = 200;
2004-12-09 23:55:43 +00:00
_constrain = false;
2005-01-14 09:43:09 +00:00
_talkSoundName = "";
_activeShadowSlot = -1;
2007-03-12 21:50:44 +00:00
_shadowArray = new Shadow[5];
2008-09-26 17:48:46 +00:00
_winX1 = _winY1 = 1000;
_winX2 = _winY2 = -1000;
_toClean = false;
_lastWasLeft = false;
_lastStepTime = 0;
_running = false;
2011-04-28 22:07:36 +02:00
_scale = 1.f;
2011-06-06 15:11:50 +02:00
_timeScale = 1.f;
2007-03-19 12:47:14 +00:00
for (int i = 0; i < 5; i++) {
_shadowArray[i].active = false;
_shadowArray[i].dontNegate = false;
_shadowArray[i].shadowMask = NULL;
_shadowArray[i].shadowMaskSize = 0;
2007-03-19 12:47:14 +00:00
}
for (int i = 0; i < 10; i++) {
2004-12-09 23:55:43 +00:00
_talkCostume[i] = NULL;
_talkChore[i] = -1;
}
g_grim->registerActor(this);
2003-08-15 18:00:22 +00:00
}
Actor::Actor() :
Object() {
_shadowArray = new Shadow[5];
_winX1 = _winY1 = 1000;
_winX2 = _winY2 = -1000;
_toClean = false;
_lastWasLeft = false;
_lastStepTime = 0;
_running = false;
2011-04-28 22:07:36 +02:00
_scale = 1.f;
2011-06-06 15:11:50 +02:00
_timeScale = 1.f;
for (int i = 0; i < 5; i++) {
_shadowArray[i].active = false;
_shadowArray[i].dontNegate = false;
_shadowArray[i].shadowMask = NULL;
_shadowArray[i].shadowMaskSize = 0;
}
}
2009-06-22 14:34:15 +00:00
Actor::~Actor() {
if (_shadowArray) {
clearShadowPlanes();
delete[] _shadowArray;
}
2011-05-04 18:55:24 +02:00
while (!_costumeStack.empty()) {
delete _costumeStack.back();
_costumeStack.pop_back();
}
g_grim->killActor(this);
}
void Actor::saveState(SaveGame *savedState) const {
2009-06-22 14:34:15 +00:00
// store actor name
savedState->writeString(_name);
savedState->writeString(_setName);
if (_talkColor) {
2011-05-02 03:50:09 +08:00
savedState->writeLEUint32(_talkColor->getId());
} else {
savedState->writeLEUint32(0);
}
savedState->writeVector3d(_pos);
savedState->writeFloat(_pitch);
savedState->writeFloat(_yaw);
savedState->writeFloat(_roll);
savedState->writeFloat(_walkRate);
savedState->writeFloat(_turnRate);
savedState->writeLESint32(_constrain);
savedState->writeFloat(_reflectionAngle);
savedState->writeLESint32(_visible);
savedState->writeLESint32(_lookingMode),
2011-06-08 17:30:30 +02:00
savedState->writeFloat(_scale);
savedState->writeFloat(_timeScale);
savedState->writeString(_talkSoundName);
if (_lipSync) {
savedState->writeLEUint32(1);
savedState->writeString(_lipSync->getFilename());
} else {
savedState->writeLEUint32(0);
}
savedState->writeLESint32(_costumeStack.size());
for (Common::List<Costume *>::const_iterator i = _costumeStack.begin(); i != _costumeStack.end(); ++i) {
Costume *c = *i;
savedState->writeString(c->getFilename());
2011-05-05 11:08:05 +02:00
Costume *pc = c->getPreviousCostume();
int depth = 0;
while (pc) {
++depth;
2011-05-05 11:08:05 +02:00
pc = pc->getPreviousCostume();
}
savedState->writeLEUint32(depth);
2011-05-05 11:08:05 +02:00
pc = c->getPreviousCostume();
for (int j = 0; j < depth; ++j) { //save the previousCostume hierarchy
savedState->writeString(pc->getFilename());
2011-05-05 11:08:05 +02:00
pc = pc->getPreviousCostume();
}
c->saveState(savedState);
}
savedState->writeLESint32(_turning);
savedState->writeFloat(_destYaw);
savedState->writeLESint32(_walking);
savedState->writeVector3d(_destPos);
2009-06-22 14:34:15 +00:00
if (_restCostume) {
savedState->writeLEUint32(1);
savedState->writeString(_restCostume->getFilename());
} else {
savedState->writeLEUint32(0);
}
savedState->writeLESint32(_restChore);
if (_walkCostume) {
savedState->writeLEUint32(1);
savedState->writeString(_walkCostume->getFilename());
} else {
savedState->writeLEUint32(0);
}
savedState->writeLESint32(_walkChore);
savedState->writeLESint32(_walkedLast);
savedState->writeLESint32(_walkedCur);
if (_turnCostume) {
savedState->writeLEUint32(1);
savedState->writeString(_turnCostume->getFilename());
} else {
savedState->writeLEUint32(0);
}
savedState->writeLESint32(_leftTurnChore);
savedState->writeLESint32(_rightTurnChore);
savedState->writeLESint32(_lastTurnDir);
savedState->writeLESint32(_currTurnDir);
for (int i = 0; i < 10; ++i) {
if (_talkCostume[i]) {
savedState->writeLEUint32(1);
savedState->writeString(_talkCostume[i]->getFilename());
} else {
savedState->writeLEUint32(0);
}
savedState->writeLESint32(_talkChore[i]);
}
savedState->writeLESint32(_talkAnim);
if (_mumbleCostume) {
savedState->writeLEUint32(1);
savedState->writeString(_mumbleCostume->getFilename());
} else {
savedState->writeLEUint32(0);
}
savedState->writeLESint32(_mumbleChore);
for (int i = 0; i < 5; ++i) {
Shadow &shadow = _shadowArray[i];
savedState->writeString(shadow.name);
savedState->writeVector3d(shadow.pos);
savedState->writeLESint32(shadow.planeList.size());
2011-05-05 11:25:08 +02:00
// Cannot use g_grim->getCurrScene() 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(shadow.shadowMaskSize);
savedState->write(shadow.shadowMask, shadow.shadowMaskSize);
savedState->writeLESint32(shadow.active);
savedState->writeLESint32(shadow.dontNegate);
}
savedState->writeLESint32(_activeShadowSlot);
savedState->writeLEUint32(_sayLineText);
savedState->writeVector3d(_lookAtVector);
savedState->writeFloat(_lookAtRate);
savedState->writeLESint32(_winX1);
savedState->writeLESint32(_winY1);
savedState->writeLESint32(_winX2);
savedState->writeLESint32(_winY2);
savedState->writeLESint32(_path.size());
for (Common::List<Graphics::Vector3d>::const_iterator i = _path.begin(); i != _path.end(); ++i) {
savedState->writeVector3d(*i);
}
}
bool Actor::restoreState(SaveGame *savedState) {
for (Common::List<Costume *>::const_iterator i = _costumeStack.begin(); i != _costumeStack.end(); ++i) {
delete *i;
}
_costumeStack.clear();
// load actor name
_name = savedState->readString();
_setName = savedState->readString();
2011-05-05 11:25:08 +02:00
_talkColor = g_grim->getColor(savedState->readLEUint32());
_pos = savedState->readVector3d();
_pitch = savedState->readFloat();
_yaw = savedState->readFloat();
_roll = savedState->readFloat();
_walkRate = savedState->readFloat();
_turnRate = savedState->readFloat();
_constrain = savedState->readLESint32();
_reflectionAngle = savedState->readFloat();
_visible = savedState->readLESint32();
_lookingMode = savedState->readLESint32();
2011-06-08 17:30:30 +02:00
_scale = savedState->readFloat();
_timeScale = savedState->readFloat();
_talkSoundName = savedState->readString();
if (savedState->readLEUint32()) {
Common::String fn = savedState->readString();
_lipSync = g_resourceloader->getLipSync(fn);
} else {
_lipSync = NULL;
}
int32 size = savedState->readLESint32();
for (int32 i = 0; i < size; ++i) {
Common::String fname = savedState->readString();
const int depth = savedState->readLEUint32();
Costume *pc = NULL;
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], pc);
}
}
delete[] names;
}
Costume *c = g_resourceloader->loadCostume(fname, pc);
c->restoreState(savedState);
_costumeStack.push_back(c);
}
_turning = savedState->readLESint32();
_destYaw = savedState->readFloat();
_walking = savedState->readLESint32();
_destPos = savedState->readVector3d();
if (savedState->readLEUint32()) {
Common::String fname = savedState->readString();
_restCostume = findCostume(fname);
} else {
_restCostume = NULL;
}
_restChore = savedState->readLESint32();
if (savedState->readLEUint32()) {
Common::String fname = savedState->readString();
_walkCostume = findCostume(fname);
} else {
_walkCostume = NULL;
}
_walkChore = savedState->readLESint32();
_walkedLast = savedState->readLESint32();
_walkedCur = savedState->readLESint32();
if (savedState->readLEUint32()) {
Common::String fname = savedState->readString();
_turnCostume = findCostume(fname);
} else {
_turnCostume = NULL;
}
_leftTurnChore = savedState->readLESint32();
_rightTurnChore = savedState->readLESint32();
_lastTurnDir = savedState->readLESint32();
_currTurnDir = savedState->readLESint32();
for (int i = 0; i < 10; ++i) {
if (savedState->readLEUint32()) {
Common::String fname = savedState->readString();
_talkCostume[i] = findCostume(fname);
} else {
_talkCostume[i] = NULL;
}
_talkChore[i] = savedState->readLESint32();
}
_talkAnim = savedState->readLESint32();
if (savedState->readLEUint32()) {
Common::String fname = savedState->readString();
_mumbleCostume = findCostume(fname);
} else {
_mumbleCostume = NULL;
}
_mumbleChore = savedState->readLESint32();
for (int i = 0; i < 5; ++i) {
Shadow &shadow = _shadowArray[i];
shadow.name = savedState->readString();
shadow.pos = savedState->readVector3d();
size = savedState->readLESint32();
shadow.planeList.clear();
Scene *scene = NULL;
for (int j = 0; j < size; ++j) {
Common::String setName = savedState->readString();
Common::String secName = savedState->readString();
if (!scene || scene->getName() != setName) {
scene = g_grim->findScene(setName);
}
if (scene) {
addShadowPlane(secName.c_str(), scene, i);
} else {
warning("%s: No scene \"%s\" found, cannot restore shadow on sector \"%s\"", getName().c_str(), setName.c_str(), secName.c_str());
}
}
shadow.shadowMaskSize = savedState->readLESint32();
delete[] shadow.shadowMask;
if (shadow.shadowMaskSize > 0) {
shadow.shadowMask = new byte[shadow.shadowMaskSize];
savedState->read(shadow.shadowMask, shadow.shadowMaskSize);
} else {
shadow.shadowMask = NULL;
}
shadow.active = savedState->readLESint32();
shadow.dontNegate = savedState->readLESint32();
}
_activeShadowSlot = savedState->readLESint32();
_sayLineText = savedState->readLEUint32();
_lookAtVector = savedState->readVector3d();
_lookAtRate = savedState->readFloat();
_winX1 = savedState->readLESint32();
_winY1 = savedState->readLESint32();
_winX2 = savedState->readLESint32();
_winY2 = savedState->readLESint32();
size = savedState->readLESint32();
for (int i = 0; i < size; ++i) {
_path.push_back(savedState->readVector3d());
}
return true;
2009-06-22 14:34:15 +00:00
}
2009-10-17 12:48:23 +00:00
void Actor::setYaw(float yawParam) {
// While the program correctly handle yaw angles outside
// of the range [0, 360), proper convention is to roll
// these values over correctly
2009-10-17 12:48:23 +00:00
if (yawParam >= 360.0)
_yaw = yawParam - 360;
else if (yawParam < 0.0)
_yaw = yawParam + 360;
else
2009-10-17 12:48:23 +00:00
_yaw = yawParam;
}
2009-10-17 12:48:23 +00:00
void Actor::setRot(float pitchParam, float yawParam, float rollParam) {
_pitch = pitchParam;
setYaw(yawParam);
_roll = rollParam;
_turning = false;
}
void Actor::setPos(Graphics::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 (_constrain && !_walking) {
g_grim->getCurrScene()->findClosestSector(_pos, NULL, &_pos);
}
}
2009-10-17 12:48:23 +00:00
void Actor::turnTo(float pitchParam, float yawParam, float rollParam) {
_pitch = pitchParam;
_roll = rollParam;
if (_yaw != yawParam) {
2004-12-09 23:55:43 +00:00
_turning = true;
2009-10-17 12:48:23 +00:00
_destYaw = yawParam;
} else
2004-12-09 23:55:43 +00:00
_turning = false;
}
void Actor::walkTo(const Graphics::Vector3d &p) {
2004-12-09 23:55:43 +00:00
if (p == _pos)
_walking = false;
else {
2004-12-09 23:55:43 +00:00
_walking = true;
_destPos = p;
_path.clear();
if (_constrain) {
2011-05-05 11:25:08 +02:00
g_grim->getCurrScene()->findClosestSector(p, NULL, &_destPos);
Common::List<PathNode *> openList;
Common::List<PathNode *> closedList;
PathNode *start = new PathNode;
start->parent = NULL;
start->pos = _pos;
start->dist = 0.f;
start->cost = 0.f;
openList.push_back(start);
2011-05-05 11:25:08 +02:00
g_grim->getCurrScene()->findClosestSector(_pos, &start->sect, NULL);
Common::List<Sector *> sectors;
2011-05-05 11:25:08 +02:00
for (int i = 0; i < g_grim->getCurrScene()->getSectorCount(); ++i) {
Sector *s = g_grim->getCurrScene()->getSectorBase(i);
2011-06-07 17:21:40 +02:00
int type = s->getType();
if ((type == Sector::WalkType || type == Sector::HotType || type == Sector::FunnelType) && s->isVisible()) {
sectors.push_back(s);
}
}
Sector *endSec = NULL;
2011-05-05 11:25:08 +02:00
g_grim->getCurrScene()->findClosestSector(_destPos, &endSec, NULL);
do {
PathNode *node = NULL;
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 == endSec) {
PathNode *n = closedList.back();
2011-04-11 11:57:00 +02:00
while (n) {
_path.push_back(n->pos);
n = n->parent;
}
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;
Common::List<Graphics::Line3d> bridges = sector->getBridgesTo(s);
if (bridges.empty())
continue; // The sectors are not adjacent.
Graphics::Vector3d closestPoint = s->getClosestPoint(_destPos);
Graphics::Vector3d best;
float bestDist = 1e6f;
Graphics::Line3d l(node->pos, closestPoint);
while (!bridges.empty()) {
Graphics::Line3d bridge = bridges.back();
Graphics::Vector3d pos;
if (!bridge.intersectLine2d(l, &pos)) {
pos = bridge.middle();
}
float dist = (pos - closestPoint).magnitude();
if (dist < bestDist) {
bestDist = dist;
best = pos;
}
bridges.pop_back();
}
PathNode *n = NULL;
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).magnitude();
if (newCost < n->cost) {
n->cost = newCost;
n->parent = node;
n->pos = best;
n->dist = (n->pos - _destPos).magnitude();
}
} else {
n = new PathNode;
n->parent = node;
n->sect = s;
n->pos = best;
n->dist = (n->pos - _destPos).magnitude();
n->cost = node->cost + (n->pos - node->pos).magnitude();
openList.push_back(n);
}
}
} while (!openList.empty());
2011-04-08 18:55:16 +02:00
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;
}
}
_path.push_front(_destPos);
}
}
bool Actor::isWalking() const {
2004-12-09 23:55:43 +00:00
return _walkedLast || _walkedCur || _walking;
}
bool Actor::isTurning() const {
2004-12-09 23:55:43 +00:00
if (_turning)
return true;
2004-12-09 23:55:43 +00:00
if (_lastTurnDir != 0 || _currTurnDir != 0)
return true;
2004-12-09 23:55:43 +00:00
return false;
}
2003-08-15 18:00:22 +00:00
void Actor::walkForward() {
2011-05-05 11:25:08 +02:00
float dist = g_grim->getPerSecond(_walkRate);
_walking = false;
float yaw_rad = _yaw * (LOCAL_PI / 180.f), pitch_rad = _pitch * (LOCAL_PI / 180.f);
2005-08-08 07:05:58 +00:00
//float yaw;
Graphics::Vector3d forwardVec(-sin(yaw_rad) * cos(pitch_rad),
cos(yaw_rad) * cos(pitch_rad), sin(pitch_rad));
if (_lastWasLeft)
if (_running)
costumeMarkerCallback(RightRun);
else
costumeMarkerCallback(RightWalk);
else
if (_running)
costumeMarkerCallback(LeftRun);
else
costumeMarkerCallback(LeftWalk);
if (! _constrain) {
_pos += forwardVec * dist;
2004-12-09 23:55:43 +00:00
_walkedCur = true;
return;
}
if (dist < 0) {
dist = -dist;
forwardVec = -forwardVec;
}
Sector *currSector = NULL, *prevSector = NULL;
Sector::ExitInfo ei;
2011-05-05 11:25:08 +02:00
g_grim->getCurrScene()->findClosestSector(_pos, &currSector, &_pos);
2008-09-10 11:16:57 +00:00
if (!currSector) { // Shouldn't happen...
_pos += forwardVec * dist;
_walkedCur = true;
return;
}
2008-09-10 11:16:57 +00:00
while (currSector) {
prevSector = currSector;
2011-05-05 12:04:09 +02:00
Graphics::Vector3d puckVec = currSector->getProjectionToPuckVector(forwardVec);
2009-10-17 12:48:23 +00:00
puckVec /= puckVec.magnitude();
currSector->getExitInfo(_pos, puckVec, &ei);
float exitDist = (ei.exitPoint - _pos).magnitude();
if (dist < exitDist) {
2009-10-17 12:48:23 +00:00
_pos += puckVec * dist;
2004-12-09 23:55:43 +00:00
_walkedCur = true;
return;
}
_pos = ei.exitPoint;
dist -= exitDist;
if (exitDist > 0.0001)
_walkedCur = true;
// Check for an adjacent sector which can continue
// the path
2011-05-05 11:25:08 +02:00
currSector = g_grim->getCurrScene()->findPointSector(ei.exitPoint + (float)0.0001 * puckVec, Sector::WalkType);
if (currSector == prevSector)
break;
}
ei.angleWithEdge *= (float)(180.f / LOCAL_PI);
int turnDir = 1;
if (ei.angleWithEdge > 90) {
ei.angleWithEdge = 180 - ei.angleWithEdge;
ei.edgeDir = -ei.edgeDir;
turnDir = -1;
}
if (ei.angleWithEdge > _reflectionAngle)
return;
2005-04-03 10:41:57 +00:00
ei.angleWithEdge += (float)0.1;
2011-05-05 11:25:08 +02:00
float turnAmt = g_grim->getPerSecond(_turnRate) * 5.;
if (turnAmt > ei.angleWithEdge)
turnAmt = ei.angleWithEdge;
setYaw(_yaw + turnAmt * turnDir);
}
Graphics::Vector3d Actor::getPuckVector() const {
float yaw_rad = _yaw * (LOCAL_PI / 180.f);
Graphics::Vector3d forwardVec(-sin(yaw_rad), cos(yaw_rad), 0);
2011-05-05 11:25:08 +02:00
Sector *sector = g_grim->getCurrScene()->findPointSector(_pos, Sector::WalkType);
2008-09-10 11:16:57 +00:00
if (!sector)
return forwardVec;
else
2011-05-05 12:04:09 +02:00
return sector->getProjectionToPuckVector(forwardVec);
}
void Actor::setRestChore(int chore, Costume *cost) {
if (_restCostume == cost && _restChore == chore)
return;
2004-12-09 23:55:43 +00:00
if (_restChore >= 0)
_restCostume->stopChore(_restChore);
_restCostume = cost;
_restChore = chore;
if (_restChore >= 0)
_restCostume->playChoreLooping(_restChore);
}
void Actor::setWalkChore(int chore, Costume *cost) {
if (_walkCostume == cost && _walkChore == chore)
return;
2004-12-09 23:55:43 +00:00
if (_walkChore >= 0 && _walkCostume->isChoring(_walkChore, false) >= 0) {
_walkCostume->fadeChoreOut(_walkChore, 150);
if (_restChore >= 0)
_restCostume->fadeChoreIn(_restChore, 150);
}
2004-12-09 23:55:43 +00:00
_walkCostume = cost;
_walkChore = chore;
}
void Actor::setTurnChores(int left_chore, int right_chore, Costume *cost) {
if (_turnCostume == cost && _leftTurnChore == left_chore &&
2009-05-09 17:47:28 +00:00
_rightTurnChore == right_chore)
return;
2004-12-09 23:55:43 +00:00
if (_leftTurnChore >= 0) {
_turnCostume->stopChore(_leftTurnChore);
_turnCostume->stopChore(_rightTurnChore);
}
2004-12-09 23:55:43 +00:00
_turnCostume = cost;
_leftTurnChore = left_chore;
_rightTurnChore = 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);
2004-12-09 23:55:43 +00:00
index--;
2004-12-09 23:55:43 +00:00
if (_talkCostume[index] == cost && _talkChore[index] == chore)
return;
2004-12-09 23:55:43 +00:00
if (_talkChore[index] >= 0)
_talkCostume[index]->stopChore(_talkChore[index]);
_talkCostume[index] = cost;
_talkChore[index] = chore;
}
void Actor::setMumbleChore(int chore, Costume *cost) {
2004-12-09 23:55:43 +00:00
if (_mumbleChore >= 0)
_mumbleCostume->stopChore(_mumbleChore);
_mumbleCostume = cost;
_mumbleChore = chore;
2003-08-15 18:00:22 +00:00
}
void Actor::turn(int dir) {
_walking = false;
2011-05-05 11:25:08 +02:00
float delta = g_grim->getPerSecond(_turnRate) * dir;
setYaw(_yaw + delta);
2004-12-09 23:55:43 +00:00
_currTurnDir = dir;
if (_lastWasLeft)
costumeMarkerCallback(RightTurn);
else
costumeMarkerCallback(LeftTurn);
2003-08-15 18:00:22 +00:00
}
float Actor::getAngleTo(const Actor &a) const {
float yaw_rad = _yaw * (LOCAL_PI / 180.f);
Graphics::Vector3d forwardVec(-sin(yaw_rad), cos(yaw_rad), 0);
Graphics::Vector3d delta = a.getPos() - _pos;
delta.z() = 0;
2004-12-09 23:55:43 +00:00
return angle(forwardVec, delta) * (180.f / LOCAL_PI);
2003-08-15 18:00:22 +00:00
}
float Actor::getYawTo(Graphics::Vector3d p) const {
Graphics::Vector3d dpos = p - _pos;
2004-12-09 23:55:43 +00:00
if (dpos.x() == 0 && dpos.y() == 0)
return 0;
else
return atan2(-dpos.x(), dpos.y()) * (180.f / LOCAL_PI);
}
void Actor::sayLine(const char *msg, const char *msgId, bool background) {
2005-01-02 13:34:50 +00:00
assert(msg);
assert(msgId);
if (msgId[0] == 0) {
error("Actor::sayLine: No message ID for text");
return;
}
2004-12-09 23:55:43 +00:00
// 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 = msgId;
soundName += ".wav";
if (_talkSoundName == soundName)
return;
if (g_imuse->getSoundStatus(_talkSoundName.c_str()) || msg[0] == 0)
shutUp();
_talkSoundName = soundName;
if (g_grim->getSpeechMode() != GrimEngine::TextOnly) {
if (g_imuse->startVoice(_talkSoundName.c_str()) && g_grim->getCurrScene()) {
g_grim->getCurrScene()->setSoundPosition(_talkSoundName.c_str(), _pos);
}
2005-01-05 18:28:50 +00:00
}
// If the actor is clearly not visible then don't try to play the lip sync
2011-07-18 17:51:58 +02:00
if (_visible && (!g_movie->isPlaying() || g_grim->getMode() == ENGINE_MODE_NORMAL)) {
Common::String soundLip = msgId;
soundLip += ".lip";
// 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 != -1)
_mumbleCostume->playChoreLooping(_mumbleChore);
_talkAnim = -1;
}
g_grim->setTalkingActor(this);
if (_sayLineText) {
TextObject *textObject = g_grim->getTextObject(_sayLineText);
if (textObject)
g_grim->killTextObject(textObject);
_sayLineText = 0;
}
if (msg[0] != 0) {
GrimEngine::SpeechMode m = g_grim->getSpeechMode();
if (!g_grim->_sayLineDefaults.getFont() || m == GrimEngine::VoiceOnly || background)
return;
2009-05-10 13:14:02 +00:00
TextObject *textObject = new TextObject(false, true);
textObject->setDefaults(&g_grim->_sayLineDefaults);
textObject->setFGColor(_talkColor);
if (g_grim->getMode() == ENGINE_MODE_SMUSH)
g_grim->killTextObjects();
if (m == GrimEngine::TextOnly || g_grim->getMode() == ENGINE_MODE_SMUSH) {
textObject->setDuration(500 + strlen(msg) * 15 * (11 - g_grim->getTextSpeed()));
}
if (g_grim->getMode() == ENGINE_MODE_SMUSH) {
textObject->setX(640 / 2);
textObject->setY(456);
2008-09-26 19:07:26 +00:00
} else {
if (_winX1 == 1000 || _winX2 == -1000 || _winY2 == -1000) {
textObject->setX(640 / 2);
textObject->setY(463);
} else {
textObject->setX((_winX1 + _winX2) / 2);
textObject->setY(_winY1);
}
2008-09-26 19:07:26 +00:00
}
textObject->setText(msg);
g_grim->registerTextObject(textObject);
if (g_grim->getMode() != ENGINE_MODE_SMUSH)
_sayLineText = textObject->getId();
}
}
2003-08-15 18:00:22 +00:00
bool Actor::isTalking() {
// If there's no sound file then we're obviously not talking
2011-05-23 17:58:40 +02:00
GrimEngine::SpeechMode m = g_grim->getSpeechMode();
TextObject *textObject = NULL;
if (_sayLineText)
textObject = g_grim->getTextObject(_sayLineText);
if ((m == GrimEngine::TextOnly && (!textObject || textObject->getDisabled())) ||
(m != GrimEngine::TextOnly && (strlen(_talkSoundName.c_str()) == 0 || !g_imuse->getSoundStatus(_talkSoundName.c_str())))) {
return false;
}
2009-05-09 17:47:28 +00:00
return true;
2003-08-15 18:00:22 +00:00
}
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_imuse->stopSound(_talkSoundName.c_str());
_talkSoundName = "";
}
2009-04-21 18:04:24 +00:00
if (_lipSync) {
2008-09-10 11:16:57 +00:00
if (_talkAnim != -1 && _talkChore[_talkAnim] >= 0)
_talkCostume[_talkAnim]->stopChore(_talkChore[_talkAnim]);
2009-04-21 18:04:24 +00:00
_lipSync = NULL;
} else if (_mumbleChore >= 0 && _mumbleCostume->isChoring(_mumbleChore, false) >= 0) {
_mumbleCostume->stopChore(_mumbleChore);
}
if (_talkChore[0] >= 0) {
// _talkChore[0] is *_stop_talk
_talkCostume[0]->playChore(_talkChore[0]);
}
2008-09-10 11:16:57 +00:00
if (_sayLineText) {
TextObject *textObject = g_grim->getTextObject(_sayLineText);
if (textObject)
g_grim->killTextObject(textObject);
_sayLineText = 0;
}
if (g_grim->getTalkingActor() == this) {
g_grim->setTalkingActor(NULL);
}
}
2009-10-17 12:48:23 +00:00
void Actor::pushCostume(const char *n) {
Costume *newCost = g_resourceloader->loadCostume(n, getCurrentCostume());
2009-05-09 17:47:28 +00:00
newCost->setColormap(NULL);
2004-12-09 23:55:43 +00:00
_costumeStack.push_back(newCost);
2003-08-15 18:00:22 +00:00
}
void Actor::setColormap(const char *map) {
2006-05-16 22:16:31 +00:00
if (!_costumeStack.empty()) {
Costume *cost = _costumeStack.back();
2009-10-17 13:25:12 +00:00
cost->setColormap(map);
2006-05-16 22:16:31 +00:00
} else {
warning("Actor::setColormap: No costumes");
}
}
2009-10-17 12:48:23 +00:00
void Actor::setCostume(const char *n) {
2004-12-09 23:55:43 +00:00
if (!_costumeStack.empty())
popCostume();
2004-12-09 23:55:43 +00:00
2009-10-17 12:48:23 +00:00
pushCostume(n);
2003-08-15 18:00:22 +00:00
}
void Actor::popCostume() {
2004-12-09 23:55:43 +00:00
if (!_costumeStack.empty()) {
freeCostumeChore(_costumeStack.back(), _restCostume, _restChore);
freeCostumeChore(_costumeStack.back(), _walkCostume, _walkChore);
2004-12-09 23:55:43 +00:00
if (_turnCostume == _costumeStack.back()) {
_turnCostume = NULL;
_leftTurnChore = -1;
_rightTurnChore = -1;
}
2004-12-09 23:55:43 +00:00
freeCostumeChore(_costumeStack.back(), _mumbleCostume, _mumbleChore);
for (int i = 0; i < 10; i++)
2004-12-09 23:55:43 +00:00
freeCostumeChore(_costumeStack.back(), _talkCostume[i], _talkChore[i]);
delete _costumeStack.back();
2004-12-09 23:55:43 +00:00
_costumeStack.pop_back();
2011-05-23 18:29:00 +02:00
if (_costumeStack.empty()) {
if (gDebugLevel == DEBUG_NORMAL || gDebugLevel == DEBUG_ALL)
printf("Popped (freed) the last costume for an actor.\n");
}
} else {
if (gDebugLevel == DEBUG_WARN || gDebugLevel == DEBUG_ALL)
warning("Attempted to pop (free) a costume when the stack is empty!");
}
2003-08-15 18:00:22 +00:00
}
void Actor::clearCostumes() {
// Make sure to destroy costume copies in reverse order
2004-12-09 23:55:43 +00:00
while (!_costumeStack.empty())
popCostume();
2003-08-15 18:00:22 +00:00
}
2009-05-12 19:53:37 +00:00
void Actor::setHead(int joint1, int joint2, int joint3, float maxRoll, float maxPitch, float maxYaw) {
2004-12-09 23:55:43 +00:00
if (!_costumeStack.empty()) {
_costumeStack.back()->setHead(joint1, joint2, joint3, maxRoll, maxPitch, maxYaw);
2003-08-26 09:36:47 +00:00
}
}
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;
2011-03-21 17:18:04 +01:00
}
2004-12-09 23:55:43 +00:00
return NULL;
2003-08-15 18:00:22 +00:00
}
void Actor::updateWalk() {
if (_path.empty()) {
return;
}
Graphics::Vector3d destPos = _path.back();
float y = getYawTo(destPos);
if (y < 0.f) {
y += 360.f;
}
turnTo(_pitch, y, _roll);
Graphics::Vector3d dir = destPos - _pos;
float dist = dir.magnitude();
if (dist > 0)
dir /= dist;
2011-05-05 11:25:08 +02:00
float walkAmt = g_grim->getPerSecond(_walkRate);
if (walkAmt >= dist) {
_pos = destPos;
_path.pop_back();
if (_path.empty()) {
_walking = false;
// 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;
}
} else {
_pos += dir * walkAmt;
}
_walkedCur = true;
if (_lastWasLeft)
if (_running)
costumeMarkerCallback(RightRun);
else
costumeMarkerCallback(RightWalk);
else
if (_running)
costumeMarkerCallback(LeftRun);
else
costumeMarkerCallback(LeftWalk);
}
2003-08-15 18:00:22 +00:00
void Actor::update() {
// Snap actor to walkboxes if following them. This might be
// necessary for example after activating/deactivating
// walkboxes, etc.
2004-12-09 23:55:43 +00:00
if (_constrain && !_walking) {
2011-05-05 11:25:08 +02:00
g_grim->getCurrScene()->findClosestSector(_pos, NULL, &_pos);
}
2004-12-09 23:55:43 +00:00
if (_turning) {
2011-05-05 11:25:08 +02:00
float turnAmt = g_grim->getPerSecond(_turnRate) * 5.f;
2004-12-09 23:55:43 +00:00
float dyaw = _destYaw - _yaw;
while (dyaw > 180)
dyaw -= 360;
while (dyaw < -180)
dyaw += 360;
// 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.
2009-05-17 09:34:28 +00:00
if (turnAmt == 0 || turnAmt >= fabs(dyaw)) {
setYaw(_destYaw);
2004-12-09 23:55:43 +00:00
_turning = false;
}
else if (dyaw > 0)
setYaw(_yaw + turnAmt);
else
setYaw(_yaw - turnAmt);
2004-12-09 23:55:43 +00:00
_currTurnDir = (dyaw > 0 ? 1 : -1);
if (_lastWasLeft)
costumeMarkerCallback(RightTurn);
else
costumeMarkerCallback(LeftTurn);
}
2004-12-09 23:55:43 +00:00
if (_walking) {
updateWalk();
}
2004-12-09 23:55:43 +00:00
if (_walkChore >= 0) {
if (_walkedCur) {
if (_walkCostume->isChoring(_walkChore, false) < 0) {
_lastStepTime = 0;
_walkCostume->stopChore(_walkChore);
2004-12-09 23:55:43 +00:00
_walkCostume->playChoreLooping(_walkChore);
_walkCostume->fadeChoreIn(_walkChore, 150);
}
if (_restChore >= 0) {
_restCostume->fadeChoreOut(_restChore, 150);
}
2004-12-09 23:55:43 +00:00
} else {
if (_walkedLast && _walkCostume->isChoring(_walkChore, false) >= 0) {
_walkCostume->fadeChoreOut(_walkChore, 150);
if (_restChore >= 0) {
_restCostume->fadeChoreIn(_restChore, 150);
}
}
}
}
2004-12-09 23:55:43 +00:00
if (_leftTurnChore >= 0) {
if (_walkedCur || _walkedLast)
2004-12-09 23:55:43 +00:00
_currTurnDir = 0;
if (_restChore >= 0) {
if (_currTurnDir != 0) {
if (_turnCostume->isChoring(getTurnChore(_currTurnDir), false) >= 0)
_restCostume->fadeChoreOut(_restChore, 500);
}
else if (_lastTurnDir != 0) {
if (!_walkedCur && _turnCostume->isChoring(getTurnChore(_lastTurnDir), false) >= 0)
_restCostume->fadeChoreIn(_restChore, 150);
}
}
2004-12-09 23:55:43 +00:00
if (_lastTurnDir != 0 && _lastTurnDir != _currTurnDir)
_turnCostume->fadeChoreOut(getTurnChore(_lastTurnDir), 150);
if (_currTurnDir != 0 && _currTurnDir != _lastTurnDir) {
_turnCostume->playChoreLooping(getTurnChore(_currTurnDir));
_turnCostume->fadeChoreIn(getTurnChore(_currTurnDir), 500);
}
2004-12-09 23:55:43 +00:00
} else
_currTurnDir = 0;
// The rest chore might have been stopped because of a
// StopActorChore(nil). Restart it if so.
if (_restChore >= 0 && _restCostume->isChoring(_restChore, false) < 0)
_restCostume->playChoreLooping(_restChore);
2004-12-09 23:55:43 +00:00
_walkedLast = _walkedCur;
_walkedCur = false;
_lastTurnDir = _currTurnDir;
_currTurnDir = 0;
2009-04-21 18:04:24 +00:00
// Update lip syncing
if (_lipSync) {
int posSound;
2009-05-09 17:47:28 +00:00
// While getPosIn60HzTicks 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_imuse->getSoundStatus(_talkSoundName.c_str()))
posSound = g_imuse->getPosIn60HzTicks(_talkSoundName.c_str());
else
posSound = -1;
2005-01-01 21:07:52 +00:00
if (posSound != -1) {
2009-04-21 18:04:24 +00:00
int anim = _lipSync->getAnim(posSound);
2005-01-01 21:07:52 +00:00
if (_talkAnim != anim) {
if (anim != -1) {
if (_talkChore[anim] >= 0) {
if (_talkAnim != -1 && _talkChore[_talkAnim] >= 0)
_talkCostume[_talkAnim]->stopChore(_talkChore[_talkAnim]);
_talkAnim = anim;
2005-01-01 21:07:52 +00:00
_talkCostume[_talkAnim]->playChoreLooping(_talkChore[_talkAnim]);
}
} else {
if (_talkAnim != -1 && _talkChore[_talkAnim] >= 0)
_talkCostume[_talkAnim]->stopChore(_talkChore[_talkAnim]);
_talkCostume[0]->playChore(_talkChore[0]);
2005-01-01 21:07:52 +00:00
}
}
}
2004-12-09 23:55:43 +00:00
}
for (Common::List<Costume *>::iterator i = _costumeStack.begin(); i != _costumeStack.end(); ++i) {
Costume *c = *i;
c->setPosRotate(_pos, _pitch, _yaw, _roll);
c->update();
}
for (Common::List<Costume *>::iterator i = _costumeStack.begin(); i != _costumeStack.end(); ++i) {
Costume *c = *i;
c->animate();
}
for (Common::List<Costume *>::iterator i = _costumeStack.begin(); i != _costumeStack.end(); ++i) {
Costume *c = *i;
c->moveHead(_lookingMode, _lookAtVector, _lookAtRate);
}
2003-08-15 18:00:22 +00:00
}
void Actor::draw() {
2008-09-26 17:48:46 +00:00
g_winX1 = g_winY1 = 1000;
g_winX2 = g_winY2 = -1000;
for (Common::List<Costume *>::iterator i = _costumeStack.begin(); i != _costumeStack.end(); ++i) {
Costume *c = *i;
c->setupTextures();
}
2004-03-23 13:05:08 +00:00
2009-05-17 08:24:17 +00:00
if (!g_driver->isHardwareAccelerated() && g_grim->getFlagRefreshShadowMask()) {
for (int l = 0; l < 5; l++) {
if (!_shadowArray[l].active)
continue;
g_driver->setShadow(&_shadowArray[l]);
g_driver->drawShadowPlanes();
g_driver->setShadow(NULL);
}
}
2004-12-09 23:55:43 +00:00
if (!_costumeStack.empty()) {
Costume *costume = _costumeStack.back();
if (!g_driver->isHardwareAccelerated()) {
for (int l = 0; l < 5; l++) {
if (!shouldDrawShadow(l))
continue;
g_driver->setShadow(&_shadowArray[l]);
g_driver->setShadowMode();
2011-04-28 22:07:36 +02:00
g_driver->startActorDraw(_pos, _scale, _yaw, _pitch, _roll);
costume->draw();
g_driver->finishActorDraw();
g_driver->clearShadowMode();
g_driver->setShadow(NULL);
}
// normal draw actor
2011-04-28 22:07:36 +02:00
g_driver->startActorDraw(_pos, _scale, _yaw, _pitch, _roll);
costume->draw();
g_driver->finishActorDraw();
} else {
// normal draw actor
2011-04-28 22:07:36 +02:00
g_driver->startActorDraw(_pos, _scale, _yaw, _pitch, _roll);
costume->draw();
g_driver->finishActorDraw();
for (int l = 0; l < 5; l++) {
if (!shouldDrawShadow(l))
continue;
g_driver->setShadow(&_shadowArray[l]);
g_driver->setShadowMode();
g_driver->drawShadowPlanes();
2011-04-28 22:07:36 +02:00
g_driver->startActorDraw(_pos, _scale, _yaw, _pitch, _roll);
costume->draw();
g_driver->finishActorDraw();
g_driver->clearShadowMode();
g_driver->setShadow(NULL);
}
}
}
2008-09-26 17:48:46 +00:00
_winX1 = g_winX1;
_winX2 = g_winX2;
_winY1 = g_winY1;
_winY2 = g_winY2;
2004-12-09 23:55:43 +00:00
}
// "Undraw objects" (handle objects for actors that may not be on screen)
2005-08-10 08:33:45 +00:00
void Actor::undraw(bool /*visible*/) {
2011-05-23 17:58:40 +02:00
if (!isTalking())
shutUp();
}
2009-10-17 12:48:23 +00:00
void Actor::setShadowPlane(const char *n) {
assert(_activeShadowSlot != -1);
2009-10-17 12:48:23 +00:00
_shadowArray[_activeShadowSlot].name = n;
}
void Actor::addShadowPlane(const char *n, Scene *scene, int shadowId) {
assert(shadowId != -1);
int numSectors = scene->getSectorCount();
for (int i = 0; i < numSectors; i++) {
// Create a copy so we are sure it will not be deleted by the Scene 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.
Sector *sector = scene->getSectorBase(i);
if (!strcmp(sector->getName(), n)) {
Plane p = { scene->getName(), new Sector(*sector) };
_shadowArray[shadowId].planeList.push_back(p);
2009-05-17 08:24:17 +00:00
g_grim->flagRefreshShadowMask(true);
return;
}
}
}
bool Actor::shouldDrawShadow(int shadowId) {
Shadow *shadow = &_shadowArray[shadowId];
if (!shadow->active)
return false;
// Don't draw a shadow if the actor is behind the shadow plane.
Sector *sector = shadow->planeList.front().sector;
Graphics::Vector3d n = sector->getNormal();
Graphics::Vector3d p = sector->getVertices()[0];
float d = -(n.x() * p.x() + n.y() * p.y() + n.z() * p.z());
p = getPos();
if (n.x() * p.x() + n.y() * p.y() + n.z() * p.z() + d < 0.f)
return true;
else
return false;
}
void Actor::addShadowPlane(const char *n) {
addShadowPlane(n, g_grim->getCurrScene(), _activeShadowSlot);
}
void Actor::setActiveShadow(int shadowId) {
assert(shadowId >= 0 && shadowId <= 4);
_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 <= 4);
_shadowArray[shadowId].active = state;
}
2009-10-17 12:48:23 +00:00
void Actor::setShadowPoint(Graphics::Vector3d p) {
assert(_activeShadowSlot != -1);
2009-10-17 12:48:23 +00:00
_shadowArray[_activeShadowSlot].pos = p;
}
void Actor::clearShadowPlanes() {
for (int i = 0; i < 5; i++) {
Shadow *shadow = &_shadowArray[i];
while (!shadow->planeList.empty()) {
delete shadow->planeList.back().sector;
shadow->planeList.pop_back();
}
delete[] shadow->shadowMask;
shadow->shadowMaskSize = 0;
shadow->shadowMask = NULL;
shadow->active = false;
shadow->dontNegate = false;
}
}
2007-03-12 21:50:44 +00:00
bool Actor::isInSet(const Common::String &setName) const {
return _setName == setName;
}
void Actor::freeCostumeChore(Costume *toFree, Costume *&cost, int &chore) {
if (cost == toFree) {
cost = NULL;
chore = -1;
}
}
extern int refSystemTable;
void Actor::costumeMarkerCallback(Footstep step)
{
int time = g_system->getMillis();
float rate = 400;
if (_running)
rate = 300;
if (_lastStepTime != 0 && time - _lastStepTime < rate)
return;
_lastStepTime = time;
_lastWasLeft = !_lastWasLeft;
lua_beginblock();
lua_pushobject(lua_getref(refSystemTable));
lua_pushstring("costumeMarkerHandler");
lua_Object table = lua_gettable();
if (lua_istable(table)) {
lua_pushobject(table);
lua_pushstring("costumeMarkerHandler");
lua_Object func = lua_gettable();
if (lua_isfunction(func)) {
lua_pushobject(func);
lua_pushusertag(getId(), MKTAG('A','C','T','R'));
lua_pushnumber(step);
lua_callfunction(func);
}
} else if (lua_isfunction(table)) {
lua_pushusertag(getId(), MKTAG('A','C','T','R'));
lua_pushnumber(step);
lua_callfunction(table);
}
lua_endblock();
}
2009-05-25 06:49:57 +00:00
} // end of namespace Grim