2004-04-12 21:40:49 +00:00
|
|
|
/* ScummVM - Scumm Interpreter
|
2005-01-01 16:20:17 +00:00
|
|
|
* Copyright (C) 2004-2005 The ScummVM project
|
2004-04-12 21:40:49 +00:00
|
|
|
*
|
|
|
|
* The ReInherit Engine is (C)2000-2003 by Daniel Balsom.
|
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU General Public License
|
|
|
|
* as published by the Free Software Foundation; either version 2
|
|
|
|
* of the License, or (at your option) any later version.
|
|
|
|
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program; if not, write to the Free Software
|
|
|
|
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
|
|
|
|
*
|
|
|
|
* $Header$
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
2004-08-02 16:20:35 +00:00
|
|
|
#include "saga/saga.h"
|
|
|
|
#include "saga/gfx.h"
|
2004-04-12 21:40:49 +00:00
|
|
|
|
2004-08-10 18:31:33 +00:00
|
|
|
#include "saga/console.h"
|
2004-08-02 16:20:35 +00:00
|
|
|
#include "saga/rscfile_mod.h"
|
2004-08-12 23:57:45 +00:00
|
|
|
#include "saga/script.h"
|
2004-08-02 16:20:35 +00:00
|
|
|
#include "saga/sndres.h"
|
2004-08-03 01:07:34 +00:00
|
|
|
#include "saga/sprite.h"
|
2004-08-03 00:06:18 +00:00
|
|
|
#include "saga/font.h"
|
2004-08-03 00:40:16 +00:00
|
|
|
#include "saga/text.h"
|
2004-08-02 16:20:35 +00:00
|
|
|
#include "saga/sound.h"
|
2004-10-08 01:22:39 +00:00
|
|
|
#include "saga/scene.h"
|
|
|
|
#include "saga/actionmap.h"
|
2004-04-12 21:40:49 +00:00
|
|
|
|
2004-08-02 16:20:35 +00:00
|
|
|
#include "saga/actor.h"
|
|
|
|
#include "saga/actordata.h"
|
2004-12-15 00:24:12 +00:00
|
|
|
#include "saga/stream.h"
|
2004-12-19 13:38:11 +00:00
|
|
|
#include "saga/interface.h"
|
2004-12-21 06:49:07 +00:00
|
|
|
#include "common/config-manager.h"
|
2004-04-12 21:40:49 +00:00
|
|
|
|
|
|
|
namespace Saga {
|
|
|
|
|
2004-12-17 20:38:17 +00:00
|
|
|
static int actorCompare(const ActorDataPointer& actor1, const ActorDataPointer& actor2) {
|
2004-12-22 21:04:50 +00:00
|
|
|
if (actor1->location.y == actor2->location.y) {
|
2004-12-15 00:24:12 +00:00
|
|
|
return 0;
|
2004-12-22 21:04:50 +00:00
|
|
|
} else if (actor1->location.y < actor2->location.y) {
|
2004-12-15 00:24:12 +00:00
|
|
|
return -1;
|
|
|
|
} else {
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
}
|
2004-08-02 15:47:42 +00:00
|
|
|
|
2004-12-22 21:04:50 +00:00
|
|
|
// Lookup table to convert 8 cardinal directions to 4
|
2004-12-28 21:27:18 +00:00
|
|
|
int actorDirectectionsLUT[8] = {
|
2004-12-22 21:04:50 +00:00
|
|
|
ACTOR_DIRECTION_BACK, // kDirUp
|
2004-12-29 21:49:25 +00:00
|
|
|
ACTOR_DIRECTION_RIGHT, // kDirUpRight
|
2004-12-22 21:04:50 +00:00
|
|
|
ACTOR_DIRECTION_RIGHT, // kDirRight
|
|
|
|
ACTOR_DIRECTION_RIGHT, // kDirDownRight
|
|
|
|
ACTOR_DIRECTION_FORWARD,// kDirDown
|
|
|
|
ACTOR_DIRECTION_LEFT, // kDirDownLeft
|
|
|
|
ACTOR_DIRECTION_LEFT, // kDirLeft
|
|
|
|
ACTOR_DIRECTION_LEFT, // kDirUpLeft
|
|
|
|
};
|
|
|
|
|
2005-01-06 14:32:14 +00:00
|
|
|
PathDirectionData pathDirectionLUT[8][3] = {
|
2004-12-28 21:27:18 +00:00
|
|
|
{{0, 0, -1}, {7, -1, -1}, {4, 1, -1}},
|
|
|
|
{{1, 1, 0}, {4, 1, -1}, {5, 1, 1}},
|
|
|
|
{{2, 0, 1}, {5, 1, 1}, {6, -1, 1}},
|
|
|
|
{{3, -1, 0}, {6, -1, 1}, {7, -1, -1}},
|
|
|
|
{{0, 0, -1}, {1, 1, 0}, {4, 1, -1}},
|
|
|
|
{{1, 1, 0}, {2, 0, 1}, {5, 1, 1}},
|
|
|
|
{{2, 0, 1}, {3, -1, 0}, {6, -1, 1}},
|
|
|
|
{{3, -1, 0}, {0, 0, -1}, {7, -1, -1}}
|
|
|
|
};
|
|
|
|
|
2004-12-29 21:49:25 +00:00
|
|
|
int pathDirectionLUT2[8][2] = {
|
2004-12-30 14:54:08 +00:00
|
|
|
{ 0, -1},
|
|
|
|
{ 1, 0},
|
|
|
|
{ 0, 1},
|
|
|
|
{-1, 0},
|
|
|
|
{ 1, -1},
|
|
|
|
{ 1, 1},
|
|
|
|
{-1, 1},
|
|
|
|
{-1, -1}
|
2004-12-29 21:49:25 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
int angleLUT[16][2] = {
|
2004-12-30 14:54:08 +00:00
|
|
|
{ 0, -256},
|
|
|
|
{ 98, -237},
|
|
|
|
{ 181, -181},
|
|
|
|
{ 237, -98},
|
|
|
|
{ 256, 0},
|
|
|
|
{ 237, 98},
|
|
|
|
{ 181, 181},
|
|
|
|
{ 98, 237},
|
|
|
|
{ 0, 256},
|
|
|
|
{ -98, 237},
|
|
|
|
{-181, 181},
|
|
|
|
{-237, 98},
|
|
|
|
{-256, 0},
|
|
|
|
{-237, -98},
|
|
|
|
{-181, -181},
|
|
|
|
{ -98, -237}
|
2004-12-29 21:49:25 +00:00
|
|
|
};
|
|
|
|
|
2005-01-04 16:10:43 +00:00
|
|
|
int directionLUT[8][2] = {
|
|
|
|
{ 0*2, -2*2},
|
|
|
|
{ 2*2, -1*2},
|
|
|
|
{ 3*2, 0*2},
|
|
|
|
{ 2*2, 1*2},
|
|
|
|
{ 0*2, 2*2},
|
|
|
|
{-2*2, 1*2},
|
|
|
|
{-4*2, 0*2},
|
|
|
|
{-2*2, -1*2}
|
|
|
|
};
|
|
|
|
|
|
|
|
int tileDirectionLUT[8][2] = {
|
|
|
|
{ 1, 1},
|
|
|
|
{ 2, 0},
|
|
|
|
{ 1, -1},
|
|
|
|
{ 0, -2},
|
|
|
|
{-1, -1},
|
|
|
|
{-2, 0},
|
|
|
|
{-1, 1},
|
|
|
|
{ 0, 2}
|
|
|
|
};
|
2004-12-28 21:27:18 +00:00
|
|
|
|
2004-12-17 11:18:56 +00:00
|
|
|
Actor::Actor(SagaEngine *vm) : _vm(vm) {
|
2004-04-12 21:40:49 +00:00
|
|
|
int i;
|
2004-12-17 20:38:17 +00:00
|
|
|
ActorData *actor;
|
2004-12-21 06:49:07 +00:00
|
|
|
debug(9, "Actor::Actor()");
|
2004-04-12 21:40:49 +00:00
|
|
|
|
2004-12-28 23:18:11 +00:00
|
|
|
if (_vm->_gameType == GType_IHNM) {
|
|
|
|
warning("Actors aren't implemented for IHNM yet");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2005-01-06 19:15:01 +00:00
|
|
|
_debugPoints = NULL;
|
|
|
|
_debugPointsAlloced = _debugPointsCount = 0;
|
|
|
|
|
2005-01-06 21:57:10 +00:00
|
|
|
_pathNodeList = _newPathNodeList = NULL;
|
|
|
|
_pathList = NULL;
|
|
|
|
_pathListAlloced = _pathNodeListAlloced = 0;
|
|
|
|
_pathListIndex = _pathNodeListIndex = -1;
|
|
|
|
|
|
|
|
|
2004-12-22 21:04:50 +00:00
|
|
|
_centerActor = _protagonist = NULL;
|
2004-12-24 12:47:16 +00:00
|
|
|
_lastTickMsec = 0;
|
2004-12-28 21:27:18 +00:00
|
|
|
|
2005-01-05 19:03:49 +00:00
|
|
|
_yCellCount = _vm->getStatusYOffset() + 1;
|
|
|
|
_xCellCount = _vm->getDisplayWidth();
|
2004-12-28 21:27:18 +00:00
|
|
|
|
2005-01-06 21:57:10 +00:00
|
|
|
_pathCell = (int*) malloc(_yCellCount * _xCellCount * sizeof(*_pathCell));
|
2004-12-28 21:27:18 +00:00
|
|
|
|
2004-12-29 21:49:25 +00:00
|
|
|
|
2004-12-28 21:27:18 +00:00
|
|
|
_pathRect.left = 0;
|
2005-01-05 19:03:49 +00:00
|
|
|
_pathRect.right = _vm->getDisplayWidth() - 1;
|
2004-12-28 21:27:18 +00:00
|
|
|
_pathRect.top = _vm->getPathYOffset();
|
2005-01-04 18:54:29 +00:00
|
|
|
_pathRect.bottom = _vm->getStatusYOffset();
|
2004-12-28 21:27:18 +00:00
|
|
|
|
2004-05-01 06:16:57 +00:00
|
|
|
// Get actor resource file context
|
2004-12-22 13:09:47 +00:00
|
|
|
_actorContext = _vm->getFileContext(GAME_RESOURCEFILE, 0);
|
2004-11-15 03:03:48 +00:00
|
|
|
if (_actorContext == NULL) {
|
2004-08-02 15:47:42 +00:00
|
|
|
error("Actor::Actor(): Couldn't load actor module resource context.");
|
2004-04-12 21:40:49 +00:00
|
|
|
}
|
|
|
|
|
2004-10-27 21:32:28 +00:00
|
|
|
for (i = 0; i < ACTORCOUNT; i++) {
|
2004-12-17 20:38:17 +00:00
|
|
|
actor = &_actors[i];
|
2004-12-18 20:33:02 +00:00
|
|
|
actor->actorId = ACTOR_INDEX_TO_ID(i);
|
|
|
|
actor->index = i;
|
2004-12-30 18:45:48 +00:00
|
|
|
debug(9, "init actorId=%d index=%d", actor->actorId, actor->index);
|
2004-12-18 20:33:02 +00:00
|
|
|
actor->nameIndex = ActorTable[i].nameIndex;
|
2004-12-17 20:38:17 +00:00
|
|
|
actor->spriteListResourceId = ActorTable[i].spriteListResourceId;
|
|
|
|
actor->frameListResourceId = ActorTable[i].frameListResourceId;
|
2004-12-18 16:08:59 +00:00
|
|
|
actor->speechColor = ActorTable[i].speechColor;
|
2004-12-18 20:33:02 +00:00
|
|
|
actor->sceneNumber = ActorTable[i].sceneIndex;
|
|
|
|
actor->flags = ActorTable[i].flags;
|
2004-12-22 21:04:50 +00:00
|
|
|
actor->currentAction = ActorTable[i].currentAction;
|
|
|
|
actor->facingDirection = ActorTable[i].facingDirection;
|
|
|
|
actor->actionDirection = ActorTable[i].actionDirection;
|
|
|
|
actor->frameNumber = 0;
|
|
|
|
actor->targetObject = ID_NOTHING;
|
2004-12-29 22:31:21 +00:00
|
|
|
actor->actorFlags = 0;
|
2004-12-22 21:04:50 +00:00
|
|
|
|
|
|
|
actor->location.x = ActorTable[i].x;
|
|
|
|
actor->location.y = ActorTable[i].y;
|
|
|
|
actor->location.z = ActorTable[i].z;
|
2004-12-18 16:08:59 +00:00
|
|
|
|
2004-12-18 20:33:02 +00:00
|
|
|
actor->disabled = !loadActorResources(actor);
|
|
|
|
if (actor->disabled) {
|
2004-12-30 18:45:48 +00:00
|
|
|
warning("Disabling actorId=%d index=%d", actor->actorId, actor->index);
|
2004-12-18 20:33:02 +00:00
|
|
|
}
|
2004-04-12 21:40:49 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2004-08-02 15:47:42 +00:00
|
|
|
Actor::~Actor() {
|
2004-12-17 20:38:17 +00:00
|
|
|
int i;
|
|
|
|
ActorData *actor;
|
|
|
|
|
2004-12-21 06:49:07 +00:00
|
|
|
debug(9, "Actor::~Actor()");
|
2005-01-06 19:15:01 +00:00
|
|
|
free(_debugPoints);
|
2005-01-06 21:57:10 +00:00
|
|
|
free(_pathNodeList);
|
|
|
|
free(_newPathNodeList);
|
|
|
|
free(_pathList);
|
2004-12-28 21:27:18 +00:00
|
|
|
free(_pathCell);
|
2004-12-17 20:38:17 +00:00
|
|
|
//release resources
|
|
|
|
for (i = 0; i < ACTORCOUNT; i++) {
|
|
|
|
actor = &_actors[i];
|
|
|
|
free(actor->frames);
|
|
|
|
_vm->_sprite->freeSprite(actor->spriteList);
|
|
|
|
}
|
2004-12-17 11:18:56 +00:00
|
|
|
}
|
|
|
|
|
2004-12-30 18:45:48 +00:00
|
|
|
bool Actor::loadActorResources(ActorData *actor) {
|
2004-12-17 20:38:17 +00:00
|
|
|
byte *resourcePointer;
|
|
|
|
size_t resourceLength;
|
2004-12-18 20:33:02 +00:00
|
|
|
int framesCount;
|
2004-12-22 21:04:50 +00:00
|
|
|
ActorFrameSequence *framesPointer;
|
2004-12-17 20:38:17 +00:00
|
|
|
int lastFrame;
|
|
|
|
int i, orient;
|
|
|
|
|
2004-12-30 18:45:48 +00:00
|
|
|
if (actor->frameListResourceId == 0) {
|
|
|
|
warning("Frame List ID = 0 for actor index %d", actor->index);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
debug(9, "Loading frame resource id %d", actor->frameListResourceId);
|
|
|
|
if (RSC_LoadResource(_actorContext, actor->frameListResourceId, &resourcePointer, &resourceLength) != SUCCESS) {
|
2004-12-17 20:38:17 +00:00
|
|
|
warning("Couldn't load sprite action index resource");
|
2004-12-18 16:08:59 +00:00
|
|
|
return false;
|
2004-12-17 20:38:17 +00:00
|
|
|
}
|
|
|
|
|
2004-12-18 20:33:02 +00:00
|
|
|
framesCount = resourceLength / 16;
|
2004-12-21 06:49:07 +00:00
|
|
|
debug(9, "Frame resource contains %d frames", framesCount);
|
2004-12-17 11:18:56 +00:00
|
|
|
|
2004-12-22 21:04:50 +00:00
|
|
|
framesPointer = (ActorFrameSequence *)malloc(sizeof(ActorFrameSequence) * framesCount);
|
2004-12-17 20:38:17 +00:00
|
|
|
if (framesPointer == NULL) {
|
2004-12-18 16:08:59 +00:00
|
|
|
error("Couldn't allocate memory for sprite frames");
|
2004-12-17 20:38:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
MemoryReadStreamEndian readS(resourcePointer, resourceLength, IS_BIG_ENDIAN);
|
|
|
|
|
|
|
|
lastFrame = 0;
|
|
|
|
|
2004-12-18 20:33:02 +00:00
|
|
|
for (i = 0; i < framesCount; i++) {
|
2004-12-22 21:04:50 +00:00
|
|
|
for (orient = 0; orient < ACTOR_DIRECTIONS_COUNT; orient++) {
|
2004-12-17 20:38:17 +00:00
|
|
|
// Load all four orientations
|
2004-12-22 21:04:50 +00:00
|
|
|
framesPointer[i].directions[orient].frameIndex = readS.readUint16();
|
2004-12-24 10:22:01 +00:00
|
|
|
framesPointer[i].directions[orient].frameCount = readS.readSint16();
|
|
|
|
if (framesPointer[i].directions[orient].frameCount < 0)
|
|
|
|
warning("frameCount < 0", framesPointer[i].directions[orient].frameCount);
|
2004-12-22 21:04:50 +00:00
|
|
|
if (framesPointer[i].directions[orient].frameIndex > lastFrame) {
|
|
|
|
lastFrame = framesPointer[i].directions[orient].frameIndex;
|
2004-12-17 20:38:17 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
RSC_FreeResource(resourcePointer);
|
|
|
|
|
|
|
|
actor->frames = framesPointer;
|
2004-12-18 20:33:02 +00:00
|
|
|
actor->framesCount = framesCount;
|
2004-12-17 20:38:17 +00:00
|
|
|
|
|
|
|
|
2004-12-30 18:45:48 +00:00
|
|
|
debug(9, "Loading sprite resource id %d", actor->spriteListResourceId);
|
2004-12-17 20:38:17 +00:00
|
|
|
if (_vm->_sprite->loadList(actor->spriteListResourceId, &actor->spriteList) != SUCCESS) {
|
|
|
|
warning("Unable to load sprite list");
|
2004-12-18 16:08:59 +00:00
|
|
|
return false;
|
2004-12-17 20:38:17 +00:00
|
|
|
}
|
|
|
|
|
2004-12-18 16:08:59 +00:00
|
|
|
i = _vm->_sprite->getListLen(actor->spriteList);
|
|
|
|
|
2004-12-28 21:27:18 +00:00
|
|
|
if (lastFrame >= i) {
|
2004-12-30 18:45:48 +00:00
|
|
|
debug(9, "Appending to sprite list %d (+ %d)", actor->spriteListResourceId, lastFrame);
|
2004-12-17 20:38:17 +00:00
|
|
|
if (_vm->_sprite->appendList(actor->spriteListResourceId + 1, actor->spriteList) != SUCCESS) {
|
|
|
|
warning("Unable append sprite list");
|
2004-12-18 16:08:59 +00:00
|
|
|
return false;
|
2004-12-17 20:38:17 +00:00
|
|
|
}
|
|
|
|
}
|
2004-12-17 11:18:56 +00:00
|
|
|
|
2004-12-18 16:08:59 +00:00
|
|
|
return true;
|
2004-12-17 11:18:56 +00:00
|
|
|
}
|
|
|
|
|
2004-12-29 21:49:25 +00:00
|
|
|
void Actor::realLocation(ActorLocation &location, uint16 objectId, uint16 walkFlags) {
|
|
|
|
int angle;
|
|
|
|
int distance;
|
|
|
|
ActorData *actor;
|
|
|
|
if (walkFlags & kWalkUseAngle) {
|
|
|
|
// tiled stuff
|
|
|
|
if (_vm->_scene->getMode() == SCENE_MODE_ISO) {
|
|
|
|
//todo: it
|
|
|
|
} else {
|
|
|
|
angle = location.x & 15;
|
|
|
|
distance = location.y;
|
|
|
|
|
|
|
|
location.x = (angleLUT[angle][0] * distance) >> 6; //fixme - call real angle calc
|
|
|
|
location.y = (angleLUT[angle][1] * distance) >> 6;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (objectId != ID_NOTHING) {
|
|
|
|
if (IS_VALID_ACTOR_ID(objectId)) {
|
|
|
|
actor = getActor(objectId);
|
2005-01-05 19:03:49 +00:00
|
|
|
location.add( actor->location);
|
2004-12-29 21:49:25 +00:00
|
|
|
} else {
|
2005-01-04 16:10:43 +00:00
|
|
|
warning("ObjectId unsupported"); //todo: do it
|
2004-12-29 21:49:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2005-01-06 19:15:01 +00:00
|
|
|
void Actor::actorFaceTowardsPoint(uint16 actorId, const ActorLocation &toLocation) {
|
|
|
|
ActorData *actor;
|
|
|
|
ActorLocation delta;
|
|
|
|
|
|
|
|
actor = getActor(actorId);
|
|
|
|
|
|
|
|
// tiled stuff
|
|
|
|
if (_vm->_scene->getMode() == SCENE_MODE_ISO) {
|
|
|
|
//todo: it
|
|
|
|
} else {
|
|
|
|
toLocation.delta(actor->location, delta);
|
|
|
|
|
|
|
|
if (ABS(delta.y) > ABS(delta.x * 2)) {
|
|
|
|
actor->facingDirection = (delta.y > 0) ? kDirDown : kDirUp;
|
|
|
|
} else {
|
|
|
|
actor->facingDirection = (delta.x > 0) ? kDirRight : kDirLeft;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Actor::actorFaceTowardsObject(uint16 actorId, uint16 objectId) {
|
|
|
|
ActorData *actor;
|
|
|
|
|
|
|
|
if (IS_VALID_ACTOR_ID(objectId)) {
|
|
|
|
actor = getActor(objectId);
|
|
|
|
actorFaceTowardsPoint(actorId, actor->location);
|
|
|
|
} else {
|
|
|
|
warning("ObjectId unsupported"); //todo: do it
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2004-12-17 20:38:17 +00:00
|
|
|
ActorData *Actor::getActor(uint16 actorId) {
|
2004-12-18 16:08:59 +00:00
|
|
|
ActorData *actor;
|
|
|
|
|
|
|
|
if (!IS_VALID_ACTOR_ID(actorId))
|
2004-12-17 20:38:17 +00:00
|
|
|
error("Actor::getActor Wrong actorId 0x%X", actorId);
|
|
|
|
|
2004-12-24 14:21:47 +00:00
|
|
|
if (actorId == ID_PROTAG) {
|
|
|
|
if (_protagonist == NULL) {
|
|
|
|
error("_protagonist == NULL");
|
|
|
|
}
|
|
|
|
return _protagonist;
|
|
|
|
}
|
|
|
|
|
2004-12-18 16:08:59 +00:00
|
|
|
actor = &_actors[ACTOR_ID_TO_INDEX(actorId)];
|
|
|
|
|
2004-12-18 20:33:02 +00:00
|
|
|
if (actor->disabled)
|
2004-12-18 16:08:59 +00:00
|
|
|
error("Actor::getActor disabled actorId 0x%X", actorId);
|
|
|
|
|
|
|
|
return actor;
|
2004-12-17 20:38:17 +00:00
|
|
|
}
|
2004-12-17 11:18:56 +00:00
|
|
|
|
2005-01-04 16:10:43 +00:00
|
|
|
bool Actor::validFollowerLocation(const ActorLocation &location) {
|
|
|
|
Point point;
|
2005-01-05 19:03:49 +00:00
|
|
|
location.toScreenPointXY(point);
|
2005-01-04 16:10:43 +00:00
|
|
|
|
|
|
|
if ((point.x < 5) || (point.x >= _vm->getDisplayWidth() - 5) ||
|
|
|
|
(point.y < 0) || (point.y >= _vm->getStatusYOffset())) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return (_vm->_scene->canWalk(point));
|
|
|
|
}
|
|
|
|
|
2004-12-18 20:33:02 +00:00
|
|
|
void Actor::updateActorsScene() {
|
2005-01-04 16:10:43 +00:00
|
|
|
int i, j;
|
|
|
|
int followerDirection;
|
2004-12-18 20:33:02 +00:00
|
|
|
ActorData *actor;
|
2005-01-04 16:10:43 +00:00
|
|
|
ActorLocation tempLocation;
|
|
|
|
ActorLocation possibleLocation;
|
|
|
|
Point delta;
|
2004-12-22 21:04:50 +00:00
|
|
|
|
|
|
|
_activeSpeech.stringsCount = 0;
|
2005-01-04 16:10:43 +00:00
|
|
|
_protagonist = NULL;
|
2004-12-17 20:38:17 +00:00
|
|
|
|
2004-12-18 20:33:02 +00:00
|
|
|
for (i = 0; i < ACTORCOUNT; i++) {
|
2004-12-22 21:04:50 +00:00
|
|
|
actor = &_actors[i];
|
2004-12-18 20:33:02 +00:00
|
|
|
if (actor->flags & (kProtagonist | kFollower)) {
|
|
|
|
actor->sceneNumber = _vm->_scene->currentSceneNumber();
|
2004-12-22 21:04:50 +00:00
|
|
|
if (actor->flags & kProtagonist) {
|
2005-01-04 16:10:43 +00:00
|
|
|
actor->finalTarget = actor->location;
|
2004-12-22 21:04:50 +00:00
|
|
|
_centerActor = _protagonist = actor;
|
|
|
|
}
|
2004-12-17 20:38:17 +00:00
|
|
|
}
|
2005-01-04 16:10:43 +00:00
|
|
|
if (actor->sceneNumber == _vm->_scene->currentSceneNumber()) {
|
2004-12-22 21:04:50 +00:00
|
|
|
actor->actionCycle = (rand() & 7) * 4;
|
2005-01-04 16:10:43 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
assert(_protagonist);
|
|
|
|
|
|
|
|
/* setup protagonist entry
|
|
|
|
// tiled stuff
|
|
|
|
if (_vm->_scene->getMode() == SCENE_MODE_ISO) {
|
|
|
|
//todo: it
|
|
|
|
} else {
|
|
|
|
}
|
|
|
|
*/
|
|
|
|
_protagonist->currentAction = kActionWait;
|
|
|
|
|
|
|
|
if (_vm->_scene->getMode() == SCENE_MODE_ISO) {
|
|
|
|
//todo: it
|
|
|
|
} else {
|
|
|
|
_vm->_scene->initDoorsState();
|
|
|
|
}
|
|
|
|
|
|
|
|
followerDirection = _protagonist->facingDirection + 3;
|
|
|
|
calcActorScreenPosition(_protagonist);
|
|
|
|
|
|
|
|
for (i = 0; i < ACTORCOUNT; i++) {
|
|
|
|
actor = &_actors[i];
|
|
|
|
if (actor->flags & (kFollower)) {
|
|
|
|
actor->facingDirection = actor->actionDirection = _protagonist->facingDirection;
|
|
|
|
actor->currentAction = kActionWait;
|
|
|
|
actor->walkStepsCount = actor->walkStepIndex = 0;
|
|
|
|
actor->location.z = _protagonist->location.z;
|
|
|
|
|
|
|
|
|
|
|
|
if (_vm->_scene->getMode() == SCENE_MODE_ISO) {
|
|
|
|
//todo: it
|
|
|
|
} else {
|
|
|
|
followerDirection &= 0x07;
|
|
|
|
|
|
|
|
possibleLocation = _protagonist->location;
|
|
|
|
|
|
|
|
|
|
|
|
delta.x = directionLUT[followerDirection][0];
|
|
|
|
delta.y = directionLUT[followerDirection][1];
|
|
|
|
|
|
|
|
|
|
|
|
for (j = 0; j < 30; j++) {
|
|
|
|
tempLocation = possibleLocation;
|
|
|
|
tempLocation.x += delta.x;
|
|
|
|
tempLocation.y += delta.y;
|
|
|
|
|
|
|
|
if (validFollowerLocation( tempLocation)) {
|
|
|
|
possibleLocation = tempLocation;
|
|
|
|
} else {
|
|
|
|
tempLocation = possibleLocation;
|
|
|
|
tempLocation.x += delta.x;
|
|
|
|
if (validFollowerLocation( tempLocation)) {
|
|
|
|
possibleLocation = tempLocation;
|
|
|
|
} else {
|
|
|
|
tempLocation = possibleLocation;
|
|
|
|
tempLocation.y += delta.y;
|
|
|
|
if (validFollowerLocation( tempLocation)) {
|
|
|
|
possibleLocation = tempLocation;
|
|
|
|
} else {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
actor->location = possibleLocation;
|
|
|
|
}
|
|
|
|
followerDirection += 2;
|
|
|
|
}
|
|
|
|
|
2004-12-18 20:33:02 +00:00
|
|
|
}
|
2004-12-22 21:04:50 +00:00
|
|
|
|
|
|
|
handleActions(0, true);
|
|
|
|
}
|
|
|
|
|
|
|
|
ActorFrameRange *Actor::getActorFrameRange(uint16 actorId, int frameType) {
|
|
|
|
ActorData *actor;
|
2004-12-24 10:22:01 +00:00
|
|
|
int fourDirection;
|
2004-12-22 21:04:50 +00:00
|
|
|
|
|
|
|
actor = getActor(actorId);
|
|
|
|
if (actor->disabled)
|
|
|
|
error("Actor::getActorFrameRange Wrong actorId 0x%X", actorId);
|
|
|
|
|
|
|
|
if (frameType >= actor->framesCount)
|
|
|
|
error("Actor::getActorFrameRange Wrong frameType 0x%X actorId 0x%X", frameType, actorId);
|
|
|
|
|
|
|
|
if ((actor->facingDirection < kDirUp) || (actor->facingDirection > kDirUpLeft))
|
|
|
|
error("Actor::getActorFrameRange Wrong direction 0x%X actorId 0x%X", actor->facingDirection, actorId);
|
|
|
|
|
2004-12-28 21:27:18 +00:00
|
|
|
fourDirection = actorDirectectionsLUT[actor->facingDirection];
|
2004-12-24 10:22:01 +00:00
|
|
|
return &actor->frames[frameType].directions[fourDirection];
|
2004-04-12 21:40:49 +00:00
|
|
|
}
|
|
|
|
|
2004-12-21 06:49:07 +00:00
|
|
|
void Actor::handleSpeech(int msec) {
|
|
|
|
int stringLength;
|
|
|
|
int sampleLength;
|
|
|
|
bool removeFirst;
|
|
|
|
int i;
|
|
|
|
int talkspeed;
|
|
|
|
ActorData *actor;
|
|
|
|
|
|
|
|
if (!isSpeaking()) return;
|
|
|
|
|
|
|
|
stringLength = strlen(_activeSpeech.strings[0]);
|
|
|
|
|
|
|
|
if (stringLength == 0)
|
|
|
|
error("Empty strings not allowed");
|
|
|
|
|
|
|
|
if (_vm->_script->_skipSpeeches) {
|
|
|
|
_activeSpeech.stringsCount = 0;
|
|
|
|
_vm->_sound->stopVoice();
|
|
|
|
_vm->_script->wakeUpThreads(kWaitTypeSpeech);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!_activeSpeech.playing) { // just added
|
|
|
|
talkspeed = ConfMan.getInt("talkspeed");
|
|
|
|
if (_activeSpeech.speechFlags & kSpeakSlow) {
|
|
|
|
if (_activeSpeech.slowModeCharIndex >= stringLength)
|
|
|
|
error("Wrong string index");
|
|
|
|
|
|
|
|
debug(0 , "Slow string!");
|
|
|
|
_activeSpeech.playingTime = 10 * talkspeed;
|
|
|
|
// 10 - fix it
|
|
|
|
|
|
|
|
} else {
|
|
|
|
sampleLength = _vm->_sndRes->getVoiceLength(_activeSpeech.sampleResourceId); //fixme - too fast
|
|
|
|
|
|
|
|
if (sampleLength < 0) {
|
|
|
|
_activeSpeech.playingTime = stringLength * talkspeed;
|
|
|
|
} else {
|
|
|
|
_activeSpeech.playingTime = sampleLength;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2004-12-21 10:30:49 +00:00
|
|
|
if (_activeSpeech.sampleResourceId != -1) {
|
2004-12-21 06:49:07 +00:00
|
|
|
_vm->_sndRes->playVoice(_activeSpeech.sampleResourceId);
|
2004-12-21 10:30:49 +00:00
|
|
|
_activeSpeech.sampleResourceId++;
|
|
|
|
}
|
2004-12-21 06:49:07 +00:00
|
|
|
|
|
|
|
if (_activeSpeech.actorIds[0] != 0) {
|
|
|
|
actor = getActor(_activeSpeech.actorIds[0]);
|
|
|
|
if (!(_activeSpeech.speechFlags & kSpeakNoAnimate)) {
|
|
|
|
actor->currentAction = kActionSpeak;
|
2004-12-24 12:47:16 +00:00
|
|
|
actor->actionCycle = rand() % 64;
|
2004-12-21 06:49:07 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
_activeSpeech.playing = true;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
_activeSpeech.playingTime -= msec;
|
|
|
|
|
|
|
|
removeFirst = false;
|
|
|
|
if (_activeSpeech.playingTime <= 0) {
|
|
|
|
if (_activeSpeech.speechFlags & kSpeakSlow) {
|
|
|
|
_activeSpeech.slowModeCharIndex++;
|
|
|
|
if (_activeSpeech.slowModeCharIndex >= stringLength)
|
|
|
|
removeFirst = true;
|
|
|
|
} else {
|
|
|
|
removeFirst = true;
|
|
|
|
}
|
2004-12-24 14:21:47 +00:00
|
|
|
_activeSpeech.playing = false;
|
2004-12-24 14:47:52 +00:00
|
|
|
if (_activeSpeech.actorIds[0] != 0) {
|
|
|
|
actor = getActor(_activeSpeech.actorIds[0]);
|
|
|
|
if (!(_activeSpeech.speechFlags & kSpeakNoAnimate)) {
|
|
|
|
actor->currentAction = kActionWait;
|
|
|
|
}
|
|
|
|
}
|
2004-12-21 06:49:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (removeFirst) {
|
|
|
|
for (i = 1; i < _activeSpeech.stringsCount; i++) {
|
|
|
|
_activeSpeech.strings[i - 1] = _activeSpeech.strings[i];
|
|
|
|
}
|
|
|
|
_activeSpeech.stringsCount--;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!isSpeaking())
|
|
|
|
_vm->_script->wakeUpThreadsDelayed(kWaitTypeSpeech, ticksToMSec(kScriptTimeTicksPerSecond / 3));
|
|
|
|
}
|
|
|
|
|
2004-12-22 21:04:50 +00:00
|
|
|
void Actor::handleActions(int msec, bool setup) {
|
2004-12-18 20:33:02 +00:00
|
|
|
int i;
|
2004-12-17 20:38:17 +00:00
|
|
|
ActorData *actor;
|
2004-12-22 21:04:50 +00:00
|
|
|
ActorFrameRange *frameRange;
|
|
|
|
int state;
|
2004-12-28 21:27:18 +00:00
|
|
|
int speed;
|
|
|
|
ActorLocation delta;
|
|
|
|
ActorLocation addDelta;
|
|
|
|
|
2004-12-22 21:04:50 +00:00
|
|
|
for (i = 0; i < ACTORCOUNT; i++) {
|
|
|
|
actor = &_actors[i];
|
|
|
|
if (actor->disabled) continue;
|
|
|
|
if (actor->sceneNumber != _vm->_scene->currentSceneNumber()) continue;
|
|
|
|
|
|
|
|
//todo: dragon stuff
|
|
|
|
|
2005-01-06 19:15:01 +00:00
|
|
|
/* if (actor->index == 2)
|
|
|
|
debug(9, "Action: %d Flags: %x", actor->currentAction, actor->flags);*/
|
2005-01-02 20:29:27 +00:00
|
|
|
|
2004-12-22 21:04:50 +00:00
|
|
|
switch(actor->currentAction) {
|
2005-01-05 19:03:49 +00:00
|
|
|
case kActionWait:
|
|
|
|
if (!setup && (actor->flags & kFollower)) {
|
|
|
|
followProtagonist(actor);
|
|
|
|
if (actor->currentAction != kActionWait)
|
|
|
|
break;
|
2004-12-22 21:04:50 +00:00
|
|
|
}
|
2005-01-02 20:29:27 +00:00
|
|
|
|
2005-01-05 19:03:49 +00:00
|
|
|
if (actor->targetObject != ID_NOTHING) {
|
2005-01-06 19:15:01 +00:00
|
|
|
actorFaceTowardsObject(actor->actorId, actor->targetObject);
|
2005-01-05 19:03:49 +00:00
|
|
|
}
|
2004-12-28 21:27:18 +00:00
|
|
|
|
2005-01-05 19:03:49 +00:00
|
|
|
if (actor->flags & kCycle) {
|
2005-01-05 07:19:58 +00:00
|
|
|
frameRange = getActorFrameRange(actor->actorId, kFrameStand);
|
2005-01-05 19:03:49 +00:00
|
|
|
if (frameRange->frameCount > 0) {
|
|
|
|
actor->actionCycle++;
|
|
|
|
actor->actionCycle = (actor->actionCycle) % frameRange->frameCount;
|
|
|
|
} else {
|
|
|
|
actor->actionCycle = 0;
|
|
|
|
}
|
|
|
|
actor->frameNumber = frameRange->frameIndex + actor->actionCycle;
|
|
|
|
break;
|
2005-01-05 07:19:58 +00:00
|
|
|
}
|
2004-12-28 21:27:18 +00:00
|
|
|
|
2005-01-05 19:03:49 +00:00
|
|
|
if ((actor->actionCycle & 3) == 0) {
|
|
|
|
actor->cycleWrap(100);
|
2004-12-28 21:27:18 +00:00
|
|
|
|
2005-01-05 19:03:49 +00:00
|
|
|
frameRange = getActorFrameRange(actor->actorId, kFrameWait);
|
|
|
|
if ((frameRange->frameCount < 1 || actor->actionCycle > 33))
|
|
|
|
frameRange = getActorFrameRange(actor->actorId, kFrameStand);
|
2004-12-28 21:27:18 +00:00
|
|
|
|
2005-01-05 19:03:49 +00:00
|
|
|
if (frameRange->frameCount) {
|
|
|
|
actor->frameNumber = frameRange->frameIndex + (uint16)rand() % frameRange->frameCount;
|
|
|
|
} else {
|
|
|
|
actor->frameNumber = frameRange->frameIndex;
|
2004-12-28 21:27:18 +00:00
|
|
|
}
|
2005-01-05 19:03:49 +00:00
|
|
|
}
|
|
|
|
actor->actionCycle++;
|
|
|
|
break;
|
2004-12-28 21:27:18 +00:00
|
|
|
|
2005-01-05 19:03:49 +00:00
|
|
|
case kActionWalkToPoint:
|
|
|
|
case kActionWalkToLink:
|
|
|
|
// tiled stuff
|
|
|
|
if (_vm->_scene->getMode() == SCENE_MODE_ISO) {
|
|
|
|
//todo: it
|
|
|
|
} else {
|
|
|
|
actor->partialTarget.delta(actor->location, delta);
|
2004-12-28 21:27:18 +00:00
|
|
|
|
2005-01-05 19:03:49 +00:00
|
|
|
while ((delta.x == 0) && (delta.y == 0)) {
|
2004-12-28 21:27:18 +00:00
|
|
|
|
2005-01-05 19:03:49 +00:00
|
|
|
if (actor->walkStepIndex >= actor->walkStepsCount) {
|
|
|
|
actorEndWalk(actor->actorId, true);
|
|
|
|
break;
|
|
|
|
}
|
2004-12-28 21:27:18 +00:00
|
|
|
|
2005-01-05 19:03:49 +00:00
|
|
|
actor->partialTarget.fromScreenPoint(actor->walkStepsPoints[actor->walkStepIndex++]);
|
|
|
|
if (actor->partialTarget.x > 224 * 2 * ACTOR_LMULT) {
|
|
|
|
actor->partialTarget.x -= 256 * 2 * ACTOR_LMULT;
|
|
|
|
}
|
|
|
|
|
|
|
|
actor->partialTarget.delta(actor->location, delta);
|
|
|
|
|
|
|
|
if (ABS(delta.y) > ABS(delta.x)) {
|
|
|
|
actor->actionDirection = delta.y > 0 ? kDirDown : kDirUp;
|
|
|
|
} else {
|
|
|
|
actor->actionDirection = delta.x > 0 ? kDirRight : kDirLeft;
|
|
|
|
}
|
2005-01-05 07:19:58 +00:00
|
|
|
}
|
2004-12-28 21:27:18 +00:00
|
|
|
|
2005-01-05 19:03:49 +00:00
|
|
|
speed = (ACTOR_LMULT * 2 * actor->screenScale + 63) / 256;
|
|
|
|
if (speed < 1) {
|
|
|
|
speed = 1;
|
|
|
|
}
|
2004-12-28 21:27:18 +00:00
|
|
|
|
2005-01-05 07:19:58 +00:00
|
|
|
if ((actor->actionDirection == kDirUp) || (actor->actionDirection == kDirDown)) {
|
|
|
|
// move by 2's in vertical dimension
|
2005-01-05 19:03:49 +00:00
|
|
|
addDelta.y = clamp(-speed, delta.y, speed);
|
|
|
|
if (addDelta.y == delta.y) {
|
|
|
|
addDelta.x = delta.x;
|
|
|
|
} else {
|
|
|
|
addDelta.x = delta.x * addDelta.y;
|
|
|
|
addDelta.x += (addDelta.x > 0) ? (delta.y / 2) : (-delta.y / 2);
|
|
|
|
addDelta.x /= delta.y;
|
|
|
|
actor->facingDirection = actor->actionDirection;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
addDelta.x = clamp(-2 * speed, delta.x, 2 * speed);
|
|
|
|
if (addDelta.x == delta.x) {
|
|
|
|
addDelta.y = delta.y;
|
|
|
|
} else {
|
|
|
|
addDelta.y = delta.y * addDelta.x;
|
|
|
|
addDelta.y += (addDelta.y > 0) ? (delta.x / 2) : (-delta.x / 2);
|
|
|
|
addDelta.y /= delta.x;
|
|
|
|
actor->facingDirection = actor->actionDirection;
|
|
|
|
}
|
2004-12-28 21:27:18 +00:00
|
|
|
}
|
|
|
|
|
2005-01-05 19:03:49 +00:00
|
|
|
actor->location.add(addDelta);
|
|
|
|
}
|
2005-01-02 20:29:27 +00:00
|
|
|
|
2005-01-05 19:03:49 +00:00
|
|
|
if (actor->actorFlags & kActorBackwards) {
|
|
|
|
actor->facingDirection = (actor->actionDirection + 4) & 7;
|
|
|
|
actor->actionCycle--;
|
|
|
|
} else {
|
|
|
|
actor->actionCycle++;
|
|
|
|
}
|
2005-01-04 16:10:43 +00:00
|
|
|
|
2005-01-05 19:03:49 +00:00
|
|
|
frameRange = getActorFrameRange(actor->actorId, actor->walkFrameSequence);
|
2005-01-05 07:19:58 +00:00
|
|
|
|
2005-01-05 19:03:49 +00:00
|
|
|
if (actor->actionCycle < 0) {
|
|
|
|
actor->actionCycle = frameRange->frameCount - 1;
|
|
|
|
} else {
|
|
|
|
if (actor->actionCycle >= frameRange->frameCount) {
|
|
|
|
actor->actionCycle = 0;
|
|
|
|
}
|
2005-01-04 16:10:43 +00:00
|
|
|
}
|
2005-01-02 20:29:27 +00:00
|
|
|
|
2005-01-05 19:03:49 +00:00
|
|
|
actor->frameNumber = frameRange->frameIndex + actor->actionCycle;
|
|
|
|
break;
|
2005-01-05 07:19:58 +00:00
|
|
|
|
2005-01-05 19:03:49 +00:00
|
|
|
case kActionWalkDir:
|
|
|
|
// tiled stuff
|
|
|
|
if (_vm->_scene->getMode() == SCENE_MODE_ISO) {
|
|
|
|
//todo: it
|
|
|
|
} else {
|
|
|
|
actor->location.x += directionLUT[actor->actionDirection][0] * 2;
|
|
|
|
actor->location.y += directionLUT[actor->actionDirection][1] * 2;
|
2005-01-05 07:19:58 +00:00
|
|
|
|
2005-01-05 19:03:49 +00:00
|
|
|
frameRange = getActorFrameRange(actor->actorId, actor->walkFrameSequence);
|
|
|
|
actor->actionCycle++;
|
|
|
|
actor->cycleWrap(frameRange->frameCount);
|
|
|
|
actor->frameNumber = frameRange->frameIndex + actor->actionCycle;
|
|
|
|
}
|
|
|
|
break;
|
2004-12-22 21:04:50 +00:00
|
|
|
|
2005-01-05 19:03:49 +00:00
|
|
|
case kActionSpeak:
|
|
|
|
actor->actionCycle++;
|
|
|
|
actor->cycleWrap(64);
|
2004-12-24 12:47:16 +00:00
|
|
|
|
2005-01-05 19:03:49 +00:00
|
|
|
frameRange = getActorFrameRange(actor->actorId, kFrameGesture);
|
|
|
|
if (actor->actionCycle >= frameRange->frameCount) {
|
2005-01-05 07:19:58 +00:00
|
|
|
if (actor->actionCycle & 1)
|
|
|
|
break;
|
2005-01-05 19:03:49 +00:00
|
|
|
frameRange = getActorFrameRange(actor->actorId, kFrameSpeak);
|
2004-12-22 21:04:50 +00:00
|
|
|
|
2005-01-05 19:03:49 +00:00
|
|
|
state = (uint16)rand() % (frameRange->frameCount + 1);
|
2005-01-05 07:19:58 +00:00
|
|
|
|
2005-01-05 19:03:49 +00:00
|
|
|
if (state == 0) {
|
|
|
|
frameRange = getActorFrameRange(actor->actorId, kFrameStand);
|
|
|
|
} else {
|
|
|
|
state--;
|
|
|
|
}
|
2004-12-22 21:04:50 +00:00
|
|
|
} else {
|
2005-01-05 19:03:49 +00:00
|
|
|
state = actor->actionCycle;
|
2004-12-22 21:04:50 +00:00
|
|
|
}
|
|
|
|
|
2005-01-05 19:03:49 +00:00
|
|
|
actor->frameNumber = frameRange->frameIndex + state;
|
|
|
|
break;
|
2004-12-22 21:04:50 +00:00
|
|
|
|
2005-01-05 19:03:49 +00:00
|
|
|
case kActionAccept:
|
|
|
|
case kActionStoop:
|
2005-01-05 07:19:58 +00:00
|
|
|
break;
|
2004-12-22 21:04:50 +00:00
|
|
|
|
2005-01-05 19:03:49 +00:00
|
|
|
case kActionCycleFrames:
|
|
|
|
case kActionPongFrames:
|
|
|
|
if (actor->cycleTimeCount > 0) {
|
|
|
|
actor->cycleTimeCount--;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
actor->cycleTimeCount = actor->cycleDelay;
|
|
|
|
actor->actionCycle++;
|
2004-12-22 21:04:50 +00:00
|
|
|
|
2005-01-05 19:03:49 +00:00
|
|
|
frameRange = getActorFrameRange(actor->actorId, actor->cycleFrameSequence);
|
2004-12-22 21:04:50 +00:00
|
|
|
|
2005-01-05 19:03:49 +00:00
|
|
|
if (actor->currentAction == kActionPongFrames) {
|
|
|
|
if (actor->actionCycle >= frameRange->frameCount * 2 - 2) {
|
|
|
|
if (actor->actorFlags & kActorContinuous) {
|
|
|
|
actor->actionCycle = 0;
|
|
|
|
} else {
|
|
|
|
actor->currentAction = kActionFreeze;
|
|
|
|
break;
|
|
|
|
}
|
2004-12-22 21:04:50 +00:00
|
|
|
}
|
|
|
|
|
2005-01-05 19:03:49 +00:00
|
|
|
state = actor->actionCycle;
|
|
|
|
if (state >= frameRange->frameCount) {
|
|
|
|
state = frameRange->frameCount * 2 - 2 - state;
|
2004-12-22 21:04:50 +00:00
|
|
|
}
|
2005-01-05 19:03:49 +00:00
|
|
|
} else {
|
|
|
|
if (actor->actionCycle >= frameRange->frameCount) {
|
|
|
|
if (actor->actorFlags & kActorContinuous) {
|
|
|
|
actor->actionCycle = 0;
|
|
|
|
} else {
|
|
|
|
actor->currentAction = kActionFreeze;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
state = actor->actionCycle;
|
2004-12-22 21:04:50 +00:00
|
|
|
}
|
|
|
|
|
2005-01-05 19:03:49 +00:00
|
|
|
if (frameRange->frameCount && (actor->actorFlags & kActorRandom)) {
|
|
|
|
state = rand() % frameRange->frameCount;
|
|
|
|
}
|
2004-12-22 21:04:50 +00:00
|
|
|
|
2005-01-05 19:03:49 +00:00
|
|
|
if (actor->actorFlags & kActorBackwards) {
|
|
|
|
actor->frameNumber = frameRange->frameIndex + frameRange->frameCount - 1 - state;
|
|
|
|
} else {
|
|
|
|
actor->frameNumber = frameRange->frameIndex + state;
|
|
|
|
}
|
|
|
|
break;
|
2005-01-02 20:29:27 +00:00
|
|
|
|
2005-01-05 19:03:49 +00:00
|
|
|
case kActionFall:
|
|
|
|
debug(9,"kActionFall not implemented");
|
2004-12-22 21:04:50 +00:00
|
|
|
|
2005-01-05 19:03:49 +00:00
|
|
|
//todo: do it
|
|
|
|
break;
|
2005-01-02 20:29:27 +00:00
|
|
|
|
2005-01-05 19:03:49 +00:00
|
|
|
case kActionClimb:
|
|
|
|
debug(9,"kActionClimb not implemented");
|
2004-12-22 21:04:50 +00:00
|
|
|
|
2005-01-05 19:03:49 +00:00
|
|
|
//todo: do it
|
|
|
|
break;
|
2004-12-22 21:04:50 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
int Actor::direct(int msec) {
|
2004-12-24 13:14:23 +00:00
|
|
|
|
|
|
|
// FIXME: HACK. This should be turned into cycle event.
|
2004-12-24 12:47:16 +00:00
|
|
|
_lastTickMsec += msec;
|
|
|
|
|
2004-12-24 13:42:05 +00:00
|
|
|
if (_lastTickMsec > ticksToMSec(5)) { // fixme
|
2004-12-24 12:47:16 +00:00
|
|
|
_lastTickMsec = 0;
|
|
|
|
//process actions
|
|
|
|
handleActions(msec, false);
|
|
|
|
}
|
2004-04-12 21:40:49 +00:00
|
|
|
|
2004-12-21 06:49:07 +00:00
|
|
|
//process speech
|
|
|
|
handleSpeech(msec);
|
2004-10-27 21:32:28 +00:00
|
|
|
return SUCCESS;
|
2004-04-12 21:40:49 +00:00
|
|
|
}
|
|
|
|
|
2004-12-30 18:45:48 +00:00
|
|
|
void Actor::calcActorScreenPosition(ActorData *actor) {
|
2004-12-25 11:17:03 +00:00
|
|
|
int beginSlope, endSlope, middle;
|
2004-12-28 21:27:18 +00:00
|
|
|
if (_vm->_scene->getMode() == SCENE_MODE_ISO) {
|
|
|
|
//todo: it
|
|
|
|
} else {
|
|
|
|
middle = _vm->getStatusYOffset() - actor->location.y / ACTOR_LMULT;
|
2004-12-25 11:17:03 +00:00
|
|
|
|
|
|
|
_vm->_scene->getSlopes(beginSlope, endSlope);
|
|
|
|
|
|
|
|
actor->screenDepth = (14 * middle) / endSlope + 1;
|
|
|
|
|
|
|
|
if (middle <= beginSlope) {
|
|
|
|
actor->screenScale = 256;
|
|
|
|
} else {
|
|
|
|
if (middle >= endSlope) {
|
|
|
|
actor->screenScale = 1;
|
|
|
|
} else {
|
|
|
|
middle -= beginSlope;
|
|
|
|
endSlope -= beginSlope;
|
|
|
|
actor->screenScale = 256 - (middle * 256) / endSlope;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2005-01-05 19:03:49 +00:00
|
|
|
actor->location.toScreenPointXYZ(actor->screenPosition);
|
2004-12-25 11:17:03 +00:00
|
|
|
}
|
2005-01-02 20:29:27 +00:00
|
|
|
|
2005-01-06 19:15:01 +00:00
|
|
|
/*if (actor->index == 2)
|
|
|
|
debug(9, "act: %d. x: %d y: %d", actor->index, actor->screenPosition.x, actor->screenPosition.y);*/
|
2004-12-25 11:17:03 +00:00
|
|
|
}
|
|
|
|
|
2004-12-18 20:33:02 +00:00
|
|
|
void Actor::createDrawOrderList() {
|
|
|
|
int i;
|
|
|
|
ActorData *actor;
|
|
|
|
|
|
|
|
_drawOrderList.clear();
|
2004-12-21 06:49:07 +00:00
|
|
|
for (i = 0; i < ACTORCOUNT; i++) {
|
2004-12-18 20:33:02 +00:00
|
|
|
actor = &_actors[i];
|
|
|
|
if (actor->disabled) continue;
|
2004-12-19 13:38:11 +00:00
|
|
|
if (actor->sceneNumber != _vm->_scene->currentSceneNumber()) continue;
|
2004-12-18 20:33:02 +00:00
|
|
|
|
2004-12-19 13:38:11 +00:00
|
|
|
_drawOrderList.pushBack(actor, actorCompare);
|
|
|
|
|
2004-12-25 11:17:03 +00:00
|
|
|
calcActorScreenPosition(actor);
|
2004-12-18 20:33:02 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
int Actor::drawActors() {
|
|
|
|
ActorOrderList::iterator actorDrawOrderIterator;
|
2004-12-17 20:38:17 +00:00
|
|
|
ActorData *actor;
|
2004-12-22 21:04:50 +00:00
|
|
|
int frameNumber;
|
|
|
|
SPRITELIST *spriteList;
|
2004-04-12 21:40:49 +00:00
|
|
|
|
2004-10-27 21:32:28 +00:00
|
|
|
SURFACE *back_buf;
|
2004-04-12 21:40:49 +00:00
|
|
|
|
2004-08-01 11:48:53 +00:00
|
|
|
back_buf = _vm->_gfx->getBackBuffer();
|
2004-04-12 21:40:49 +00:00
|
|
|
|
2004-12-18 20:33:02 +00:00
|
|
|
createDrawOrderList();
|
|
|
|
|
|
|
|
for (actorDrawOrderIterator = _drawOrderList.begin(); actorDrawOrderIterator != _drawOrderList.end(); ++actorDrawOrderIterator) {
|
|
|
|
actor = actorDrawOrderIterator.operator*();
|
2004-12-18 16:08:59 +00:00
|
|
|
|
2004-12-22 21:04:50 +00:00
|
|
|
if (_vm->_scene->currentSceneNumber() == RID_ITE_OVERMAP_SCENE) {
|
|
|
|
if (!(actor->flags & kProtagonist)){
|
|
|
|
warning("not protagonist");
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
frameNumber = 8;
|
|
|
|
spriteList = _vm->_mainSprites;
|
|
|
|
} else {
|
|
|
|
frameNumber = actor->frameNumber;
|
|
|
|
spriteList = actor->spriteList;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((frameNumber < 0) || (spriteList->sprite_count <= frameNumber)) {
|
|
|
|
warning("Actor::drawActors frameNumber invalid for actorId 0x%X", actor->actorId);
|
2004-12-19 13:38:11 +00:00
|
|
|
continue;
|
|
|
|
}
|
2004-12-24 20:44:39 +00:00
|
|
|
|
2004-12-28 21:27:18 +00:00
|
|
|
if (_vm->_scene->getMode() == SCENE_MODE_ISO) {
|
|
|
|
//todo: it
|
|
|
|
} else {
|
|
|
|
_vm->_sprite->drawOccluded(back_buf, spriteList, frameNumber, actor->screenPosition, actor->screenScale, actor->screenDepth);
|
2004-12-24 20:44:39 +00:00
|
|
|
}
|
2004-04-12 21:40:49 +00:00
|
|
|
}
|
|
|
|
|
2004-12-21 06:49:07 +00:00
|
|
|
// draw speeches
|
|
|
|
if (isSpeaking() && !_vm->_script->_skipSpeeches) {
|
|
|
|
int i;
|
|
|
|
int textDrawFlags, speechColor;
|
|
|
|
Point speechCoord;
|
|
|
|
char oneChar[2];
|
|
|
|
oneChar[1] = 0;
|
|
|
|
const char *outputString;
|
|
|
|
|
|
|
|
if (_activeSpeech.speechFlags & kSpeakSlow) {
|
|
|
|
outputString = oneChar;
|
|
|
|
oneChar[0] = _activeSpeech.strings[0][_activeSpeech.slowModeCharIndex];
|
|
|
|
} else {
|
|
|
|
outputString = _activeSpeech.strings[0];
|
2004-04-12 21:40:49 +00:00
|
|
|
}
|
|
|
|
|
2004-12-21 06:49:07 +00:00
|
|
|
textDrawFlags = FONT_CENTERED;
|
|
|
|
if (_activeSpeech.outlineColor != 0) {
|
|
|
|
textDrawFlags |= FONT_OUTLINE;
|
2004-04-12 21:40:49 +00:00
|
|
|
}
|
|
|
|
|
2004-12-21 06:49:07 +00:00
|
|
|
if (_activeSpeech.actorIds[0] != 0) {
|
|
|
|
|
|
|
|
for (i = 0; i < _activeSpeech.actorsCount; i++){
|
|
|
|
actor = getActor(_activeSpeech.actorIds[i]);
|
|
|
|
speechCoord.x = actor->screenPosition.x;
|
|
|
|
speechCoord.y = actor->screenPosition.y;
|
|
|
|
speechCoord.y -= ACTOR_DIALOGUE_HEIGHT;
|
|
|
|
if (_activeSpeech.actorsCount > 1)
|
|
|
|
speechColor = actor->speechColor;
|
|
|
|
else
|
|
|
|
speechColor = _activeSpeech.speechColor;
|
|
|
|
|
|
|
|
_vm->textDraw(MEDIUM_FONT_ID, back_buf, outputString, speechCoord.x, speechCoord.y, speechColor, _activeSpeech.outlineColor, textDrawFlags);
|
2004-04-12 21:40:49 +00:00
|
|
|
}
|
|
|
|
|
2004-12-21 06:49:07 +00:00
|
|
|
} else { // non actors speech
|
|
|
|
warning("non actors speech occures");
|
|
|
|
//todo: write it
|
2004-04-12 21:40:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2004-10-27 21:32:28 +00:00
|
|
|
return SUCCESS;
|
2004-04-12 21:40:49 +00:00
|
|
|
}
|
|
|
|
|
2004-12-30 18:45:48 +00:00
|
|
|
bool Actor::followProtagonist(ActorData *actor) {
|
2004-12-28 21:27:18 +00:00
|
|
|
ActorLocation protagonistLocation;
|
|
|
|
ActorLocation newLocation;
|
|
|
|
ActorLocation delta;
|
|
|
|
int protagonistBGMaskType;
|
|
|
|
Point prefer1;
|
|
|
|
Point prefer2;
|
|
|
|
Point prefer3;
|
|
|
|
|
|
|
|
assert(_protagonist);
|
|
|
|
|
|
|
|
actor->flags &= ~(kFaster | kFastest);
|
|
|
|
protagonistLocation = _protagonist->location;
|
2005-01-04 16:10:43 +00:00
|
|
|
calcActorScreenPosition(_protagonist);
|
2004-12-28 21:27:18 +00:00
|
|
|
|
|
|
|
if (_vm->_scene->getMode() == SCENE_MODE_ISO) {
|
|
|
|
//todo: it
|
|
|
|
} else {
|
|
|
|
prefer1.x = (100 * _protagonist->screenScale) >> 8;
|
|
|
|
prefer1.y = (50 * _protagonist->screenScale) >> 8;
|
|
|
|
|
|
|
|
if (_protagonist->currentAction == kActionWalkDir) {
|
|
|
|
prefer1.x /= 2;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (prefer1.x < 8) {
|
|
|
|
prefer1.x = 8;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (prefer1.y < 8) {
|
|
|
|
prefer1.y = 8;
|
|
|
|
}
|
|
|
|
|
|
|
|
prefer2.x = prefer1.x * 2;
|
|
|
|
prefer2.y = prefer1.y * 2;
|
|
|
|
prefer3.x = prefer1.x + prefer1.x / 2;
|
|
|
|
prefer3.y = prefer1.y + prefer1.y / 2;
|
|
|
|
|
|
|
|
actor->location.delta(protagonistLocation, delta);
|
|
|
|
|
2005-01-04 16:10:43 +00:00
|
|
|
protagonistBGMaskType = 0;
|
|
|
|
if (_vm->_scene->isBGMaskPresent()) {
|
|
|
|
if (_vm->_scene->validBGMaskPoint(_protagonist->screenPosition)) {
|
|
|
|
protagonistBGMaskType = _vm->_scene->getBGMaskType(_protagonist->screenPosition);
|
|
|
|
}
|
|
|
|
}
|
2004-12-28 21:27:18 +00:00
|
|
|
|
2005-01-04 16:10:43 +00:00
|
|
|
|
2004-12-28 21:27:18 +00:00
|
|
|
|
|
|
|
if ((rand() & 0x7) == 0)
|
|
|
|
actor->actorFlags &= ~kActorNoFollow;
|
|
|
|
|
|
|
|
if (actor->actorFlags & kActorNoFollow) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((delta.x > prefer2.x) || (delta.x < -prefer2.x) ||
|
|
|
|
(delta.y > prefer2.y) || (delta.y < -prefer2.y) ||
|
|
|
|
((_protagonist->currentAction == kActionWait) &&
|
|
|
|
(delta.x * 2 < prefer1.x) && (delta.x * 2 > -prefer1.x) &&
|
|
|
|
(delta.y < prefer1.y) && (delta.y > -prefer1.y))) {
|
|
|
|
|
|
|
|
if (ABS(delta.x) > ABS(delta.y)) {
|
|
|
|
|
|
|
|
delta.x = (delta.x > 0) ? prefer3.x : -prefer3.x;
|
|
|
|
|
|
|
|
newLocation.x = delta.x + protagonistLocation.x;
|
|
|
|
newLocation.y = clamp(-prefer2.y, delta.y, prefer2.y) + protagonistLocation.y;
|
|
|
|
} else {
|
|
|
|
delta.y = (delta.y > 0) ? prefer3.y : -prefer3.y;
|
|
|
|
|
|
|
|
newLocation.x = clamp(-prefer2.x, delta.x, prefer2.x) + protagonistLocation.x;
|
|
|
|
newLocation.y = delta.y + protagonistLocation.y;
|
|
|
|
}
|
|
|
|
newLocation.z = 0;
|
|
|
|
|
|
|
|
if (protagonistBGMaskType != 3) {
|
|
|
|
newLocation.x += (rand() % prefer1.x) - prefer1.x / 2;
|
|
|
|
newLocation.y += (rand() % prefer1.y) - prefer1.y / 2;
|
|
|
|
}
|
|
|
|
|
|
|
|
newLocation.x = clamp(-31*4, newLocation.x, (_vm->getDisplayWidth() + 31) * 4); //fixme
|
|
|
|
|
|
|
|
return actorWalkTo(actor->actorId, newLocation);
|
|
|
|
}
|
|
|
|
}
|
2004-12-25 11:17:03 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Actor::actorEndWalk(uint16 actorId, bool recurse) {
|
|
|
|
bool walkMore = false;
|
2004-12-17 20:38:17 +00:00
|
|
|
ActorData *actor;
|
2004-04-12 21:40:49 +00:00
|
|
|
|
2004-12-17 20:38:17 +00:00
|
|
|
actor = getActor(actorId);
|
2004-12-25 11:17:03 +00:00
|
|
|
actor->actorFlags &= ~kActorBackwards;
|
2004-04-12 21:40:49 +00:00
|
|
|
|
2004-12-25 11:17:03 +00:00
|
|
|
if (actor->location.distance(actor->finalTarget) > 8) {
|
|
|
|
if ((actor->flags & kProtagonist) && recurse && !(actor->actorFlags & kActorNoCollide)) {
|
|
|
|
actor->actorFlags |= kActorNoCollide;
|
|
|
|
return actorWalkTo(actorId, actor->finalTarget);
|
|
|
|
}
|
2004-04-12 21:40:49 +00:00
|
|
|
}
|
2004-12-25 11:17:03 +00:00
|
|
|
|
|
|
|
actor->currentAction = kActionWait;
|
|
|
|
if (actor->actorFlags & kActorFinalFace) {
|
|
|
|
actor->facingDirection = actor->actionDirection = (actor->actorFlags >> 6) & 0x07; //?
|
|
|
|
}
|
|
|
|
|
|
|
|
actor->actorFlags &= ~(kActorNoCollide | kActorCollided | kActorFinalFace | kActorFacingMask);
|
|
|
|
actor->flags &= ~(kFaster | kFastest);
|
|
|
|
|
|
|
|
if (actor == _protagonist) {
|
|
|
|
_vm->_script->wakeUpActorThread(kWaitTypeWalk, actor);
|
|
|
|
//todo: it
|
|
|
|
|
|
|
|
} else {
|
|
|
|
if (recurse && (actor->flags & kFollower))
|
|
|
|
walkMore = followProtagonist(actor);
|
|
|
|
|
|
|
|
_vm->_script->wakeUpActorThread(kWaitTypeWalk, actor);
|
|
|
|
}
|
|
|
|
return walkMore;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Actor::actorWalkTo(uint16 actorId, const ActorLocation &toLocation) {
|
|
|
|
ActorData *actor;
|
2004-12-28 21:27:18 +00:00
|
|
|
ActorData *anotherActor;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
Rect testBox;
|
|
|
|
Rect testBox2;
|
|
|
|
Point anotherActorScreenPosition;
|
|
|
|
Point collision;
|
|
|
|
Point pointFrom, pointTo, pointBest, pointAdd;
|
|
|
|
Point delta, bestDelta;
|
|
|
|
bool extraStartNode;
|
|
|
|
bool extraEndNode;
|
2004-12-25 11:17:03 +00:00
|
|
|
|
|
|
|
actor = getActor(actorId);
|
2005-01-02 20:29:27 +00:00
|
|
|
|
2004-12-28 21:27:18 +00:00
|
|
|
if (actor == _protagonist) {
|
|
|
|
_vm->_scene->setDoorState(2, 0xff);
|
|
|
|
_vm->_scene->setDoorState(3, 0);
|
|
|
|
} else {
|
|
|
|
_vm->_scene->setDoorState(2, 0);
|
|
|
|
_vm->_scene->setDoorState(3, 0xff);
|
|
|
|
}
|
2004-12-30 18:45:48 +00:00
|
|
|
|
2004-12-28 21:27:18 +00:00
|
|
|
if (_vm->_scene->getMode() == SCENE_MODE_ISO) {
|
2004-12-25 11:17:03 +00:00
|
|
|
//todo: it
|
2004-12-28 21:27:18 +00:00
|
|
|
} else {
|
2005-01-05 19:03:49 +00:00
|
|
|
|
|
|
|
actor->location.toScreenPointXY(pointFrom);
|
2004-12-25 11:17:03 +00:00
|
|
|
|
2004-12-28 21:27:18 +00:00
|
|
|
extraStartNode = _vm->_scene->offscreenPath(pointFrom);
|
2004-12-25 11:17:03 +00:00
|
|
|
|
2005-01-05 19:03:49 +00:00
|
|
|
toLocation.toScreenPointXY(pointTo);
|
2004-12-25 11:17:03 +00:00
|
|
|
|
2004-12-28 21:27:18 +00:00
|
|
|
extraEndNode = _vm->_scene->offscreenPath(pointTo);
|
2004-12-25 11:17:03 +00:00
|
|
|
|
|
|
|
if (_vm->_scene->isBGMaskPresent()) {
|
2004-12-28 21:27:18 +00:00
|
|
|
|
|
|
|
if ((((actor->currentAction >= kActionWalkToPoint) &&
|
|
|
|
(actor->currentAction <= kActionWalkDir)) || (actor == _protagonist)) &&
|
|
|
|
!_vm->_scene->canWalk(pointFrom)) {
|
|
|
|
for (i = 1; i < 8; i++) {
|
|
|
|
pointAdd = pointFrom;
|
|
|
|
pointAdd.y += i;
|
|
|
|
if (_vm->_scene->canWalk(pointAdd)) {
|
|
|
|
pointFrom = pointAdd;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
pointAdd = pointFrom;
|
|
|
|
pointAdd.y -= i;
|
|
|
|
if (_vm->_scene->canWalk(pointAdd)) {
|
|
|
|
pointFrom = pointAdd;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
pointAdd = pointFrom;
|
|
|
|
pointAdd.x += i;
|
|
|
|
if (_vm->_scene->canWalk(pointAdd)) {
|
|
|
|
pointFrom = pointAdd;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
pointAdd = pointFrom;
|
|
|
|
pointAdd.x -= i;
|
|
|
|
if (_vm->_scene->canWalk(pointAdd)) {
|
|
|
|
pointFrom = pointAdd;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!(actor->actorFlags & kActorNoCollide)) {
|
|
|
|
collision.x = ACTOR_COLLISION_WIDTH * actor->screenScale / (256 * 2);
|
|
|
|
collision.y = ACTOR_COLLISION_HEIGHT * actor->screenScale / (256 * 2);
|
|
|
|
|
|
|
|
_barrierCount = 0;
|
|
|
|
|
|
|
|
for (i = 0; (i < ACTORCOUNT) && (_barrierCount < ACTOR_BARRIERS_MAX); i++) {
|
|
|
|
anotherActor = &_actors[i];
|
|
|
|
if (anotherActor->disabled) continue;
|
|
|
|
if (anotherActor->sceneNumber != _vm->_scene->currentSceneNumber()) continue;
|
|
|
|
if (anotherActor == actor ) continue;
|
|
|
|
|
|
|
|
|
|
|
|
anotherActorScreenPosition = anotherActor->screenPosition;
|
2005-01-06 14:32:14 +00:00
|
|
|
testBox.left = anotherActorScreenPosition.x - collision.x;
|
|
|
|
testBox.right = anotherActorScreenPosition.x + collision.x;
|
2004-12-28 21:27:18 +00:00
|
|
|
testBox.top = anotherActorScreenPosition.y - collision.y;
|
|
|
|
testBox.bottom = anotherActorScreenPosition.y + collision.y;
|
|
|
|
testBox2 = testBox;
|
|
|
|
testBox2.right += 2;
|
|
|
|
testBox2.left -= 1;
|
|
|
|
testBox2.bottom += 1;
|
|
|
|
|
|
|
|
if (testBox2.contains(pointFrom)) {
|
|
|
|
if (pointFrom.x > anotherActorScreenPosition.x + 4) {
|
|
|
|
testBox.right = pointFrom.x - 2;
|
|
|
|
} else {
|
|
|
|
if (pointFrom.x < anotherActorScreenPosition.x - 4) {
|
|
|
|
testBox.left = pointFrom.x + 2;
|
|
|
|
} else {
|
|
|
|
if (pointFrom.y > anotherActorScreenPosition.y) {
|
|
|
|
testBox.bottom = pointFrom.y - 1;
|
|
|
|
} else {
|
|
|
|
testBox.top = pointFrom.y + 1 ;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((testBox.left <= testBox.right) && (testBox.top <= testBox.bottom)) {
|
|
|
|
_barrierList[_barrierCount++] = testBox;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-12-29 22:31:21 +00:00
|
|
|
pointBest = pointTo;
|
|
|
|
actor->walkStepsCount = 0;
|
|
|
|
findActorPath(actor, pointFrom, pointTo);
|
2004-12-28 21:27:18 +00:00
|
|
|
|
2005-01-05 19:03:49 +00:00
|
|
|
if (actor->walkStepsCount == 0) {
|
|
|
|
error("actor->walkStepsCount == 0");
|
|
|
|
}
|
|
|
|
|
2004-12-29 22:31:21 +00:00
|
|
|
if (extraStartNode) {
|
|
|
|
actor->walkStepIndex = 0;
|
|
|
|
} else {
|
2005-01-05 19:03:49 +00:00
|
|
|
actor->walkStepIndex = 1;
|
2005-01-04 18:54:29 +00:00
|
|
|
}
|
|
|
|
|
2004-12-29 22:31:21 +00:00
|
|
|
if (extraEndNode) {
|
2005-01-05 19:03:49 +00:00
|
|
|
Point tempPoint;
|
|
|
|
toLocation.toScreenPointXY(tempPoint);
|
|
|
|
actor->walkStepsCount--;
|
|
|
|
actor->addWalkStepPoint(tempPoint);
|
2004-12-29 22:31:21 +00:00
|
|
|
}
|
2004-12-28 21:27:18 +00:00
|
|
|
|
2005-01-05 19:03:49 +00:00
|
|
|
|
|
|
|
pointBest = actor->walkStepsPoints[actor->walkStepsCount - 1];
|
2004-12-28 21:27:18 +00:00
|
|
|
|
2004-12-29 22:31:21 +00:00
|
|
|
pointFrom.x &= ~1;
|
|
|
|
delta.x = ABS(pointFrom.x - pointTo.x);
|
|
|
|
delta.y = ABS(pointFrom.y - pointTo.y);
|
2004-12-28 21:27:18 +00:00
|
|
|
|
2004-12-29 22:31:21 +00:00
|
|
|
bestDelta.x = ABS(pointBest.x - pointTo.x);
|
|
|
|
bestDelta.y = ABS(pointBest.y - pointTo.y);
|
2004-12-28 21:27:18 +00:00
|
|
|
|
2004-12-29 22:31:21 +00:00
|
|
|
if (delta.x + delta.y <= bestDelta.x + bestDelta.y) {
|
|
|
|
if (actor->flags & kFollower)
|
|
|
|
actor->actorFlags |= kActorNoFollow;
|
|
|
|
}
|
2004-12-28 21:27:18 +00:00
|
|
|
|
2004-12-29 22:31:21 +00:00
|
|
|
if (pointBest == pointFrom) {
|
|
|
|
actor->walkStepsCount = 0;
|
|
|
|
}
|
2004-12-25 11:17:03 +00:00
|
|
|
} else {
|
2005-01-05 19:03:49 +00:00
|
|
|
actor->walkStepsCount = 0;
|
|
|
|
actor->addWalkStepPoint(pointTo);
|
2004-12-25 11:17:03 +00:00
|
|
|
actor->walkStepIndex = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
actor->partialTarget = actor->location;
|
|
|
|
actor->finalTarget = toLocation;
|
|
|
|
if (actor->walkStepsCount == 0) {
|
|
|
|
actorEndWalk(actorId, false);
|
|
|
|
return false;
|
|
|
|
} else {
|
|
|
|
if (actor->flags & kProtagonist) {
|
|
|
|
_actors[1].actorFlags &= ~kActorNoFollow;
|
|
|
|
_actors[2].actorFlags &= ~kActorNoFollow;
|
2004-12-28 21:27:18 +00:00
|
|
|
}
|
2005-01-05 19:03:49 +00:00
|
|
|
actor->currentAction = (actor->walkStepsCount >= ACTOR_MAX_STEPS_COUNT) ? kActionWalkToLink : kActionWalkToPoint;
|
2004-12-25 11:17:03 +00:00
|
|
|
actor->walkFrameSequence = kFrameWalk;
|
|
|
|
}
|
|
|
|
|
2004-04-12 21:40:49 +00:00
|
|
|
}
|
2004-12-29 22:31:21 +00:00
|
|
|
return true;
|
2004-04-12 21:40:49 +00:00
|
|
|
}
|
|
|
|
|
2004-12-21 06:49:07 +00:00
|
|
|
void Actor::actorSpeech(uint16 actorId, const char **strings, int stringsCount, uint16 sampleResourceId, int speechFlags) {
|
|
|
|
ActorData *actor;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
actor = getActor(actorId);
|
|
|
|
for (i = 0; i < stringsCount; i++) {
|
|
|
|
_activeSpeech.strings[i] = strings[i];
|
|
|
|
}
|
|
|
|
_activeSpeech.stringsCount = stringsCount;
|
|
|
|
_activeSpeech.speechFlags = speechFlags;
|
|
|
|
_activeSpeech.actorsCount = 1;
|
|
|
|
_activeSpeech.actorIds[0] = actorId;
|
|
|
|
_activeSpeech.speechColor = actor->speechColor;
|
|
|
|
_activeSpeech.outlineColor = 15; // fixme - BLACK
|
|
|
|
_activeSpeech.sampleResourceId = sampleResourceId;
|
|
|
|
_activeSpeech.playing = false;
|
|
|
|
_activeSpeech.slowModeCharIndex = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Actor::nonActorSpeech(const char **strings, int stringsCount, int speechFlags) {
|
|
|
|
int i;
|
|
|
|
|
|
|
|
_vm->_script->wakeUpThreads(kWaitTypeSpeech);
|
|
|
|
|
|
|
|
for (i = 0; i < stringsCount; i++) {
|
|
|
|
_activeSpeech.strings[i] = strings[i];
|
|
|
|
}
|
|
|
|
_activeSpeech.stringsCount = stringsCount;
|
|
|
|
_activeSpeech.speechFlags = speechFlags;
|
|
|
|
_activeSpeech.actorsCount = 1;
|
|
|
|
_activeSpeech.actorIds[0] = 0;
|
|
|
|
//_activeSpeech.speechColor = ;
|
|
|
|
//_activeSpeech.outlineColor = ;
|
|
|
|
_activeSpeech.sampleResourceId = -1;
|
|
|
|
_activeSpeech.playing = false;
|
|
|
|
_activeSpeech.slowModeCharIndex = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Actor::simulSpeech(const char *string, uint16 *actorIds, int actorIdsCount, int speechFlags) {
|
|
|
|
int i;
|
2004-12-22 21:04:50 +00:00
|
|
|
|
2004-12-21 06:49:07 +00:00
|
|
|
for (i = 0; i < actorIdsCount; i++) {
|
|
|
|
_activeSpeech.actorIds[i] = actorIds[i];
|
|
|
|
}
|
2004-12-22 21:04:50 +00:00
|
|
|
_activeSpeech.actorsCount = actorIdsCount;
|
2004-12-21 06:49:07 +00:00
|
|
|
_activeSpeech.strings[0] = string;
|
|
|
|
_activeSpeech.stringsCount = 1;
|
|
|
|
_activeSpeech.speechFlags = speechFlags;
|
|
|
|
//_activeSpeech.speechColor = ; // get's from every actor
|
|
|
|
_activeSpeech.outlineColor = 0; // disable outline
|
|
|
|
_activeSpeech.sampleResourceId = -1;
|
|
|
|
_activeSpeech.playing = false;
|
|
|
|
_activeSpeech.slowModeCharIndex = 0;
|
|
|
|
|
|
|
|
// caller should call thread->wait(kWaitTypeSpeech) by itself
|
|
|
|
}
|
|
|
|
|
|
|
|
void Actor::abortAllSpeeches() {
|
|
|
|
if (_vm->_script->_abortEnabled)
|
|
|
|
_vm->_script->_skipSpeeches = true;
|
|
|
|
|
|
|
|
for (int i = 0; i < 10; i++)
|
|
|
|
_vm->_script->executeThreads(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Actor::abortSpeech() {
|
|
|
|
_vm->_sound->stopVoice();
|
|
|
|
_activeSpeech.playingTime = 0;
|
|
|
|
}
|
|
|
|
|
2004-12-30 18:45:48 +00:00
|
|
|
void Actor::findActorPath(ActorData *actor, const Point &fromPoint, const Point &toPoint) {
|
2004-12-28 21:27:18 +00:00
|
|
|
Point iteratorPoint;
|
|
|
|
Point bestPoint;
|
2005-01-05 19:03:49 +00:00
|
|
|
int maskType;
|
2004-12-29 14:33:14 +00:00
|
|
|
int cellValue;
|
2004-12-28 21:27:18 +00:00
|
|
|
int i;
|
|
|
|
Rect intersect;
|
|
|
|
|
2005-01-06 19:15:01 +00:00
|
|
|
_debugPointsCount = 0;
|
2004-12-28 21:27:18 +00:00
|
|
|
|
|
|
|
actor->walkStepsCount = 0;
|
2004-12-29 21:49:25 +00:00
|
|
|
if (fromPoint == toPoint) {
|
2005-01-05 19:03:49 +00:00
|
|
|
actor->addWalkStepPoint(toPoint);
|
2004-12-28 21:27:18 +00:00
|
|
|
return;
|
|
|
|
}
|
2005-01-06 14:32:14 +00:00
|
|
|
|
2004-12-28 21:27:18 +00:00
|
|
|
for (iteratorPoint.y = 0; iteratorPoint.y < _yCellCount; iteratorPoint.y++) {
|
|
|
|
for (iteratorPoint.x = 0; iteratorPoint.x < _xCellCount; iteratorPoint.x++) {
|
2005-01-05 19:03:49 +00:00
|
|
|
maskType = _vm->_scene->getBGMaskType(iteratorPoint);
|
|
|
|
cellValue = maskType ? kPathCellBarrier : kPathCellEmpty;
|
2005-01-06 19:15:01 +00:00
|
|
|
setPathCell(iteratorPoint, cellValue);
|
2004-12-28 21:27:18 +00:00
|
|
|
}
|
|
|
|
}
|
2005-01-06 14:32:14 +00:00
|
|
|
|
2004-12-28 21:27:18 +00:00
|
|
|
for (i = 0; i < _barrierCount; i++) {
|
|
|
|
intersect.left = MAX(_pathRect.left, _barrierList[i].left);
|
|
|
|
intersect.top = MAX(_pathRect.top, _barrierList[i].top);
|
|
|
|
intersect.right = MIN(_pathRect.right, _barrierList[i].right);
|
|
|
|
intersect.bottom = MIN(_pathRect.bottom, _barrierList[i].bottom);
|
2005-01-04 18:54:29 +00:00
|
|
|
|
2005-01-05 19:03:49 +00:00
|
|
|
|
|
|
|
for (iteratorPoint.y = intersect.top; iteratorPoint.y <= intersect.bottom; iteratorPoint.y++) {
|
|
|
|
for (iteratorPoint.x = intersect.left; iteratorPoint.x <= intersect.right; iteratorPoint.x++) {
|
2004-12-29 14:33:14 +00:00
|
|
|
setPathCell(iteratorPoint, kPathCellBarrier);
|
2004-12-28 21:27:18 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2005-01-06 19:15:01 +00:00
|
|
|
#if 1
|
2004-12-28 21:27:18 +00:00
|
|
|
|
2005-01-06 19:15:01 +00:00
|
|
|
for (iteratorPoint.y = 0; iteratorPoint.y < _yCellCount; iteratorPoint.y++) {
|
|
|
|
for (iteratorPoint.x = 0; iteratorPoint.x < _xCellCount; iteratorPoint.x++) {
|
|
|
|
if (getPathCell(iteratorPoint) == kPathCellBarrier) {
|
|
|
|
addDebugPoint(iteratorPoint, 24);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
2004-12-28 21:27:18 +00:00
|
|
|
|
2004-12-29 21:49:25 +00:00
|
|
|
if (scanPathLine(fromPoint, toPoint)) {
|
2005-01-05 19:03:49 +00:00
|
|
|
actor->addWalkStepPoint(fromPoint);
|
|
|
|
actor->addWalkStepPoint(toPoint);
|
2004-12-28 21:27:18 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-12-29 21:49:25 +00:00
|
|
|
i = fillPathArray(fromPoint, toPoint, bestPoint);
|
2004-12-28 21:27:18 +00:00
|
|
|
|
2005-01-06 14:32:14 +00:00
|
|
|
#if 0
|
|
|
|
{
|
|
|
|
Point iteratorPoint;
|
|
|
|
int cellValue;
|
|
|
|
FILE *fp = fopen("d:\\FINDPATH.DAT", "w");
|
|
|
|
char c;
|
|
|
|
|
|
|
|
fprintf(fp, "from = (%d,%d)\n", fromPoint.x, fromPoint.y);
|
|
|
|
fprintf(fp, "to = (%d,%d)\n", toPoint.x, toPoint.y);
|
|
|
|
fprintf(fp, "bestPoint = (%d,%d)\n", bestPoint.x, bestPoint.y);
|
|
|
|
|
|
|
|
for (iteratorPoint.y = 0; iteratorPoint.y < _yCellCount; iteratorPoint.y++) {
|
|
|
|
for (iteratorPoint.x = 0; iteratorPoint.x < _xCellCount; iteratorPoint.x++) {
|
|
|
|
cellValue = getPathCell(iteratorPoint);
|
|
|
|
c = (cellValue < 0) ? ' ' : (cellValue == kPathCellBarrier) ? kPathCellBarrier : (cellValue < 8) ? "^>v<?jLP"[cellValue] : '.';
|
|
|
|
putc(c, fp);
|
|
|
|
}
|
|
|
|
putc('\n', fp);
|
|
|
|
}
|
|
|
|
fclose(fp);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2004-12-29 21:49:25 +00:00
|
|
|
if (fromPoint == bestPoint) {
|
2005-01-05 19:03:49 +00:00
|
|
|
actor->addWalkStepPoint(bestPoint);
|
2004-12-28 21:27:18 +00:00
|
|
|
return;
|
|
|
|
}
|
2004-12-29 21:49:25 +00:00
|
|
|
|
2005-01-06 21:57:10 +00:00
|
|
|
if (i == 0) {
|
|
|
|
error("fillPathArray returns zero");
|
|
|
|
}
|
|
|
|
|
|
|
|
setActorPath(actor, fromPoint, bestPoint);
|
2004-12-28 21:27:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool Actor::scanPathLine(const Point &point1, const Point &point2) {
|
|
|
|
Point point;
|
|
|
|
Point delta;
|
|
|
|
bool interchange = false;
|
|
|
|
Point fDelta;
|
|
|
|
int errterm;
|
|
|
|
int s1;
|
|
|
|
int s2;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
point = point1;
|
|
|
|
delta.x = ABS(point1.x - point2.x);
|
|
|
|
delta.y = ABS(point1.y - point2.y);
|
|
|
|
s1 = integerCompare(point2.x, point1.x);
|
|
|
|
s2 = integerCompare(point2.y, point1.y);
|
|
|
|
|
|
|
|
if (delta.y > delta.x) {
|
|
|
|
SWAP(delta.y, delta.x);
|
|
|
|
interchange = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
fDelta.x = delta.x * 2;
|
|
|
|
fDelta.y = delta.y * 2;
|
|
|
|
|
|
|
|
errterm = fDelta.y - delta.x;
|
|
|
|
|
|
|
|
for (i = 0; i < delta.x; i++) {
|
|
|
|
while (errterm >= 0) {
|
|
|
|
if (interchange) {
|
|
|
|
point.x += s1;
|
|
|
|
} else {
|
|
|
|
point.y += s2;
|
|
|
|
}
|
|
|
|
errterm -= fDelta.x;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (interchange)
|
|
|
|
point.y += s2;
|
|
|
|
else
|
|
|
|
point.x += s1;
|
|
|
|
|
|
|
|
errterm += fDelta.y;
|
|
|
|
|
2005-01-05 19:03:49 +00:00
|
|
|
if (!validPathCellPoint(point)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (getPathCell(point) == kPathCellBarrier) {
|
|
|
|
return false;
|
2005-01-04 18:54:29 +00:00
|
|
|
}
|
2004-12-28 21:27:18 +00:00
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2004-12-29 21:49:25 +00:00
|
|
|
int Actor::fillPathArray(const Point &fromPoint, const Point &toPoint, Point &bestPoint) {
|
2004-12-28 21:27:18 +00:00
|
|
|
int bestRating;
|
|
|
|
int currentRating;
|
|
|
|
Point bestPath;
|
|
|
|
int pointCounter;
|
|
|
|
int startDirection;
|
|
|
|
PathDirectionList pathDirectionList;
|
|
|
|
PathDirectionData *pathDirection;
|
|
|
|
PathDirectionData *samplePathDirection;
|
|
|
|
PathDirectionList::iterator pathDirectionIterator;
|
|
|
|
PathDirectionList::iterator newPathDirectionIterator;
|
|
|
|
int directionCount;
|
|
|
|
|
|
|
|
pointCounter = 0;
|
2005-01-05 19:03:49 +00:00
|
|
|
bestRating = quickDistance(fromPoint, toPoint);
|
|
|
|
bestPath = fromPoint;
|
2004-12-28 21:27:18 +00:00
|
|
|
|
|
|
|
for (startDirection = 0; startDirection < 4; startDirection++) {
|
|
|
|
newPathDirectionIterator = pathDirectionList.pushBack();
|
|
|
|
pathDirection = newPathDirectionIterator.operator->();
|
2005-01-05 19:03:49 +00:00
|
|
|
pathDirection->x = fromPoint.x;
|
|
|
|
pathDirection->y = fromPoint.y;
|
2004-12-28 21:27:18 +00:00
|
|
|
pathDirection->direction = startDirection;
|
|
|
|
}
|
2005-01-05 19:03:49 +00:00
|
|
|
|
|
|
|
if (validPathCellPoint(fromPoint)) {
|
|
|
|
setPathCell(fromPoint, 0);
|
2005-01-06 19:15:01 +00:00
|
|
|
|
2005-01-06 21:57:10 +00:00
|
|
|
addDebugPoint(fromPoint, 24+36);
|
2005-01-05 19:03:49 +00:00
|
|
|
}
|
2004-12-28 21:27:18 +00:00
|
|
|
|
|
|
|
pathDirectionIterator = pathDirectionList.begin();
|
|
|
|
|
|
|
|
do {
|
|
|
|
pathDirection = pathDirectionIterator.operator->();
|
2005-01-06 14:32:14 +00:00
|
|
|
for (directionCount = 0; directionCount < 3; directionCount++) {
|
2004-12-28 21:27:18 +00:00
|
|
|
samplePathDirection = &pathDirectionLUT[pathDirection->direction][directionCount];
|
|
|
|
Point nextPoint;
|
|
|
|
nextPoint.x = samplePathDirection->x + pathDirection->x;
|
|
|
|
nextPoint.y = samplePathDirection->y + pathDirection->y;
|
2005-01-05 19:03:49 +00:00
|
|
|
if (validPathCellPoint(nextPoint) &&
|
|
|
|
(getPathCell(nextPoint) == kPathCellEmpty)) {
|
2004-12-28 21:27:18 +00:00
|
|
|
setPathCell(nextPoint, samplePathDirection->direction);
|
2005-01-06 19:15:01 +00:00
|
|
|
|
|
|
|
addDebugPoint(nextPoint, samplePathDirection->direction + 96);
|
2004-12-28 21:27:18 +00:00
|
|
|
newPathDirectionIterator = pathDirectionList.pushBack();
|
|
|
|
pathDirection = newPathDirectionIterator.operator->();
|
|
|
|
pathDirection->x = nextPoint.x;
|
|
|
|
pathDirection->y = nextPoint.y;
|
|
|
|
pathDirection->direction = samplePathDirection->direction;
|
|
|
|
++pointCounter;
|
2005-01-05 19:03:49 +00:00
|
|
|
if (nextPoint == toPoint) {
|
2005-01-06 14:32:14 +00:00
|
|
|
bestPoint = toPoint;
|
2004-12-28 21:27:18 +00:00
|
|
|
return pointCounter;
|
|
|
|
}
|
2005-01-05 19:03:49 +00:00
|
|
|
currentRating = quickDistance(nextPoint, toPoint);
|
2004-12-28 21:27:18 +00:00
|
|
|
if (currentRating < bestRating) {
|
|
|
|
bestRating = currentRating;
|
|
|
|
bestPath = nextPoint;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
++pathDirectionIterator;
|
|
|
|
} while (pathDirectionIterator != pathDirectionList.end());
|
|
|
|
|
2005-01-05 19:03:49 +00:00
|
|
|
bestPoint = bestPath;
|
2004-12-28 21:27:18 +00:00
|
|
|
return pointCounter;
|
|
|
|
}
|
|
|
|
|
2004-12-30 18:45:48 +00:00
|
|
|
void Actor::setActorPath(ActorData *actor, const Point &fromPoint, const Point &toPoint) {
|
2004-12-29 21:49:25 +00:00
|
|
|
Point nextPoint;
|
|
|
|
int direction;
|
2005-01-06 21:57:10 +00:00
|
|
|
int i;
|
2004-12-29 21:49:25 +00:00
|
|
|
|
2005-01-06 21:57:10 +00:00
|
|
|
_pathListIndex = -1;
|
|
|
|
addPathListPoint(toPoint);
|
2005-01-05 19:03:49 +00:00
|
|
|
nextPoint = toPoint;
|
2004-12-29 21:49:25 +00:00
|
|
|
|
2005-01-05 19:03:49 +00:00
|
|
|
while ( !(nextPoint == fromPoint)) {
|
2004-12-29 21:49:25 +00:00
|
|
|
direction = getPathCell(nextPoint);
|
2005-01-05 19:03:49 +00:00
|
|
|
if ((direction < 0) || (direction > 8)) {
|
2005-01-06 14:32:14 +00:00
|
|
|
error("Actor::setActorPath error direction 0x%X", direction);
|
2005-01-05 19:03:49 +00:00
|
|
|
}
|
2004-12-29 21:49:25 +00:00
|
|
|
nextPoint.x -= pathDirectionLUT2[direction][0];
|
|
|
|
nextPoint.y -= pathDirectionLUT2[direction][1];
|
2005-01-06 21:57:10 +00:00
|
|
|
addPathListPoint(nextPoint);
|
|
|
|
|
|
|
|
addDebugPoint(nextPoint, 0x8a);
|
2004-12-29 21:49:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pathToNode();
|
|
|
|
|
2005-01-05 19:03:49 +00:00
|
|
|
removeNodes();
|
2004-12-29 21:49:25 +00:00
|
|
|
|
|
|
|
nodeToPath();
|
|
|
|
|
|
|
|
removePathPoints();
|
|
|
|
|
2005-01-06 21:57:10 +00:00
|
|
|
|
|
|
|
for (i = 0; i <= _pathNodeListIndex; i++) {
|
|
|
|
actor->addWalkStepPoint(_pathNodeList[i].point);
|
2004-12-29 21:49:25 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Actor::pathToNode() {
|
|
|
|
Point point1, point2, delta;
|
|
|
|
int direction;
|
|
|
|
int i;
|
|
|
|
Point *point;
|
|
|
|
|
|
|
|
point= &_pathList[_pathListIndex];
|
2004-12-29 22:00:47 +00:00
|
|
|
direction = 0;
|
2004-12-29 21:49:25 +00:00
|
|
|
|
2005-01-06 21:57:10 +00:00
|
|
|
_pathNodeListIndex = -1;
|
|
|
|
addPathNodeListPoint(*point);
|
2004-12-29 21:49:25 +00:00
|
|
|
|
|
|
|
for (i = _pathListIndex; i > 0; i--) {
|
|
|
|
point1 = *point;
|
|
|
|
--point;
|
|
|
|
point2 = *point;
|
|
|
|
if (direction == 0) {
|
|
|
|
delta.x = integerCompare(point2.x, point1.x);
|
|
|
|
delta.y = integerCompare(point2.y, point1.y);
|
|
|
|
direction++;
|
|
|
|
}
|
|
|
|
if ((point1.x + delta.x != point2.x) || (point1.y + delta.y != point2.y)) {
|
2005-01-06 21:57:10 +00:00
|
|
|
addPathNodeListPoint(point1);
|
2004-12-29 21:49:25 +00:00
|
|
|
direction--;
|
|
|
|
i++;
|
|
|
|
point++;
|
|
|
|
}
|
|
|
|
}
|
2005-01-06 21:57:10 +00:00
|
|
|
addPathNodeListPoint(*_pathList);
|
2004-12-29 21:49:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int pathLine(Point *pointList, const Point &point1, const Point &point2) {
|
|
|
|
Point point;
|
|
|
|
Point delta;
|
|
|
|
Point tempPoint;
|
|
|
|
int s1;
|
|
|
|
int s2;
|
|
|
|
bool interchange = false;
|
|
|
|
int errterm;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
delta.x = abs(point2.x - point1.x);
|
|
|
|
delta.y = abs(point2.y - point1.y);
|
|
|
|
point = point1;
|
|
|
|
s1 = integerCompare(point2.x, point1.x);
|
|
|
|
s2 = integerCompare(point2.y, point1.y);
|
|
|
|
|
|
|
|
if (delta.y > delta.x) {
|
|
|
|
SWAP(delta.y, delta.x);
|
|
|
|
interchange = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
tempPoint.x = delta.x * 2;
|
|
|
|
tempPoint.y = delta.y * 2;
|
|
|
|
|
|
|
|
errterm = tempPoint.y - delta.x;
|
|
|
|
|
|
|
|
for (i = 0; i < delta.x; i++) {
|
|
|
|
while (errterm >= 0) {
|
|
|
|
if (interchange) {
|
|
|
|
point.x += s1;
|
|
|
|
} else {
|
|
|
|
point.y += s2;
|
|
|
|
}
|
|
|
|
errterm -= tempPoint.x;
|
|
|
|
}
|
|
|
|
if (interchange) {
|
|
|
|
point.y += s2;
|
|
|
|
} else {
|
|
|
|
point.x += s1;
|
|
|
|
}
|
|
|
|
errterm += tempPoint.y;
|
|
|
|
|
|
|
|
pointList[i] = point;
|
|
|
|
}
|
|
|
|
return delta.x;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Actor::nodeToPath() {
|
|
|
|
int i;
|
|
|
|
Point point1, point2;
|
|
|
|
PathNode *node;
|
|
|
|
Point *point;
|
|
|
|
|
2005-01-06 21:57:10 +00:00
|
|
|
for (i = 0, point = _pathList; i < _pathListAlloced; i++, point++) {
|
2004-12-29 21:49:25 +00:00
|
|
|
point->x = point->y = PATH_NODE_EMPTY;
|
|
|
|
}
|
|
|
|
|
|
|
|
_pathListIndex = 1;
|
2005-01-06 21:57:10 +00:00
|
|
|
_pathList[0] = _pathNodeList[0].point;
|
2004-12-29 21:49:25 +00:00
|
|
|
_pathNodeList[0].link = 0;
|
2005-01-06 21:57:10 +00:00
|
|
|
for (i = 0, node = _pathNodeList; i < _pathNodeListIndex; i++) {
|
|
|
|
point1 = node->point;
|
2004-12-29 21:49:25 +00:00
|
|
|
node++;
|
2005-01-06 21:57:10 +00:00
|
|
|
point2 = node->point;
|
2004-12-29 21:49:25 +00:00
|
|
|
_pathListIndex += pathLine(&_pathList[_pathListIndex], point1, point2);
|
|
|
|
node->link = _pathListIndex - 1;
|
|
|
|
}
|
|
|
|
_pathListIndex--;
|
2005-01-06 21:57:10 +00:00
|
|
|
_pathNodeList[_pathNodeListIndex].link = _pathListIndex;
|
2004-12-29 21:49:25 +00:00
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
void Actor::removeNodes() {
|
|
|
|
int i, j, k;
|
|
|
|
PathNode *iNode, *jNode, *kNode, *fNode;
|
2005-01-06 21:57:10 +00:00
|
|
|
fNode = &_pathNodeList[_pathNodeListIndex];
|
2004-12-29 21:49:25 +00:00
|
|
|
|
2005-01-06 21:57:10 +00:00
|
|
|
if (scanPathLine(_pathNodeList[0].point, fNode->point)) {
|
2004-12-29 21:49:25 +00:00
|
|
|
_pathNodeList[1] = *fNode;
|
2005-01-06 21:57:10 +00:00
|
|
|
_pathNodeListIndex = 1;
|
2004-12-29 21:49:25 +00:00
|
|
|
}
|
|
|
|
|
2005-01-06 21:57:10 +00:00
|
|
|
if (_pathNodeListIndex < 4) {
|
2004-12-29 21:49:25 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2005-01-06 21:57:10 +00:00
|
|
|
for (i = _pathNodeListIndex - 1, iNode = fNode-1; i > 1 ; i--, iNode--) {
|
|
|
|
if (iNode->point.x == PATH_NODE_EMPTY) {
|
2004-12-29 21:49:25 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2005-01-06 21:57:10 +00:00
|
|
|
if (scanPathLine(_pathNodeList[0].point, iNode->point)) {
|
2004-12-29 21:49:25 +00:00
|
|
|
for (j = 1, jNode = _pathNodeList + 1; j < i; j++, jNode++) {
|
2005-01-06 21:57:10 +00:00
|
|
|
jNode->point.x = PATH_NODE_EMPTY;
|
2004-12-29 21:49:25 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2005-01-06 21:57:10 +00:00
|
|
|
for (i = 1, iNode = _pathNodeList + 1; i < _pathNodeListIndex - 1; i++, iNode++) {
|
|
|
|
if (iNode->point.x == PATH_NODE_EMPTY) {
|
2004-12-29 21:49:25 +00:00
|
|
|
continue;
|
|
|
|
}
|
2005-01-06 21:57:10 +00:00
|
|
|
|
|
|
|
if (scanPathLine(fNode->point, iNode->point)) {
|
|
|
|
for (j = i + 1, jNode = iNode + 1; j < _pathNodeListIndex; j++, jNode++) {
|
|
|
|
jNode->point.x = PATH_NODE_EMPTY;
|
2004-12-29 21:49:25 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
condenseNodeList();
|
|
|
|
|
2005-01-06 21:57:10 +00:00
|
|
|
for (i = 1, iNode = _pathNodeList + 1; i < _pathNodeListIndex - 1; i++, iNode++) {
|
|
|
|
if (iNode->point.x == PATH_NODE_EMPTY) {
|
2004-12-29 21:49:25 +00:00
|
|
|
continue;
|
|
|
|
}
|
2005-01-06 21:57:10 +00:00
|
|
|
for (j = i + 2, jNode = iNode + 2; j < _pathNodeListIndex; j++, jNode++)
|
2004-12-29 21:49:25 +00:00
|
|
|
{
|
2005-01-06 21:57:10 +00:00
|
|
|
if (jNode->point.x == PATH_NODE_EMPTY) {
|
2004-12-29 21:49:25 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2005-01-06 21:57:10 +00:00
|
|
|
if (scanPathLine(iNode->point, jNode->point)) {
|
2004-12-29 21:49:25 +00:00
|
|
|
for (k = i + 1,kNode = iNode + 1; k < j; k++, kNode++) {
|
2005-01-06 21:57:10 +00:00
|
|
|
kNode->point.x = PATH_NODE_EMPTY;
|
2004-12-29 21:49:25 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
condenseNodeList();
|
|
|
|
}
|
|
|
|
|
|
|
|
void Actor::condenseNodeList() {
|
|
|
|
int i, j, count;
|
|
|
|
PathNode *iNode, *jNode;
|
|
|
|
|
2005-01-06 21:57:10 +00:00
|
|
|
count = _pathNodeListIndex;
|
2004-12-29 21:49:25 +00:00
|
|
|
|
2005-01-06 21:57:10 +00:00
|
|
|
for (i = 1, iNode = _pathNodeList + 1; i < _pathNodeListIndex; i++, iNode++) {
|
|
|
|
if (iNode->point.x == PATH_NODE_EMPTY) {
|
2004-12-29 21:49:25 +00:00
|
|
|
j = i + 1;
|
|
|
|
jNode = iNode + 1;
|
2005-01-06 21:57:10 +00:00
|
|
|
while ( jNode->point.x == PATH_NODE_EMPTY ) {
|
2004-12-29 21:49:25 +00:00
|
|
|
j++;
|
|
|
|
jNode++;
|
|
|
|
}
|
|
|
|
*iNode = *jNode;
|
|
|
|
count = i;
|
2005-01-06 21:57:10 +00:00
|
|
|
jNode->point.x = PATH_NODE_EMPTY;
|
|
|
|
if (j == _pathNodeListIndex) {
|
2004-12-29 21:49:25 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2005-01-06 21:57:10 +00:00
|
|
|
_pathNodeListIndex = count;
|
2004-12-29 21:49:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void Actor::removePathPoints() {
|
|
|
|
int i, j, k, l;
|
|
|
|
PathNode *node;
|
|
|
|
int newPathNodeIndex;
|
|
|
|
int start;
|
|
|
|
int end;
|
|
|
|
Point point1, point2, point3, point4;
|
|
|
|
|
|
|
|
|
2005-01-06 21:57:10 +00:00
|
|
|
if (_pathNodeListIndex < 2)
|
2004-12-29 21:49:25 +00:00
|
|
|
return;
|
|
|
|
|
2005-01-06 21:57:10 +00:00
|
|
|
_newPathNodeList = (PathNode*)realloc(_newPathNodeList, _pathNodeListAlloced);
|
2004-12-29 21:49:25 +00:00
|
|
|
_newPathNodeList[0] = _pathNodeList[0];
|
|
|
|
newPathNodeIndex = 0;
|
|
|
|
|
2005-01-06 21:57:10 +00:00
|
|
|
for (i = 1, node = _pathNodeList + 1; i < _pathNodeListIndex; i++, node++) {
|
2004-12-29 21:49:25 +00:00
|
|
|
newPathNodeIndex++;
|
|
|
|
_newPathNodeList[newPathNodeIndex] = *node;
|
|
|
|
|
|
|
|
for (j = 5; j > 0; j--) {
|
|
|
|
start = node->link - j;
|
|
|
|
end = node->link + j;
|
|
|
|
|
|
|
|
if (start < 0 || end > _pathListIndex) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
point1.x = _pathList[start].x;
|
|
|
|
if (point1.x == PATH_NODE_EMPTY) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
point2.x = _pathList[end].x;
|
|
|
|
if (point2.x == PATH_NODE_EMPTY) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
point1.y = _pathList[start].y;
|
|
|
|
point2.y = _pathList[end].y;
|
|
|
|
|
2005-01-05 19:03:49 +00:00
|
|
|
point3.x = point1.x;
|
2004-12-29 21:49:25 +00:00
|
|
|
point3.y = point1.y;
|
2005-01-05 19:03:49 +00:00
|
|
|
point4.x = point2.x;
|
2004-12-29 21:49:25 +00:00
|
|
|
point4.y = point2.y;
|
|
|
|
if (scanPathLine( point3, point4)) {
|
|
|
|
for (l = 1; l <= newPathNodeIndex; l++) {
|
|
|
|
if (start <= _newPathNodeList[l].link) {
|
|
|
|
newPathNodeIndex = l;
|
2005-01-06 21:57:10 +00:00
|
|
|
_newPathNodeList[newPathNodeIndex].point = point1;
|
2004-12-29 21:49:25 +00:00
|
|
|
_newPathNodeList[newPathNodeIndex].link = start;
|
|
|
|
newPathNodeIndex++;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2005-01-06 21:57:10 +00:00
|
|
|
_newPathNodeList[newPathNodeIndex].point = point2;
|
2004-12-29 21:49:25 +00:00
|
|
|
_newPathNodeList[newPathNodeIndex].link = end;
|
|
|
|
|
|
|
|
for (k = start + 1; k < end; k++) {
|
|
|
|
_pathList[k].x = PATH_NODE_EMPTY;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
newPathNodeIndex++;
|
2005-01-06 21:57:10 +00:00
|
|
|
_newPathNodeList[newPathNodeIndex] = _pathNodeList[_pathNodeListIndex];
|
2004-12-29 21:49:25 +00:00
|
|
|
|
|
|
|
for (i = 0, j = 0; i <= newPathNodeIndex; i++) {
|
2005-01-06 21:57:10 +00:00
|
|
|
if (newPathNodeIndex == i || (_newPathNodeList[i].point != _newPathNodeList[i+1].point)) {
|
2004-12-29 21:49:25 +00:00
|
|
|
_pathNodeList[j++] = _newPathNodeList[i];
|
2005-01-06 21:57:10 +00:00
|
|
|
}
|
2004-12-29 21:49:25 +00:00
|
|
|
}
|
2005-01-06 21:57:10 +00:00
|
|
|
_pathNodeListIndex = j - 1;
|
2004-12-29 21:49:25 +00:00
|
|
|
}
|
|
|
|
|
2005-01-06 19:15:01 +00:00
|
|
|
void Actor::drawPathTest() {
|
|
|
|
int i;
|
|
|
|
SURFACE *surface;
|
|
|
|
surface = _vm->_gfx->getBackBuffer();
|
|
|
|
if (_debugPoints == NULL) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
for (i = 0; i < _debugPointsCount; i++) {
|
|
|
|
*((byte *)surface->pixels + (_debugPoints[i].point.y * surface->pitch) + _debugPoints[i].point.x) = _debugPoints[i].color;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2004-12-22 21:04:50 +00:00
|
|
|
/*
|
2004-12-17 11:18:56 +00:00
|
|
|
// Console wrappers - must be safe to run
|
|
|
|
// TODO - checkup ALL arguments, cause wrong arguments may fall function with "error"
|
2004-04-12 21:40:49 +00:00
|
|
|
|
2004-12-03 19:15:44 +00:00
|
|
|
void Actor::CF_actor_move(int argc, const char **argv) {
|
2004-12-17 11:18:56 +00:00
|
|
|
uint16 actorId = (uint16) atoi(argv[1]);
|
2004-12-17 20:38:17 +00:00
|
|
|
Point movePoint;
|
2004-04-12 21:40:49 +00:00
|
|
|
|
2004-12-17 20:38:17 +00:00
|
|
|
movePoint.x = atoi(argv[2]);
|
|
|
|
movePoint.y = atoi(argv[3]);
|
2004-04-12 21:40:49 +00:00
|
|
|
|
2004-12-17 20:38:17 +00:00
|
|
|
if (!IS_VALID_ACTOR_ID(actorId)) {
|
|
|
|
_vm->_console->DebugPrintf("Actor::CF_actor_move Invalid actorId 0x%X.\n", actorId);
|
2004-12-17 11:18:56 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2004-12-17 20:38:17 +00:00
|
|
|
move(actorId, movePoint);
|
2004-04-12 21:40:49 +00:00
|
|
|
}
|
|
|
|
|
2004-12-03 19:15:44 +00:00
|
|
|
void Actor::CF_actor_moverel(int argc, const char **argv) {
|
2004-12-17 11:18:56 +00:00
|
|
|
uint16 actorId = (uint16) atoi(argv[1]);
|
2004-12-17 20:38:17 +00:00
|
|
|
Point movePoint;
|
2004-04-12 21:40:49 +00:00
|
|
|
|
2004-12-17 20:38:17 +00:00
|
|
|
movePoint.x = atoi(argv[2]);
|
|
|
|
movePoint.y = atoi(argv[3]);
|
2004-04-12 21:40:49 +00:00
|
|
|
|
2004-12-17 20:38:17 +00:00
|
|
|
if (!IS_VALID_ACTOR_ID(actorId)) {
|
|
|
|
_vm->_console->DebugPrintf("Actor::CF_actor_moverel Invalid actorId 0x%X.\n", actorId);
|
2004-12-17 11:18:56 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2004-12-17 20:38:17 +00:00
|
|
|
moveRelative(actorId, movePoint);
|
2004-04-12 21:40:49 +00:00
|
|
|
}
|
|
|
|
|
2004-12-03 19:15:44 +00:00
|
|
|
void Actor::CF_actor_seto(int argc, const char **argv) {
|
2004-12-17 11:18:56 +00:00
|
|
|
uint16 actorId = (uint16) atoi(argv[1]);
|
2004-04-12 21:40:49 +00:00
|
|
|
int orient;
|
|
|
|
|
2004-12-03 19:15:44 +00:00
|
|
|
orient = atoi(argv[2]);
|
2004-12-17 11:18:56 +00:00
|
|
|
//TODO orient check
|
2004-12-17 20:38:17 +00:00
|
|
|
if (!IS_VALID_ACTOR_ID(actorId)) {
|
2004-12-17 11:18:56 +00:00
|
|
|
_vm->_console->DebugPrintf("Actor::CF_actor_seto Invalid actorId 0x%X.\n",actorId);
|
|
|
|
return;
|
|
|
|
}
|
2004-04-12 21:40:49 +00:00
|
|
|
|
|
|
|
}
|
|
|
|
|
2004-12-03 19:15:44 +00:00
|
|
|
void Actor::CF_actor_setact(int argc, const char **argv) {
|
2004-12-17 11:18:56 +00:00
|
|
|
uint16 actorId = (uint16) atoi(argv[1]);
|
2004-04-12 21:40:49 +00:00
|
|
|
int action_n = 0;
|
|
|
|
|
2004-12-03 19:15:44 +00:00
|
|
|
action_n = atoi(argv[2]);
|
2004-04-12 21:40:49 +00:00
|
|
|
|
2004-12-17 20:38:17 +00:00
|
|
|
if (!IS_VALID_ACTOR_ID(actorId)) {
|
2004-12-17 11:18:56 +00:00
|
|
|
_vm->_console->DebugPrintf("Actor::CF_actor_setact Invalid actorId 0x%X.\n",actorId);
|
2004-04-12 21:40:49 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2004-12-17 11:18:56 +00:00
|
|
|
//TODO action_n check
|
2004-12-22 21:04:50 +00:00
|
|
|
if ((action_n < 0) || (action_n >= actor->action_ct)) {
|
2004-12-03 19:15:44 +00:00
|
|
|
_vm->_console->DebugPrintf("Invalid action number.\n");
|
2004-04-12 21:40:49 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2004-12-03 19:15:44 +00:00
|
|
|
_vm->_console->DebugPrintf("Action frame counts: %d %d %d %d.\n",
|
2004-05-01 06:16:57 +00:00
|
|
|
actor->act_tbl[action_n].dir[0].frame_count,
|
|
|
|
actor->act_tbl[action_n].dir[1].frame_count,
|
|
|
|
actor->act_tbl[action_n].dir[2].frame_count,
|
|
|
|
actor->act_tbl[action_n].dir[3].frame_count);
|
2004-12-22 21:04:50 +00:00
|
|
|
|
2004-12-17 11:18:56 +00:00
|
|
|
setAction(actorId, action_n, ACTION_LOOP);
|
2004-04-12 21:40:49 +00:00
|
|
|
}
|
2004-12-22 21:04:50 +00:00
|
|
|
*/
|
2004-04-12 21:40:49 +00:00
|
|
|
} // End of namespace Saga
|