scummvm/engines/agi/motion.cpp

233 lines
5.3 KiB
C++
Raw Normal View History

2006-05-23 23:43:52 +00:00
/* ScummVM - Scumm Interpreter
* Copyright (C) 2006 The ScummVM project
*
* Copyright (C) 1999-2001 Sarien Team
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU General Public License
* as published by the Free Software Foundation; either version 2
* of the License, or (at your option) any later version.
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
*
* $URL$
* $Id$
*
*/
#include "common/stdafx.h"
#include "agi/agi.h"
namespace Agi {
int AgiEngine::checkStep(int delta, int step) {
2006-05-23 23:43:52 +00:00
return (-step >= delta) ? 0 : (step <= delta) ? 2 : 1;
}
int AgiEngine::checkBlock(int x, int y) {
if (x <= _game.block.x1 || x >= _game.block.x2)
2006-05-23 23:43:52 +00:00
return false;
if (y <= _game.block.y1 || y >= _game.block.y2)
2006-05-23 23:43:52 +00:00
return false;
return true;
}
void AgiEngine::changePos(struct VtEntry *v) {
2006-05-23 23:43:52 +00:00
int b, x, y;
int dx[9] = { 0, 0, 1, 1, 1, 0, -1, -1, -1 };
int dy[9] = { 0, -1, -1, 0, 1, 1, 1, 0, -1 };
x = v->xPos;
y = v->yPos;
b = checkBlock(x, y);
2006-05-23 23:43:52 +00:00
x += v->stepSize * dx[v->direction];
y += v->stepSize * dy[v->direction];
2006-05-23 23:43:52 +00:00
if (checkBlock(x, y) == b) {
2006-05-23 23:43:52 +00:00
v->flags &= ~MOTION;
} else {
v->flags |= MOTION;
v->direction = 0;
if (isEgoView(v))
_game.vars[vEgoDir] = 0;
2006-05-23 23:43:52 +00:00
}
}
void AgiEngine::motionWander(struct VtEntry *v) {
2006-05-23 23:43:52 +00:00
if (v->parm1--) {
if (~v->flags & DIDNT_MOVE)
return;
}
v->direction = _rnd->getRandomNumber(8);
2006-05-23 23:43:52 +00:00
if (isEgoView(v)) {
_game.vars[vEgoDir] = v->direction;
2006-05-23 23:43:52 +00:00
while (v->parm1 < 6) {
v->parm1 = _rnd->getRandomNumber(50); /* huh? */
2006-05-23 23:43:52 +00:00
}
}
}
void AgiEngine::motionFollowEgo(struct VtEntry *v) {
int egoX, egoY;
int objX, objY;
2006-05-23 23:43:52 +00:00
int dir;
egoX = _game.viewTable[0].xPos + _game.viewTable[0].xSize / 2;
egoY = _game.viewTable[0].yPos;
2006-05-23 23:43:52 +00:00
objX = v->xPos + v->xSize / 2;
objY = v->yPos;
2006-05-23 23:43:52 +00:00
/* Get direction to reach ego */
dir = getDirection(objX, objY, egoX, egoY, v->parm1);
2006-05-23 23:43:52 +00:00
/* Already at ego coordinates */
if (dir == 0) {
v->direction = 0;
v->motion = MOTION_NORMAL;
setflag(v->parm2, true);
return;
}
if (v->parm3 == 0xff) {
v->parm3 = 0;
} else if (v->flags & DIDNT_MOVE) {
int d;
while ((v->direction = _rnd->getRandomNumber(8)) == 0) {
2006-05-23 23:43:52 +00:00
}
d = (abs(egoY - objY) + abs(egoX - objX)) / 2;
2006-05-23 23:43:52 +00:00
if (d < v->stepSize) {
v->parm3 = v->stepSize;
2006-05-23 23:43:52 +00:00
return;
}
while ((v->parm3 = _rnd->getRandomNumber(d)) < v->stepSize) {
2006-05-23 23:43:52 +00:00
}
return;
}
if (v->parm3 != 0) {
int k;
/* DF: this is ugly and I dont know why this works, but
* other line does not! (watcom complained about lvalue)
*
* if (((int8)v->parm3 -= v->step_size) < 0)
* v->parm3 = 0;
*/
k = v->parm3;
k -= v->stepSize;
2006-05-23 23:43:52 +00:00
v->parm3 = k;
if ((int8) v->parm3 < 0)
v->parm3 = 0;
} else {
v->direction = dir;
}
}
void AgiEngine::motionMoveObj(struct VtEntry *v) {
v->direction = getDirection(v->xPos, v->yPos, v->parm1, v->parm2, v->stepSize);
2006-05-23 23:43:52 +00:00
/* Update V6 if ego */
if (isEgoView(v))
_game.vars[vEgoDir] = v->direction;
2006-05-23 23:43:52 +00:00
if (v->direction == 0)
inDestination(v);
2006-05-23 23:43:52 +00:00
}
void AgiEngine::checkMotion(struct VtEntry *v) {
2006-05-23 23:43:52 +00:00
switch (v->motion) {
case MOTION_WANDER:
motionWander(v);
2006-05-23 23:43:52 +00:00
break;
case MOTION_FOLLOW_EGO:
motionFollowEgo(v);
2006-05-23 23:43:52 +00:00
break;
case MOTION_MOVE_OBJ:
motionMoveObj(v);
2006-05-23 23:43:52 +00:00
break;
}
if ((_game.block.active && (~v->flags & IGNORE_BLOCKS)) && v->direction)
changePos(v);
2006-05-23 23:43:52 +00:00
}
/*
* Public functions
*/
/**
*
*/
void AgiEngine::checkAllMotions() {
struct VtEntry *v;
2006-05-23 23:43:52 +00:00
for (v = _game.viewTable; v < &_game.viewTable[MAX_VIEWTABLE]; v++) {
2006-05-23 23:43:52 +00:00
if ((v->flags & (ANIMATED | UPDATE | DRAWN)) == (ANIMATED | UPDATE | DRAWN)
&& v->stepTimeCount == 1) {
checkMotion(v);
2006-05-23 23:43:52 +00:00
}
}
}
/**
* Check if given entry is at destination point.
* This function is used to updated the flags of an object with move.obj
* type motion that * has reached its final destination coordinates.
* @param v Pointer to view table entry
*/
void AgiEngine::inDestination(struct VtEntry *v) {
2006-05-23 23:43:52 +00:00
if (v->motion == MOTION_MOVE_OBJ) {
v->stepSize = v->parm3;
2006-05-23 23:43:52 +00:00
setflag(v->parm4, true);
}
v->motion = MOTION_NORMAL;
if (isEgoView(v))
_game.playerControl = true;
2006-05-23 23:43:52 +00:00
}
/**
* Wrapper for static function motion_moveobj().
* This function is used by cmd_move_object() in the first motion cycle
* after setting the motion mode to MOTION_MOVE_OBJ.
* @param v Pointer to view table entry
*/
void AgiEngine::moveObj(struct VtEntry *v) {
motionMoveObj(v);
2006-05-23 23:43:52 +00:00
}
/**
* Get direction from motion coordinates
* This function gets the motion direction from the current and previous
* object coordinates and the step size.
* @param x0 Original x coordinate of the object
* @param y0 Original y coordinate of the object
* @param x x coordinate of the object
* @param y y coordinate of the object
* @param s step size
*/
int AgiEngine::getDirection(int x0, int y0, int x, int y, int s) {
int dirTable[9] = { 8, 1, 2, 7, 0, 3, 6, 5, 4 };
return dirTable[checkStep(x - x0, s) + 3 * checkStep(y - y0, s)];
2006-05-23 23:43:52 +00:00
}
} // End of namespace Agi