mirror of
https://github.com/libretro/scummvm.git
synced 2025-01-08 10:51:11 +00:00
2608 lines
66 KiB
C++
2608 lines
66 KiB
C++
/* ScummVM - Graphic Adventure Engine
|
|
*
|
|
* ScummVM is the legal property of its developers, whose names
|
|
* are too numerous to list here. Please refer to the COPYRIGHT
|
|
* file distributed with this source distribution.
|
|
*
|
|
* This program is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU General Public License
|
|
* as published by the Free Software Foundation; either version 2
|
|
* of the License, or (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program; if not, write to the Free Software
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
|
*
|
|
*/
|
|
|
|
#include "hdb/hdb.h"
|
|
#include "hdb/ai.h"
|
|
#include "hdb/ai-player.h"
|
|
#include "hdb/file-manager.h"
|
|
#include "hdb/gfx.h"
|
|
#include "hdb/input.h"
|
|
#include "hdb/lua-script.h"
|
|
#include "hdb/map.h"
|
|
#include "hdb/mpc.h"
|
|
#include "hdb/sound.h"
|
|
#include "hdb/window.h"
|
|
|
|
namespace HDB {
|
|
|
|
AIEntity *AI::spawn(AIType type, AIDir dir, int x, int y, const char *funcInit, const char *funcAction, const char *funcUse, AIDir dir2, int level, int value1, int value2, int callInit) {
|
|
AIEntity *e = new AIEntity;
|
|
|
|
e->type = type;
|
|
e->dir = dir;
|
|
|
|
// Set Co-ordinates & Speed
|
|
e->x = x * kTileWidth;
|
|
e->tileX = x;
|
|
e->y = y * kTileHeight;
|
|
e->tileY = y;
|
|
e->moveSpeed = kPlayerMoveSpeed; // Default Speed
|
|
if (!g_hdb->getActionMode())
|
|
e->moveSpeed /= 2;
|
|
|
|
// Other variables
|
|
e->dir2 = dir2;
|
|
if (!level)
|
|
level = 1;
|
|
e->level = level;
|
|
e->value1 = value1;
|
|
e->value2 = value2;
|
|
e->animCycle = 2; // Game frames to wait before animating graphic frames
|
|
e->animDelay = e->animCycle;
|
|
e->animFrame = 0;
|
|
|
|
if (funcInit)
|
|
Common::strlcpy(e->luaFuncInit, funcInit, 32);
|
|
|
|
if (funcAction)
|
|
Common::strlcpy(e->luaFuncAction, funcAction, 32);
|
|
|
|
if (funcUse)
|
|
Common::strlcpy(e->luaFuncUse, funcUse, 32);
|
|
|
|
if (e->luaFuncInit[0] == '*')
|
|
e->luaFuncInit[0] = 0;
|
|
if (e->luaFuncAction[0] == '*')
|
|
e->luaFuncAction[0] = 0;
|
|
if (e->luaFuncUse[0] == '*')
|
|
e->luaFuncUse[0] = 0;
|
|
|
|
e->standdownFrames = e->standupFrames = e->standleftFrames = e->standrightFrames = 0;
|
|
e->movedownFrames = e->moveupFrames = e->moveleftFrames = e->moverightFrames = 0;
|
|
e->blinkFrames = 0;
|
|
|
|
if (!cacheEntGfx(e, (bool)callInit))
|
|
return nullptr;
|
|
else
|
|
_ents->push_back(e);
|
|
|
|
return e;
|
|
}
|
|
|
|
bool AI::cacheEntGfx(AIEntity *e, bool initFlag) {
|
|
int i = 0;
|
|
while (true) {
|
|
if (aiEntList[i].type == END_AI_TYPES)
|
|
return false;
|
|
|
|
// Load Gfx for corresponding Entity
|
|
if (aiEntList[i].type == e->type) {
|
|
int j = 0;
|
|
AIStateDef *list = aiEntList[i].stateDef;
|
|
|
|
while (list[j].state != STATE_ENDSTATES) {
|
|
|
|
Common::Array<const char *> *gfxFiles = g_hdb->_fileMan->findFiles(list[j].name, TYPE_TILE32);
|
|
uint32 size;
|
|
|
|
if (gfxFiles->size() == 0)
|
|
warning("AI::cacheEntGfx: no files for %s", list[j].name);
|
|
|
|
for (Common::Array<const char *>::iterator it = gfxFiles->begin(); it != gfxFiles->end(); ++it) {
|
|
size = g_hdb->_fileMan->getLength((*it), TYPE_TILE32);
|
|
|
|
if (g_hdb->_gfx->selectGfxType((*it))) {
|
|
Tile *gfx = g_hdb->_gfx->getTileGfx((*it), size);
|
|
|
|
switch (list[j].state) {
|
|
case STATE_STANDDOWN:
|
|
e->standdownGfx[e->standdownFrames] = gfx;
|
|
e->standdownFrames++;
|
|
break;
|
|
case STATE_STANDUP:
|
|
e->standupGfx[e->standupFrames] = gfx;
|
|
e->standupFrames++;
|
|
break;
|
|
case STATE_STANDLEFT:
|
|
e->standleftGfx[e->standleftFrames] = gfx;
|
|
e->standleftFrames++;
|
|
break;
|
|
case STATE_STANDRIGHT:
|
|
e->standrightGfx[e->standrightFrames] = gfx;
|
|
e->standrightFrames++;
|
|
break;
|
|
case STATE_BLINK:
|
|
e->blinkGfx[e->blinkFrames] = gfx;
|
|
e->blinkFrames++;
|
|
break;
|
|
case STATE_MOVEDOWN:
|
|
e->movedownGfx[e->movedownFrames] = gfx;
|
|
e->movedownFrames++;
|
|
break;
|
|
case STATE_MOVEUP:
|
|
e->moveupGfx[e->moveupFrames] = gfx;
|
|
e->moveupFrames++;
|
|
break;
|
|
case STATE_MOVELEFT:
|
|
e->moveleftGfx[e->moveleftFrames] = gfx;
|
|
e->moveleftFrames++;
|
|
break;
|
|
case STATE_MOVERIGHT:
|
|
e->moverightGfx[e->moverightFrames] = gfx;
|
|
e->moverightFrames++;
|
|
break;
|
|
|
|
// Special Player Frames
|
|
case STATE_PUSHDOWN:
|
|
_pushdownGfx[_pushdownFrames] = gfx;
|
|
_pushdownFrames++;
|
|
break;
|
|
case STATE_PUSHUP:
|
|
_pushupGfx[_pushupFrames] = gfx;
|
|
_pushupFrames++;
|
|
break;
|
|
case STATE_PUSHLEFT:
|
|
_pushleftGfx[_pushleftFrames] = gfx;
|
|
_pushleftFrames++;
|
|
break;
|
|
case STATE_PUSHRIGHT:
|
|
_pushrightGfx[_pushrightFrames] = gfx;
|
|
_pushrightFrames++;
|
|
break;
|
|
case STATE_GRABUP:
|
|
_getGfx[DIR_UP] = gfx;
|
|
break;
|
|
case STATE_GRABDOWN:
|
|
_getGfx[DIR_DOWN] = gfx;
|
|
break;
|
|
case STATE_GRABLEFT:
|
|
_getGfx[DIR_LEFT] = gfx;
|
|
break;
|
|
case STATE_GRABRIGHT:
|
|
_getGfx[DIR_RIGHT] = gfx;
|
|
break;
|
|
|
|
case STATE_ATK_STUN_UP:
|
|
_stunUpGfx[_stunUpFrames] = gfx;
|
|
_stunUpFrames++;
|
|
break;
|
|
case STATE_ATK_STUN_DOWN:
|
|
_stunDownGfx[_stunDownFrames] = gfx;
|
|
_stunDownFrames++;
|
|
break;
|
|
case STATE_ATK_STUN_LEFT:
|
|
_stunLeftGfx[_stunLeftFrames] = gfx;
|
|
_stunLeftFrames++;
|
|
break;
|
|
case STATE_ATK_STUN_RIGHT:
|
|
_stunRightGfx[_stunRightFrames] = gfx;
|
|
_stunRightFrames++;
|
|
break;
|
|
|
|
case STATE_ATK_SLUG_UP:
|
|
_slugUpGfx[_slugUpFrames] = gfx;
|
|
_slugUpFrames++;
|
|
break;
|
|
case STATE_ATK_SLUG_DOWN:
|
|
_slugDownGfx[_slugDownFrames] = gfx;
|
|
_slugDownFrames++;
|
|
break;
|
|
case STATE_ATK_SLUG_LEFT:
|
|
_slugLeftGfx[_slugLeftFrames] = gfx;
|
|
_slugLeftFrames++;
|
|
break;
|
|
case STATE_ATK_SLUG_RIGHT:
|
|
_slugRightGfx[_slugRightFrames] = gfx;
|
|
_slugRightFrames++;
|
|
break;
|
|
|
|
// Maintenance Bot
|
|
case STATE_USEUP:
|
|
e->standupGfx[4] = gfx;
|
|
break;
|
|
case STATE_USEDOWN:
|
|
e->standdownGfx[4] = gfx;
|
|
break;
|
|
case STATE_USELEFT:
|
|
e->standleftGfx[4] = gfx;
|
|
break;
|
|
case STATE_USERIGHT:
|
|
e->standrightGfx[4] = gfx;
|
|
break;
|
|
|
|
// Death & Dying for Player
|
|
case STATE_DYING:
|
|
_dyingGfx[_dyingFrames] = gfx;
|
|
_dyingFrames++;
|
|
break;
|
|
case STATE_GOODJOB:
|
|
_goodjobGfx = gfx;
|
|
break;
|
|
|
|
case STATE_HORRIBLE1:
|
|
_horrible1Gfx[_horrible1Frames] = gfx;
|
|
_horrible1Frames++;
|
|
break;
|
|
case STATE_HORRIBLE2:
|
|
_horrible2Gfx[_horrible2Frames] = gfx;
|
|
_horrible2Frames++;
|
|
break;
|
|
case STATE_HORRIBLE3:
|
|
_horrible3Gfx[_horrible3Frames] = gfx;
|
|
_horrible3Frames++;
|
|
break;
|
|
case STATE_HORRIBLE4:
|
|
_horrible4Gfx[_horrible4Frames] = gfx;
|
|
_horrible4Frames++;
|
|
break;
|
|
case STATE_PLUMMET:
|
|
_plummetGfx[_plummetFrames] = gfx;
|
|
_plummetFrames++;
|
|
break;
|
|
|
|
// floating frames - overwrite "standup" info
|
|
case STATE_FLOATING:
|
|
e->blinkGfx[e->blinkFrames] = gfx;
|
|
e->blinkFrames++;
|
|
break;
|
|
|
|
// melted frames - go in the special area (lightbarrels)
|
|
// shocking frames - go in the special1 area (shockbots)
|
|
// exploding frames, same
|
|
case STATE_MELTED:
|
|
case STATE_SHOCKING:
|
|
case STATE_EXPLODING:
|
|
e->special1Gfx[e->special1Frames] = gfx;
|
|
e->special1Frames++;
|
|
break;
|
|
|
|
// ICEPUFF spawning states
|
|
case STATE_ICEP_PEEK:
|
|
e->blinkGfx[e->blinkFrames] = gfx;
|
|
e->blinkFrames++;
|
|
break;
|
|
case STATE_ICEP_APPEAR:
|
|
e->standupGfx[e->standupFrames] = gfx;
|
|
e->standupFrames++;
|
|
break;
|
|
case STATE_ICEP_THROWDOWN:
|
|
e->standdownGfx[e->standdownFrames] = gfx;
|
|
e->standdownFrames++;
|
|
break;
|
|
case STATE_ICEP_THROWRIGHT:
|
|
e->standrightGfx[e->standrightFrames] = gfx;
|
|
e->standrightFrames++;
|
|
break;
|
|
case STATE_ICEP_THROWLEFT:
|
|
e->standleftGfx[e->standleftFrames] = gfx;
|
|
e->standleftFrames++;
|
|
break;
|
|
case STATE_ICEP_DISAPPEAR:
|
|
e->special1Gfx[e->special1Frames] = gfx;
|
|
e->special1Frames++;
|
|
break;
|
|
|
|
// FATFROG spawning states
|
|
case STATE_LICKDOWN:
|
|
e->movedownGfx[e->movedownFrames] = gfx;
|
|
e->movedownFrames++;
|
|
break;
|
|
case STATE_LICKLEFT:
|
|
e->moveleftGfx[e->moveleftFrames] = gfx;
|
|
e->moveleftFrames++;
|
|
break;
|
|
case STATE_LICKRIGHT:
|
|
e->moverightGfx[e->moverightFrames] = gfx;
|
|
e->moverightFrames++;
|
|
break;
|
|
|
|
// MEERKAT spawning states
|
|
case STATE_MEER_MOVE:
|
|
e->standdownGfx[e->standdownFrames] = gfx;
|
|
e->standdownFrames++;
|
|
break;
|
|
case STATE_MEER_APPEAR:
|
|
e->standleftGfx[e->standleftFrames] = gfx;
|
|
e->standleftFrames++;
|
|
break;
|
|
case STATE_MEER_BITE:
|
|
e->standrightGfx[e->standrightFrames] = gfx;
|
|
e->standrightFrames++;
|
|
break;
|
|
case STATE_MEER_DISAPPEAR:
|
|
e->standupGfx[e->standupFrames] = gfx;
|
|
e->standupFrames++;
|
|
break;
|
|
case STATE_MEER_LOOK:
|
|
e->movedownGfx[e->movedownFrames] = gfx;
|
|
e->movedownFrames++;
|
|
break;
|
|
|
|
// DIVERTER spawning states
|
|
case STATE_DIVERTER_BL:
|
|
e->standdownGfx[e->standdownFrames] = gfx;
|
|
e->standdownFrames++;
|
|
break;
|
|
case STATE_DIVERTER_BR:
|
|
e->standupGfx[e->standupFrames] = gfx;
|
|
e->standupFrames++;
|
|
break;
|
|
case STATE_DIVERTER_TL:
|
|
e->standleftGfx[e->standleftFrames] = gfx;
|
|
e->standleftFrames++;
|
|
break;
|
|
case STATE_DIVERTER_TR:
|
|
e->standrightGfx[e->standrightFrames] = gfx;
|
|
e->standrightFrames++;
|
|
break;
|
|
// DOLLY states
|
|
// angry[4] = standright[4]
|
|
// kissright[4]/kissleft[4] = standleft[8]
|
|
// panic[4]/laugh[4] = standdown[8]
|
|
// dollyuseright[5] = special1[5]
|
|
case STATE_ANGRY:
|
|
e->standrightGfx[e->standrightFrames] = gfx;
|
|
e->standrightFrames++;
|
|
break;
|
|
case STATE_KISSRIGHT:
|
|
e->standleftGfx[e->standleftFrames] = gfx;
|
|
e->standleftFrames++;
|
|
break;
|
|
case STATE_KISSLEFT:
|
|
e->standleftGfx[4 + e->int1] = gfx;
|
|
e->int1++;
|
|
break;
|
|
case STATE_PANIC:
|
|
e->standdownGfx[e->standdownFrames] = gfx;
|
|
e->standdownFrames++;
|
|
break;
|
|
case STATE_LAUGH:
|
|
e->standdownGfx[4 + e->value1] = gfx;
|
|
e->value1++;
|
|
break;
|
|
case STATE_DOLLYUSERIGHT:
|
|
e->special1Gfx[e->special1Frames] = gfx;
|
|
e->special1Frames++;
|
|
break;
|
|
|
|
// SARGE yelling
|
|
case STATE_YELL:
|
|
e->special1Gfx[e->special1Frames] = gfx;
|
|
e->special1Frames++;
|
|
break;
|
|
default:
|
|
// no op
|
|
break;
|
|
}
|
|
} else {
|
|
Picture *gfx = g_hdb->_gfx->getPicGfx((*it), size);
|
|
|
|
switch (list[j].state) {
|
|
case STATE_ATK_CLUB_UP:
|
|
_clubUpGfx[_clubUpFrames] = gfx;
|
|
_clubUpFrames++;
|
|
break;
|
|
case STATE_ATK_CLUB_DOWN:
|
|
_clubDownGfx[_clubDownFrames] = gfx;
|
|
_clubDownFrames++;
|
|
break;
|
|
case STATE_ATK_CLUB_LEFT:
|
|
_clubLeftGfx[_clubLeftFrames] = gfx;
|
|
_clubLeftFrames++;
|
|
break;
|
|
case STATE_ATK_CLUB_RIGHT:
|
|
_clubRightGfx[_clubRightFrames] = gfx;
|
|
_clubRightFrames++;
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
j++;
|
|
|
|
delete gfxFiles;
|
|
}
|
|
|
|
e->aiInit = aiEntList[i].initFunc;
|
|
e->aiInit2 = aiEntList[i].initFunc2;
|
|
if (initFlag) {
|
|
e->aiInit(e);
|
|
if (e->aiInit2)
|
|
e->aiInit2(e);
|
|
|
|
if (e->luaFuncInit[0]) {
|
|
g_hdb->_lua->callFunction(e->luaFuncInit, 2);
|
|
|
|
const char *str1 = g_hdb->_lua->getStringOffStack();
|
|
const char *str2 = g_hdb->_lua->getStringOffStack();
|
|
if (str1)
|
|
Common::strlcpy(e->entityName, str1, 32);
|
|
|
|
if (str2)
|
|
Common::strlcpy(e->printedName, str2, 32);
|
|
}
|
|
} else if (e->aiInit2)
|
|
e->aiInit2(e);
|
|
|
|
break; // Entity Initiated
|
|
}
|
|
i++;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
// Stops the movement of an entity
|
|
void AI::stopEntity(AIEntity *e) {
|
|
if (e == _player) {
|
|
clearWaypoints();
|
|
// Reset Player speed
|
|
e->moveSpeed = kPlayerMoveSpeed;
|
|
}
|
|
|
|
// Reset animation
|
|
e->animFrame = 0;
|
|
|
|
// Align with tile boundaries
|
|
e->x = e->tileX * kTileWidth;
|
|
e->y = e->tileY * kTileHeight;
|
|
|
|
// TODO: Check in the original if also present. Removed as it's useless
|
|
// e->goalX = e->tileX;
|
|
// e->goalY = e->tileY;
|
|
|
|
e->drawXOff = e->drawYOff = 0;
|
|
e->goalX = e->goalY = e->xVel = e->yVel = 0;
|
|
|
|
// Don't change the state of Diverters or Floating entities
|
|
switch (e->state) {
|
|
case STATE_FLOATLEFT:
|
|
case STATE_FLOATRIGHT:
|
|
case STATE_FLOATUP:
|
|
case STATE_FLOATDOWN:
|
|
e->state = STATE_FLOATING;
|
|
return;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
if (e->type != AI_DIVERTER) {
|
|
switch (e->dir) {
|
|
case DIR_UP:
|
|
if (e->standupFrames)
|
|
e->state = STATE_STANDUP;
|
|
else
|
|
e->state = STATE_NONE;
|
|
break;
|
|
case DIR_DOWN:
|
|
if (e->standdownFrames)
|
|
e->state = STATE_STANDDOWN;
|
|
else
|
|
e->state = STATE_NONE;
|
|
break;
|
|
case DIR_LEFT:
|
|
if (e->standleftFrames)
|
|
e->state = STATE_STANDLEFT;
|
|
else
|
|
e->state = STATE_NONE;
|
|
break;
|
|
case DIR_RIGHT:
|
|
if (e->standrightFrames)
|
|
e->state = STATE_STANDRIGHT;
|
|
else
|
|
e->state = STATE_NONE;
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
AIEntity *AI::locateEntity(const char *luaName) {
|
|
for (Common::Array<AIEntity *>::iterator it = _ents->begin(); it != _ents->end(); ++it) {
|
|
if (Common::matchString((*it)->entityName, luaName))
|
|
return *it;
|
|
}
|
|
return nullptr;
|
|
}
|
|
|
|
AIEntity *AI::findEntity(int x, int y) {
|
|
for (Common::Array<AIEntity *>::iterator it = _ents->begin(); it != _ents->end(); ++it) {
|
|
if ((*it)->tileX == x && (*it)->tileY == y)
|
|
return *it;
|
|
}
|
|
|
|
for (Common::Array<AIEntity *>::iterator it = _floats->begin(); it != _floats->end(); ++it) {
|
|
if ((*it)->tileX == x && (*it)->tileY == y)
|
|
return *it;
|
|
}
|
|
|
|
if (g_hdb->_map->laserBeamExist(x, y))
|
|
return &_dummyLaser;
|
|
|
|
return nullptr;
|
|
}
|
|
|
|
AIEntity *AI::findEntityIgnore(int x, int y, AIEntity *ignore) {
|
|
for (Common::Array<AIEntity *>::iterator it = _ents->begin(); it != _ents->end(); ++it) {
|
|
if ((*it)->tileX == x && (*it)->tileY == y && (*it) != ignore)
|
|
return *it;
|
|
}
|
|
|
|
for (Common::Array<AIEntity *>::iterator it = _floats->begin(); it != _floats->end(); ++it) {
|
|
if ((*it)->tileX == x && (*it)->tileY == y && (*it) != ignore)
|
|
return *it;
|
|
}
|
|
|
|
if (g_hdb->_map->laserBeamExist(x, y) && ignore->type != AI_LASERBEAM)
|
|
return &_dummyLaser;
|
|
|
|
return nullptr;
|
|
}
|
|
|
|
AIEntity *AI::findEntityType(AIType type, int x, int y) {
|
|
for (Common::Array<AIEntity *>::iterator it = _ents->begin(); it != _ents->end(); ++it) {
|
|
if ((*it)->tileX == x && (*it)->tileY == y && (*it)->type == type)
|
|
return *it;
|
|
}
|
|
|
|
for (Common::Array<AIEntity *>::iterator it = _floats->begin(); it != _floats->end(); ++it) {
|
|
if ((*it)->tileX == x && (*it)->tileY == y && (*it)->type == type)
|
|
return *it;
|
|
}
|
|
|
|
if (g_hdb->_map->laserBeamExist(x, y) && type == AI_LASERBEAM)
|
|
return &_dummyLaser;
|
|
|
|
return nullptr;
|
|
}
|
|
|
|
void AI::getEntityXY(const char *entName, int *x, int *y) {
|
|
for (Common::Array<AIEntity *>::iterator it = _ents->begin(); it != _ents->end(); ++it) {
|
|
AIEntity *e = *it;
|
|
if (!scumm_stricmp(entName, e->entityName)) {
|
|
*x = e->tileX;
|
|
*y = e->tileY;
|
|
return;
|
|
}
|
|
}
|
|
|
|
for (Common::Array<AIEntity *>::iterator jt = _floats->begin(); jt != _floats->end(); ++jt) {
|
|
AIEntity *e = *jt;
|
|
if (!scumm_stricmp(entName, e->entityName)) {
|
|
*x = e->tileX;
|
|
*y = e->tileY;
|
|
return;
|
|
}
|
|
}
|
|
|
|
for (Common::Array<HereT *>::iterator kt = _hereList->begin(); kt != _hereList->end(); ++kt) {
|
|
HereT *h = *kt;
|
|
if (!scumm_stricmp(entName, h->entName)) {
|
|
*x = h->x;
|
|
*y = h->y;
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
bool AI::useLuaEntity(const char *initName) {
|
|
for (Common::Array<AIEntity *>::iterator it = _ents->begin(); it != _ents->end(); ++it) {
|
|
AIEntity *e = *it;
|
|
if (!scumm_stricmp(initName, e->entityName)) {
|
|
e->aiUse(e);
|
|
checkActionList(e, e->tileX, e->tileY, true);
|
|
if (e->luaFuncUse[0])
|
|
g_hdb->_lua->callFunction(e->luaFuncUse, 0);
|
|
return true;
|
|
}
|
|
}
|
|
|
|
// Check _actions list for activation as well
|
|
for (int i = 0; i < kMaxActions; i++) {
|
|
if (!scumm_stricmp(initName, _actions[i].entityName)) {
|
|
checkActionList(&_dummyPlayer, _actions[i].x1, _actions[i].y1, false);
|
|
checkActionList(&_dummyPlayer, _actions[i].x2, _actions[i].y2, false);
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
void AI::removeLuaEntity(const char *initName) {
|
|
for (uint i = 0; i < _ents->size(); i++) {
|
|
AIEntity *e = _ents->operator[](i);
|
|
if (!scumm_stricmp(initName, e->entityName)) {
|
|
removeEntity(e);
|
|
i--;
|
|
}
|
|
}
|
|
}
|
|
|
|
void AI::animLuaEntity(const char *initName, AIState st) {
|
|
for (Common::Array<AIEntity *>::iterator it = _ents->begin(); it != _ents->end(); ++it) {
|
|
AIEntity *e = *it;
|
|
if (!scumm_stricmp(initName, e->entityName)) {
|
|
e->state = st;
|
|
e->animFrame = 0;
|
|
e->animDelay = e->animCycle;
|
|
}
|
|
}
|
|
}
|
|
|
|
void AI::setLuaAnimFrame(const char *initName, AIState st, int frame) {
|
|
for (Common::Array<AIEntity *>::iterator it = _ents->begin(); it != _ents->end(); ++it) {
|
|
AIEntity *e = *it;
|
|
if (!scumm_stricmp(initName, e->entityName)) {
|
|
e->state = st;
|
|
e->animFrame = frame;
|
|
e->animDelay = e->animCycle;
|
|
animEntFrames(e);
|
|
e->state = STATE_NONE;
|
|
}
|
|
}
|
|
}
|
|
|
|
int AI::checkForTouchplate(int x, int y) {
|
|
int tileIndex = g_hdb->_map->getMapBGTileIndex(x, y);
|
|
if (tileIndex == _touchplateOff || tileIndex == _templeTouchpOff)
|
|
return tileIndex;
|
|
return 0;
|
|
}
|
|
|
|
void AI::removeEntity(AIEntity *e) {
|
|
for (uint i = 0; i < _ents->size(); i++) {
|
|
if (_ents->operator[](i) == e) {
|
|
delete _ents->operator[](i);
|
|
_ents->remove_at(i);
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
void AI::setEntityGoal(AIEntity *e, int x, int y) {
|
|
e->xVel = e->yVel = 0;
|
|
|
|
int xv = x - e->tileX;
|
|
if (xv < 0) {
|
|
e->xVel = -e->moveSpeed;
|
|
e->state = STATE_MOVELEFT;
|
|
e->dir = DIR_LEFT;
|
|
} else if (xv > 0) {
|
|
e->xVel = e->moveSpeed;
|
|
e->state = STATE_MOVERIGHT;
|
|
e->dir = DIR_RIGHT;
|
|
}
|
|
|
|
int yv = y - e->tileY;
|
|
if (yv < 0) {
|
|
e->yVel = -e->moveSpeed;
|
|
e->state = STATE_MOVEUP;
|
|
e->dir = DIR_UP;
|
|
} else if (yv > 0) {
|
|
e->yVel = e->moveSpeed;
|
|
e->state = STATE_MOVEDOWN;
|
|
e->dir = DIR_DOWN;
|
|
}
|
|
|
|
if (e->type == AI_GUY && _playerRunning) {
|
|
e->xVel = e->xVel << 1;
|
|
e->yVel = e->yVel << 1;
|
|
}
|
|
|
|
e->goalX = x;
|
|
e->goalY = y;
|
|
e->animFrame = 0;
|
|
e->drawXOff = e->drawYOff = 0;
|
|
}
|
|
|
|
// Initializes each entity after map is loaded
|
|
void AI::initAllEnts() {
|
|
for (Common::Array<AIEntity *>::iterator it = _ents->begin(); it != _ents->end(); ++it) {
|
|
(*it)->aiInit((*it));
|
|
if ((*it)->luaFuncInit[0]) {
|
|
if (g_hdb->_lua->callFunction((*it)->luaFuncInit, 2)) {
|
|
Common::strlcpy((*it)->entityName, g_hdb->_lua->getStringOffStack(), 32);
|
|
Common::strlcpy((*it)->printedName, g_hdb->_lua->getStringOffStack(), 32);
|
|
} else
|
|
warning("'%s' doesn't exists", (*it)->luaFuncInit);
|
|
}
|
|
}
|
|
|
|
for (int i = 0; i < _numInventory; i++) {
|
|
AIEntity *temp = &_inventory[i].ent;
|
|
|
|
// Clear out all ptrs in entity before writing
|
|
for (int j = 0; j < kMaxAnimFrames; j++) {
|
|
temp->blinkGfx[j] = nullptr;
|
|
temp->movedownGfx[j] = nullptr;
|
|
temp->moveupGfx[j] = nullptr;
|
|
temp->moveleftGfx[j] = nullptr;
|
|
temp->moverightGfx[j] = nullptr;
|
|
temp->standdownGfx[j] = nullptr;
|
|
temp->standupGfx[j] = nullptr;
|
|
temp->standleftGfx[j] = nullptr;
|
|
temp->standrightGfx[j] = nullptr;
|
|
temp->special1Gfx[j] = nullptr;
|
|
}
|
|
|
|
temp->blinkFrames = 0;
|
|
temp->movedownFrames = 0;
|
|
temp->moveupFrames = 0;
|
|
temp->moveleftFrames = 0;
|
|
temp->moverightFrames = 0;
|
|
temp->standdownFrames = 0;
|
|
temp->standupFrames = 0;
|
|
temp->standleftFrames = 0;
|
|
temp->standrightFrames = 0;
|
|
|
|
temp->draw = nullptr;
|
|
temp->aiDraw = nullptr;
|
|
temp->aiAction = temp->aiInit = temp->aiUse = nullptr;
|
|
|
|
cacheEntGfx(temp, false);
|
|
}
|
|
|
|
for (int i = 0; i < _numDeliveries; i++) {
|
|
_deliveries[i].itemGfx = g_hdb->_gfx->getTileGfx(_deliveries[i].itemGfxName, -1);
|
|
_deliveries[i].destGfx = g_hdb->_gfx->getTileGfx(_deliveries[i].destGfxName, -1);
|
|
}
|
|
|
|
// do a quick LaserScan to fill the laserbeam matrix!
|
|
laserScan();
|
|
}
|
|
|
|
void AI::killPlayer(Death method) {
|
|
int x = _player->x, y = _player->y;
|
|
|
|
stopEntity(_player);
|
|
_player->x = x;
|
|
_player->y = y;
|
|
_playerInvisible = false;
|
|
_playerDead = true;
|
|
|
|
g_hdb->_window->closeDialog();
|
|
g_hdb->_window->closeDialogChoice();
|
|
g_hdb->_window->stopPanicZone();
|
|
|
|
if (g_hdb->isPPC()) {
|
|
g_hdb->_window->closeDlvs();
|
|
g_hdb->_window->closeInv();
|
|
}
|
|
|
|
switch (method) {
|
|
case DEATH_NORMAL:
|
|
_player->state = STATE_DYING;
|
|
g_hdb->_sound->playSound(SND_GUY_DYING);
|
|
break;
|
|
case DEATH_FRIED:
|
|
_player->state = STATE_HORRIBLE1;
|
|
g_hdb->_sound->playSound(SND_GUY_FRIED);
|
|
break;
|
|
case DEATH_SHOCKED:
|
|
_player->state = STATE_HORRIBLE2;
|
|
g_hdb->_sound->playSound(SND_GUY_DYING);
|
|
g_hdb->_sound->playSound(SND_SHOCKBOT_SHOCK);
|
|
break;
|
|
case DEATH_GRABBED:
|
|
_player->state = STATE_HORRIBLE3;
|
|
g_hdb->_sound->playSound(SND_GUY_GRABBED);
|
|
break;
|
|
case DEATH_DROWNED:
|
|
_player->state = STATE_HORRIBLE4;
|
|
g_hdb->_sound->playSound(SND_GUY_DROWN);
|
|
break;
|
|
case DEATH_PANICZONE:
|
|
_player->state = STATE_DYING;
|
|
g_hdb->_sound->playSound(SND_PANIC_DEATH);
|
|
break;
|
|
case DEATH_PLUMMET:
|
|
if (!g_hdb->isDemo()) {
|
|
_player->state = STATE_PLUMMET;
|
|
g_hdb->_sound->playSound(SND_GUY_PLUMMET);
|
|
}
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
// sound.StopMusic();
|
|
if (!g_hdb->_sound->getMusicVolume())
|
|
g_hdb->_sound->playSound(SND_TRY_AGAIN);
|
|
}
|
|
|
|
void AI::stunEnemy(AIEntity *e, int time) {
|
|
bool ns = (e->stunnedWait != 0);
|
|
e->stunnedWait = g_hdb->getTimeSlice() + 1000 * time;
|
|
|
|
// Already stunned? If not, play sound
|
|
if (!ns)
|
|
switch (e->type) {
|
|
case AI_BUZZFLY:
|
|
g_hdb->_sound->playSound(SND_BUZZFLY_STUNNED);
|
|
break;
|
|
case AI_PUSHBOT:
|
|
g_hdb->_sound->playSound(SND_PUSHBOT_STUNNED);
|
|
break;
|
|
case AI_MEERKAT:
|
|
g_hdb->_sound->playSound(SND_MEERKAT_STUNNED);
|
|
break;
|
|
case AI_FATFROG:
|
|
g_hdb->_sound->playSound(SND_FATFROG_STUNNED);
|
|
break;
|
|
case AI_OMNIBOT:
|
|
case AI_SHOCKBOT:
|
|
case AI_LISTENBOT:
|
|
g_hdb->_sound->playSound(SND_ROBOT_STUNNED);
|
|
break;
|
|
case AI_GOODFAIRY:
|
|
g_hdb->_sound->playSound(SND_GOOD_FAERIE_STUNNED);
|
|
break;
|
|
case AI_BADFAIRY:
|
|
g_hdb->_sound->playSound(SND_BADFAIRY_STUNNED);
|
|
break;
|
|
case AI_ICEPUFF:
|
|
g_hdb->_sound->playSound(SND_ICEPUFF_STUNNED);
|
|
break;
|
|
case AI_RIGHTBOT:
|
|
g_hdb->_sound->playSound(SND_RIGHTBOT_STUNNED);
|
|
break;
|
|
case AI_BOOMBARREL:
|
|
g_hdb->_sound->playSound(SND_CLUB_HIT_METAL);
|
|
break;
|
|
case AI_CHICKEN:
|
|
g_hdb->_sound->playSound(SND_CHICKEN_DEATH);
|
|
// fallthrough
|
|
default:
|
|
g_hdb->_sound->playSound(g_hdb->_ai->metalOrFleshSND(e));
|
|
break;
|
|
}
|
|
}
|
|
|
|
int AI::metalOrFleshSND(AIEntity *e) {
|
|
switch (e->type) {
|
|
case AI_OMNIBOT:
|
|
case AI_TURNBOT:
|
|
case AI_SHOCKBOT:
|
|
case AI_RIGHTBOT:
|
|
case AI_PUSHBOT:
|
|
case AI_LISTENBOT:
|
|
case AI_MAINTBOT:
|
|
return SND_CLUB_HIT_METAL;
|
|
case AI_DEADEYE:
|
|
case AI_MEERKAT:
|
|
case AI_FATFROG:
|
|
case AI_GOODFAIRY:
|
|
case AI_BADFAIRY:
|
|
case AI_ICEPUFF:
|
|
case AI_BUZZFLY:
|
|
default:
|
|
return SND_CLUB_HIT_FLESH;
|
|
}
|
|
}
|
|
|
|
/*
|
|
Note from original:
|
|
Moves the entity along toward its goal, sets current frame to draw
|
|
depending on its current state. Special checking is done for the
|
|
player in here to move him along his waypoints.
|
|
*/
|
|
void AI::animateEntity(AIEntity *e) {
|
|
static const int xva[5] = {9, 0, 0, -1, 1};
|
|
static const int yva[5] = {9, -1, 1, 0, 0};
|
|
|
|
// Move entity if player is not dead
|
|
debug(9, "Before animateEntity, e->x: %d, e->y: %d", e->x, e->y);
|
|
debug(9, "Before animateEntity, e->tileX: %d, e->tileY: %d", e->tileX, e->tileY);
|
|
if (!_playerDead) {
|
|
e->x += e->xVel;
|
|
e->y += e->yVel;
|
|
e->tileX = e->x / kTileWidth;
|
|
e->tileY = e->y / kTileHeight;
|
|
debug(9, "After animateEntity, e->x: %d, e->y: %d", e->x, e->y);
|
|
debug(9, "After animateEntity, e->tileX: %d, e->tileY: %d", e->tileX, e->tileY);
|
|
}
|
|
|
|
// For non-players, check for trigger being hit
|
|
if (onEvenTile(e->x, e->y)) {
|
|
// Check if a trigger is hit
|
|
checkTriggerList(e->entityName, e->tileX, e->tileY);
|
|
|
|
/*
|
|
For Non-Players only
|
|
are we on a touchplate?
|
|
Barrels, Crates, Magic Egg & Ice Block ONLY
|
|
standing on a Touchplate will activate
|
|
something WHILE standing on it
|
|
*/
|
|
switch (e->type) {
|
|
case AI_CRATE:
|
|
case AI_BOOMBARREL:
|
|
case AI_HEAVYBARREL:
|
|
case AI_LIGHTBARREL:
|
|
case AI_MAGIC_EGG:
|
|
case AI_ICE_BLOCK:
|
|
case AI_FROGSTATUE:
|
|
{
|
|
int bgtile = g_hdb->_ai->checkForTouchplate(e->tileX, e->tileY);
|
|
if (bgtile && !e->touchpWait && e->touchpX != e->tileX && e->touchpY != e->tileY) {
|
|
if (g_hdb->_ai->checkActionList(e, e->tileX, e->tileY, false)) {
|
|
e->touchpTile = bgtile;
|
|
e->touchpX = e->tileX;
|
|
e->touchpY = e->tileY;
|
|
e->touchpWait = kPlayerTouchPWait;
|
|
}
|
|
}
|
|
_laserRescan = true;
|
|
}
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
// Are we on ice?
|
|
int bgTileFlags = g_hdb->_map->getMapBGTileFlags(e->tileX, e->tileY);
|
|
int fgTileFlags = g_hdb->_map->getMapFGTileFlags(e->tileX, e->tileY);
|
|
|
|
if (e->level == 1 ? ((bgTileFlags & kFlagIce) == kFlagIce) : (((bgTileFlags & kFlagIce) == kFlagIce) && !(fgTileFlags & kFlagGrating))) {
|
|
int nx, ny, moveOK = 0;
|
|
AIEntity *hit;
|
|
|
|
// Types allowed to slide on ice...
|
|
switch (e->type) {
|
|
case AI_GUY:
|
|
case AI_CHICKEN:
|
|
case AI_TURNBOT:
|
|
case AI_RIGHTBOT:
|
|
case AI_PUSHBOT:
|
|
case AI_CRATE:
|
|
case AI_LIGHTBARREL:
|
|
case AI_HEAVYBARREL:
|
|
case AI_BOOMBARREL:
|
|
case AI_MAGIC_EGG:
|
|
case AI_ICE_BLOCK:
|
|
case AI_DIVERTER:
|
|
e->moveSpeed = kPlayerMoveSpeed << 1;
|
|
nx = e->tileX + xva[e->dir];
|
|
ny = e->tileY + yva[e->dir];
|
|
hit = legalMove(nx, ny, e->level, &moveOK);
|
|
bgTileFlags = g_hdb->_map->getMapBGTileFlags(nx, ny);
|
|
if (hit)
|
|
switch (hit->type) {
|
|
case ITEM_GEM_WHITE:
|
|
case ITEM_GEM_BLUE:
|
|
case ITEM_GEM_GREEN:
|
|
case ITEM_GEM_RED:
|
|
case AI_GOODFAIRY:
|
|
case AI_BADFAIRY:
|
|
hit = nullptr;
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
if ((!hit && moveOK) || (bgTileFlags & kFlagPlayerDie))
|
|
setEntityGoal(e, nx, ny);
|
|
|
|
if (e == _player) {
|
|
_playerOnIce = true;
|
|
clearWaypoints();
|
|
}
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
} else if (e == _player)
|
|
_playerOnIce = false;
|
|
|
|
/*
|
|
Player only
|
|
are we trying to walk into a solid tile?
|
|
first, let's make sure we're perfectly aligned on
|
|
a tile boundary before the check so we don't snap
|
|
the player back into position...
|
|
|
|
if we're on a waypoint, nevermind!
|
|
*/
|
|
if (e == _player) {
|
|
bool result = e->x == (e->goalX * kTileWidth) && e->y == (e->goalY * kTileWidth);
|
|
if (!result) {
|
|
int xv = 0, yv = 0;
|
|
switch (e->dir) {
|
|
case DIR_UP:
|
|
yv = -1;
|
|
break;
|
|
case DIR_DOWN:
|
|
yv = 1;
|
|
break;
|
|
case DIR_LEFT:
|
|
xv = -1;
|
|
break;
|
|
case DIR_RIGHT:
|
|
xv = 1;
|
|
break;
|
|
case DIR_NONE:
|
|
default:
|
|
break;
|
|
}
|
|
|
|
bgTileFlags = g_hdb->_map->getMapBGTileFlags(e->tileX + xv, e->tileY + yv);
|
|
fgTileFlags = g_hdb->_map->getMapFGTileFlags(e->tileX + xv, e->tileY + yv);
|
|
if ((bgTileFlags & kFlagSolid) && !(fgTileFlags & kFlagGrating))
|
|
stopEntity(e);
|
|
}
|
|
}
|
|
}
|
|
|
|
// If player, then scroll the screen with the player
|
|
if (e == _player && !_playerDead) {
|
|
if (!_cameraLock)
|
|
g_hdb->_map->centerMapXY(e->x + 16, e->y + 16);
|
|
|
|
// Check if player walked into teleporter
|
|
checkTeleportList(e, e->tileX, e->tileY);
|
|
|
|
// Check for bad tiles (DEATH)
|
|
int cx = (e->x + 16) / kTileWidth;
|
|
int cy = (e->y + 16) / kTileHeight;
|
|
int bgTileFlags = g_hdb->_map->getMapBGTileFlags(cx, cy);
|
|
int fgTileFlags = g_hdb->_map->getMapFGTileFlags(cx, cy);
|
|
if ((bgTileFlags & kFlagPlayerDie) && !(checkFloating(cx, cy)) && !(fgTileFlags & kFlagGrating)) {
|
|
if ((bgTileFlags & kFlagEnergyFloor) == kFlagEnergyFloor)
|
|
killPlayer(DEATH_SHOCKED);
|
|
else if (((bgTileFlags & kFlagPlasmaFloor) == kFlagPlasmaFloor) || ((bgTileFlags & kFlagRadFloor) == kFlagRadFloor))
|
|
killPlayer(DEATH_FRIED);
|
|
else
|
|
killPlayer(DEATH_NORMAL);
|
|
return;
|
|
}
|
|
|
|
// Check if player wants to stop
|
|
// If yes, sets last waypoint right in front of player
|
|
if (_numWaypoints > 1) {
|
|
int xOff = 0;
|
|
int yOff = 0;
|
|
switch (e->dir) {
|
|
case DIR_UP:
|
|
xOff = 0;
|
|
yOff = -1;
|
|
break;
|
|
case DIR_DOWN:
|
|
xOff = 0;
|
|
yOff = 1;
|
|
break;
|
|
case DIR_LEFT:
|
|
xOff = -1;
|
|
yOff = 0;
|
|
break;
|
|
case DIR_RIGHT:
|
|
xOff = 1;
|
|
yOff = 0;
|
|
break;
|
|
case DIR_NONE:
|
|
default:
|
|
break;
|
|
}
|
|
if ((e->tileX + xOff == _waypoints[_numWaypoints - 1].x &&
|
|
e->tileY + yOff == _waypoints[_numWaypoints - 1].y) &&
|
|
e->level == _waypoints[_numWaypoints - 1].level) {
|
|
clearWaypoints();
|
|
_numWaypoints = 1;
|
|
_waypoints[0].x = e->tileX + xOff;
|
|
_waypoints[0].y = e->tileY + yOff;
|
|
_waypoints[0].level = e->level;
|
|
e->goalX = e->tileX + xOff;
|
|
e->goalY = e->tileY + yOff;
|
|
}
|
|
}
|
|
}
|
|
|
|
// Check for moving up/down stair levels
|
|
// int bgTileIndex = g_hdb->_map->getMapBGTileIndex(e->tileX, e->tileY); // CHECKME: unused?
|
|
int bgTileFlags = g_hdb->_map->getMapBGTileFlags(e->tileX, e->tileY);
|
|
// fgTileFlags = g_hdb->_map->getMapFGTileFlags(e->tileX, e->tileY); // CHECKME: unused?
|
|
if (bgTileFlags & kFlagStairTop)
|
|
e->level = 2;
|
|
else if (bgTileFlags & kFlagStairBot)
|
|
e->level = 1;
|
|
|
|
// Reached goal?
|
|
// Cinematic require less accuracy for NPCs
|
|
bool result;
|
|
if (_cineActive && e != _player)
|
|
result = (abs(e->x - (e->goalX * kTileWidth)) <= abs(e->xVel)) && (abs(e->y - (e->goalY * kTileHeight)) <= abs(e->yVel));
|
|
else
|
|
result = (e->x == e->goalX * kTileWidth) && (e->y == e->goalY * kTileHeight);
|
|
|
|
if (result) {
|
|
// If player, this is a waypoint goal.
|
|
// Drop one waypoint from list
|
|
if (e == _player) {
|
|
removeFirstWaypoint();
|
|
_playerEmerging = false;
|
|
}
|
|
|
|
// If entity not player, stop it here
|
|
// If entity is player and no waypoints are left, stop it here
|
|
if (e != _player || (!_numWaypoints && e == _player)) {
|
|
e->tileX = e->goalX;
|
|
e->tileY = e->goalY;
|
|
|
|
uint16 buttons = g_hdb->_input->getButtons();
|
|
if (e == _player && (buttons & (kButtonUp | kButtonDown | kButtonLeft | kButtonRight))) {
|
|
if (e->state != STATE_PUSHRIGHT && e->state != STATE_PUSHLEFT && e->state != STATE_PUSHUP && e->state != STATE_PUSHDOWN) {
|
|
if (buttons & kButtonUp)
|
|
e->dir = DIR_UP;
|
|
else if (buttons & kButtonDown)
|
|
e->dir = DIR_DOWN;
|
|
else if (buttons & kButtonLeft)
|
|
e->dir = DIR_LEFT;
|
|
else if (buttons & kButtonRight)
|
|
e->dir = DIR_RIGHT;
|
|
|
|
int nx = e->tileX + xva[e->dir];
|
|
int ny = e->tileY + yva[e->dir];
|
|
int result2;
|
|
AIEntity *hit = legalMove(nx, ny, e->level, &result2);
|
|
if (!hit && result2) {
|
|
switch (e->dir) {
|
|
case DIR_UP:
|
|
e->goalY = ny;
|
|
e->xVel = 0;
|
|
e->yVel = -kPlayerMoveSpeed;
|
|
e->state = STATE_MOVEUP;
|
|
break;
|
|
case DIR_DOWN:
|
|
e->goalY = ny;
|
|
e->xVel = 0;
|
|
e->yVel = kPlayerMoveSpeed;
|
|
e->state = STATE_MOVEDOWN;
|
|
break;
|
|
case DIR_LEFT:
|
|
e->goalX = nx;
|
|
e->yVel = 0;
|
|
e->xVel = -kPlayerMoveSpeed;
|
|
e->state = STATE_MOVELEFT;
|
|
break;
|
|
case DIR_RIGHT:
|
|
e->goalX = nx;
|
|
e->yVel = 0;
|
|
e->xVel = kPlayerMoveSpeed;
|
|
e->state = STATE_MOVERIGHT;
|
|
break;
|
|
case DIR_NONE:
|
|
default:
|
|
break;
|
|
}
|
|
if (_playerRunning) {
|
|
e->xVel = e->xVel << 1;
|
|
e->yVel = e->yVel << 1;
|
|
}
|
|
} else
|
|
stopEntity(e);
|
|
} else
|
|
stopEntity(e);
|
|
} else
|
|
stopEntity(e);
|
|
|
|
// Handle lasers after entity has stopped
|
|
switch (e->type) {
|
|
case AI_GUY:
|
|
case AI_CRATE:
|
|
case AI_LIGHTBARREL:
|
|
case AI_HEAVYBARREL:
|
|
case AI_BOOMBARREL:
|
|
case AI_MAGIC_EGG:
|
|
case AI_ICE_BLOCK:
|
|
case AI_DIVERTER:
|
|
if (g_hdb->_map->laserBeamExist(e->tileX, e->tileY))
|
|
_laserRescan = true;
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
// Checking at the Destination
|
|
|
|
uint64 flags = g_hdb->_map->getMapBGTileFlags(e->tileX, e->tileY);
|
|
// Can this entity float and it is over-water
|
|
if ((flags & kFlagWater) && (e->type == AI_CRATE || e->type == AI_LIGHTBARREL || e->type == AI_BOOMBARREL || e->type == AI_HEAVYBARREL || e->type == AI_FROGSTATUE || e->type == AI_DIVERTER)) {
|
|
// On a grating and level2?
|
|
if ((g_hdb->_map->getMapFGTileFlags(e->tileX, e->tileY) & kFlagGrating) && e->level == 2) {
|
|
animEntFrames(e);
|
|
return;
|
|
}
|
|
|
|
// If it fell in slime
|
|
// If it is a light barrel on a melting floor
|
|
// If it is supposed to slide across the floor
|
|
// If it is being pushed on a floating entity, don't float it
|
|
if ((flags & kFlagSlime) == kFlagSlime) {
|
|
// unless its a Heavy Barrel in which case it floats in slime
|
|
if ((e->type == AI_CRATE || e->type == AI_HEAVYBARREL) && !checkFloating(e->tileX, e->tileY)) {
|
|
addAnimateTarget(e->x, e->y, 0, 3, ANIM_NORMAL, false, false, GROUP_SLIME_SPLASH_SIT);
|
|
floatEntity(e, STATE_FLOATING);
|
|
g_hdb->_sound->playSound(SND_SPLASH);
|
|
} else if (!checkFloating(e->tileX, e->tileY)) {
|
|
if (e->type == AI_BOOMBARREL) {
|
|
aiBarrelExplode(e);
|
|
aiBarrelBlowup(e, e->tileX, e->tileY);
|
|
return;
|
|
} else {
|
|
addAnimateTarget(e->x, e->y, 0, 3, ANIM_NORMAL, false, false, GROUP_STEAM_PUFF_SIT);
|
|
removeEntity(e);
|
|
if (!g_hdb->isDemo())
|
|
g_hdb->_sound->playSound(SND_BARREL_MELTING);
|
|
return;
|
|
}
|
|
}
|
|
} else if ((flags & kFlagLightMelt) && e->type == AI_LIGHTBARREL) {
|
|
if (!checkFloating(e->tileX, e->tileY)) {
|
|
addAnimateTarget(e->x, e->y, 0, 3, ANIM_NORMAL, false, false, GROUP_STEAM_PUFF_SIT);
|
|
floatEntity(e, STATE_MELTED);
|
|
if (!g_hdb->isDemo())
|
|
g_hdb->_sound->playSound(SND_BARREL_MELTING);
|
|
}
|
|
} else if (flags & kFlagSlide) {
|
|
int xv = 0, yv = 0;
|
|
|
|
switch (e->dir) {
|
|
case DIR_UP:
|
|
yv = -1;
|
|
break;
|
|
case DIR_DOWN:
|
|
yv = 1;
|
|
break;
|
|
case DIR_LEFT:
|
|
xv = -1;
|
|
break;
|
|
case DIR_RIGHT:
|
|
xv = 1;
|
|
break;
|
|
case DIR_NONE:
|
|
default:
|
|
break;
|
|
}
|
|
|
|
AIEntity *hit = findEntityIgnore(e->tileX + xv, e->tileY + yv, &_dummyLaser);
|
|
if (!hit) {
|
|
e->state = STATE_SLIDING;
|
|
if ((flags & kFlagAnimFast) == kFlagAnimFast)
|
|
e->moveSpeed = kPlayerMoveSpeed << 1;
|
|
else if (flags & kFlagAnimSlow)
|
|
e->moveSpeed = kPlayerMoveSpeed >> 1;
|
|
setEntityGoal(e, e->tileX + xv, e->tileY + yv);
|
|
g_hdb->_sound->playSound(SND_LIGHT_SLIDE);
|
|
}
|
|
|
|
} else if (!checkFloating(e->tileX, e->tileY)) {
|
|
if (e->type == AI_BOOMBARREL || e->type == AI_HEAVYBARREL || e->type == AI_FROGSTATUE || e->type == AI_DIVERTER) {
|
|
// Make it disappear in the water
|
|
addAnimateTarget(e->x, e->y, 0, 3, ANIM_NORMAL, false, false, GROUP_WATER_SPLASH_SIT);
|
|
removeEntity(e);
|
|
g_hdb->_sound->playSound(SND_SPLASH);
|
|
return;
|
|
} else {
|
|
// Make it float and splash in water
|
|
e->state = STATE_FLOATING;
|
|
addAnimateTarget(e->x, e->y, 0, 3, ANIM_NORMAL, false, false, GROUP_WATER_SPLASH_SIT);
|
|
floatEntity(e, STATE_FLOATING);
|
|
g_hdb->_sound->playSound(SND_SPLASH);
|
|
}
|
|
}
|
|
|
|
// If it is floating downstream, keep moving it
|
|
if (flags & (kFlagPushRight | kFlagPushLeft | kFlagPushUp | kFlagPushDown)) {
|
|
int xv = 0, yv = 0;
|
|
AIState state = STATE_NONE;
|
|
|
|
if (flags & kFlagPushRight) {
|
|
e->dir = DIR_RIGHT;
|
|
xv = 1;
|
|
state = STATE_FLOATRIGHT;
|
|
} else if (flags & kFlagPushLeft) {
|
|
e->dir = DIR_LEFT;
|
|
xv = -1;
|
|
state = STATE_FLOATLEFT;
|
|
} else if (flags & kFlagPushUp) {
|
|
e->dir = DIR_UP;
|
|
yv = -1;
|
|
state = STATE_FLOATUP;
|
|
} else if (flags & kFlagPushDown) {
|
|
e->dir = DIR_DOWN;
|
|
yv = 1;
|
|
state = STATE_FLOATDOWN;
|
|
}
|
|
|
|
if (!checkFloating(e->tileX + xv, e->tileY + yv)) {
|
|
if ((flags & kFlagAnimFast) == kFlagAnimFast)
|
|
e->moveSpeed = kPlayerMoveSpeed << 1;
|
|
else if (flags & kFlagAnimMedium)
|
|
e->moveSpeed = kPlayerMoveSpeed;
|
|
else
|
|
e->moveSpeed = kPushMoveSpeed;
|
|
|
|
setEntityGoal(e, e->tileX + xv, e->tileY + yv);
|
|
e->state = state;
|
|
} else {
|
|
// Landed on a floatmove entity. Make it float really slow and then it'll speed up
|
|
uint32 flags2 = g_hdb->_map->getMapBGTileFlags(e->tileX + xv, e->tileY + yv);
|
|
if (!(flags2 & (kFlagPushRight | kFlagPushLeft | kFlagPushUp | kFlagPushDown))) {
|
|
floatEntity(e, STATE_FLOATING);
|
|
e->value1 = 0x666; // Don't move me ever again
|
|
return;
|
|
}
|
|
|
|
if (flags & kFlagPushRight) {
|
|
e->dir = DIR_RIGHT;
|
|
xv = 1;
|
|
state = STATE_FLOATRIGHT;
|
|
} else if (flags & kFlagPushLeft) {
|
|
e->dir = DIR_LEFT;
|
|
xv = -1;
|
|
state = STATE_FLOATLEFT;
|
|
} else if (flags & kFlagPushUp) {
|
|
e->dir = DIR_UP;
|
|
yv = -1;
|
|
state = STATE_FLOATUP;
|
|
} else if (flags & kFlagPushDown) {
|
|
e->dir = DIR_DOWN;
|
|
yv = 1;
|
|
state = STATE_FLOATDOWN;
|
|
}
|
|
|
|
e->moveSpeed = kPushMoveSpeed >> 1;
|
|
setEntityGoal(e, e->tileX + xv, e->tileY + yv);
|
|
e->state = state;
|
|
}
|
|
}
|
|
} else if ((flags & kFlagWater) && (e->type == AI_MAGIC_EGG || e->type == AI_ICE_BLOCK)) {
|
|
// And no foreground tile is there
|
|
if (g_hdb->_map->getMapFGTileIndex(e->tileX, e->tileY) < 0 && !checkFloating(e->tileX, e->tileY)) {
|
|
if ((flags & kFlagSlime) == kFlagSlime) {
|
|
// Evaporates in Slime
|
|
addAnimateTarget(e->x, e->y, 0, 3, ANIM_NORMAL, false, false, GROUP_STEAM_PUFF_SIT);
|
|
removeEntity(e);
|
|
g_hdb->_sound->playSound(SND_SPLASH);
|
|
return;
|
|
} else {
|
|
// Drowns in water
|
|
addAnimateTarget(e->x, e->y, 0, 3, ANIM_NORMAL, false, false, GROUP_WATER_SPLASH_SIT);
|
|
removeEntity(e);
|
|
g_hdb->_sound->playSound(SND_SPLASH);
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
} else if (onEvenTile(e->x, e->y))
|
|
setEntityGoal(e, _waypoints[0].x, _waypoints[0].y);
|
|
}
|
|
|
|
animEntFrames(e);
|
|
}
|
|
|
|
void AI::animEntFrames(AIEntity *e) {
|
|
static int click = 0;
|
|
|
|
int max = 1;
|
|
// Set current graphic to draw
|
|
switch (e->state) {
|
|
case STATE_STANDDOWN:
|
|
e->draw = e->standdownGfx[0];
|
|
max = 1;
|
|
break;
|
|
case STATE_STANDUP:
|
|
e->draw = e->standupGfx[0];
|
|
max = 1;
|
|
break;
|
|
case STATE_STANDLEFT:
|
|
e->draw = e->standleftGfx[0];
|
|
max = 1;
|
|
break;
|
|
case STATE_STANDRIGHT:
|
|
e->draw = e->standrightGfx[0];
|
|
max = 1;
|
|
break;
|
|
case STATE_BLINK:
|
|
e->draw = e->blinkGfx[e->animFrame];
|
|
max = e->blinkFrames;
|
|
break;
|
|
case STATE_MOVEUP:
|
|
e->draw = e->moveupGfx[e->animFrame];
|
|
max = e->moveupFrames;
|
|
break;
|
|
case STATE_MOVEDOWN:
|
|
e->draw = e->movedownGfx[e->animFrame];
|
|
max = e->movedownFrames;
|
|
break;
|
|
case STATE_MOVELEFT:
|
|
e->draw = e->moveleftGfx[e->animFrame];
|
|
max = e->moveleftFrames;
|
|
break;
|
|
case STATE_MOVERIGHT:
|
|
e->draw = e->moverightGfx[e->animFrame];
|
|
max = e->moverightFrames;
|
|
break;
|
|
case STATE_PUSHDOWN:
|
|
e->draw = _pushdownGfx[e->animFrame];
|
|
max = _pushdownFrames;
|
|
break;
|
|
case STATE_PUSHUP:
|
|
e->draw = _pushupGfx[e->animFrame];
|
|
max = _pushupFrames;
|
|
break;
|
|
case STATE_PUSHLEFT:
|
|
e->draw = _pushleftGfx[e->animFrame];
|
|
max = _pushleftFrames;
|
|
break;
|
|
case STATE_PUSHRIGHT:
|
|
e->draw = _pushrightGfx[e->animFrame];
|
|
max = _pushrightFrames;
|
|
break;
|
|
case STATE_GRABUP:
|
|
e->draw = _getGfx[DIR_UP];
|
|
max = 1;
|
|
break;
|
|
case STATE_GRABDOWN:
|
|
e->draw = _getGfx[DIR_DOWN];
|
|
max = 1;
|
|
break;
|
|
case STATE_GRABLEFT:
|
|
e->draw = _getGfx[DIR_LEFT];
|
|
max = 1;
|
|
break;
|
|
case STATE_GRABRIGHT:
|
|
e->draw = _getGfx[DIR_RIGHT];
|
|
max = 1;
|
|
break;
|
|
case STATE_HORRIBLE1:
|
|
e->draw = _horrible1Gfx[e->animFrame];
|
|
max = _horrible1Frames;
|
|
if (e->animFrame == max - 1)
|
|
e->state = STATE_DEAD;
|
|
break;
|
|
case STATE_HORRIBLE2:
|
|
{
|
|
e->draw = _horrible2Gfx[e->animFrame];
|
|
max = _horrible2Frames;
|
|
click++;
|
|
if (click == 16) {
|
|
g_hdb->_sound->playSound(SND_SHOCKBOT_SHOCK);
|
|
click = 0;
|
|
}
|
|
break;
|
|
}
|
|
case STATE_HORRIBLE3:
|
|
{
|
|
e->draw = _horrible3Gfx[e->animFrame];
|
|
max = _horrible3Frames;
|
|
click++;
|
|
if (click == 32) {
|
|
g_hdb->_sound->playSound(SND_GUY_GRABBED);
|
|
click = 0;
|
|
}
|
|
break;
|
|
}
|
|
case STATE_HORRIBLE4:
|
|
e->draw = _horrible4Gfx[e->animFrame];
|
|
max = _horrible4Frames;
|
|
if (e->animFrame == max - 1)
|
|
e->state = STATE_DEAD;
|
|
break;
|
|
case STATE_PLUMMET:
|
|
e->draw = _plummetGfx[e->animFrame];
|
|
max = _plummetFrames;
|
|
if (e->animFrame == max - 1) {
|
|
e->state = STATE_NONE;
|
|
setPlayerInvisible(true);
|
|
}
|
|
break;
|
|
|
|
//
|
|
// maintenance bot uses stuff
|
|
//
|
|
case STATE_USEDOWN:
|
|
e->draw = e->standdownGfx[4];
|
|
return;
|
|
case STATE_USEUP:
|
|
e->draw = e->standupGfx[4];
|
|
return;
|
|
case STATE_USELEFT:
|
|
e->draw = e->standleftGfx[4];
|
|
return;
|
|
case STATE_USERIGHT:
|
|
e->draw = e->standrightGfx[4];
|
|
return;
|
|
|
|
//
|
|
// death!
|
|
//
|
|
case STATE_DYING:
|
|
e->draw = _dyingGfx[e->animFrame];
|
|
max = _dyingFrames;
|
|
if (e->animFrame == max - 1)
|
|
e->state = STATE_DEAD;
|
|
break;
|
|
|
|
case STATE_DEAD:
|
|
e->draw = _dyingGfx[_dyingFrames - 1];
|
|
max = _dyingFrames;
|
|
break;
|
|
|
|
case STATE_GOODJOB:
|
|
e->draw = _goodjobGfx;
|
|
max = 1;
|
|
break;
|
|
|
|
//
|
|
// floating stuff uses the "standup" frames for animating the float
|
|
//
|
|
case STATE_FLOATING:
|
|
case STATE_FLOATDOWN:
|
|
case STATE_FLOATUP:
|
|
case STATE_FLOATLEFT:
|
|
case STATE_FLOATRIGHT:
|
|
e->draw = e->blinkGfx[e->animFrame];
|
|
max = e->blinkFrames;
|
|
break;
|
|
case STATE_MELTED:
|
|
case STATE_EXPLODING:
|
|
e->draw = e->special1Gfx[e->animFrame];
|
|
max = e->special1Frames;
|
|
if (e->type == AI_BOOMBARREL) {
|
|
// while exploding, call this function
|
|
aiBarrelExplodeSpread(e);
|
|
if (e->animFrame == max - 1) {
|
|
removeEntity(e);
|
|
return;
|
|
}
|
|
}
|
|
break;
|
|
|
|
//
|
|
// ICEPUFF states
|
|
//
|
|
case STATE_ICEP_PEEK:
|
|
e->draw = e->blinkGfx[e->animFrame];
|
|
max = e->blinkFrames;
|
|
break;
|
|
case STATE_ICEP_APPEAR:
|
|
e->draw = e->standupGfx[e->animFrame];
|
|
max = e->standupFrames;
|
|
break;
|
|
case STATE_ICEP_THROWDOWN:
|
|
e->draw = e->standdownGfx[e->animFrame];
|
|
max = e->standdownFrames;
|
|
break;
|
|
case STATE_ICEP_THROWRIGHT:
|
|
e->draw = e->standrightGfx[e->animFrame];
|
|
max = e->standrightFrames;
|
|
break;
|
|
case STATE_ICEP_THROWLEFT:
|
|
e->draw = e->standleftGfx[e->animFrame];
|
|
max = e->standleftFrames;
|
|
break;
|
|
case STATE_ICEP_DISAPPEAR:
|
|
e->draw = e->special1Gfx[e->animFrame];
|
|
max = e->special1Frames;
|
|
break;
|
|
|
|
//
|
|
// MEERKAT states
|
|
//
|
|
case STATE_MEER_MOVE:
|
|
e->draw = e->standdownGfx[e->animFrame];
|
|
max = e->standdownFrames;
|
|
break;
|
|
case STATE_MEER_APPEAR:
|
|
e->draw = e->standleftGfx[e->animFrame];
|
|
max = e->standleftFrames;
|
|
break;
|
|
case STATE_MEER_BITE:
|
|
e->draw = e->standrightGfx[e->animFrame];
|
|
max = e->standrightFrames;
|
|
break;
|
|
case STATE_MEER_DISAPPEAR:
|
|
e->draw = e->standupGfx[e->animFrame];
|
|
max = e->standupFrames;
|
|
break;
|
|
case STATE_MEER_LOOK:
|
|
e->draw = e->movedownGfx[e->animFrame];
|
|
max = e->movedownFrames;
|
|
break;
|
|
|
|
//
|
|
// DIVERTER spawning states
|
|
//
|
|
case STATE_DIVERTER_BL:
|
|
e->draw = e->standdownGfx[e->animFrame];
|
|
max = e->standdownFrames;
|
|
break;
|
|
case STATE_DIVERTER_BR:
|
|
e->draw = e->standupGfx[e->animFrame];
|
|
max = e->standupFrames;
|
|
break;
|
|
case STATE_DIVERTER_TL:
|
|
e->draw = e->standleftGfx[e->animFrame];
|
|
max = e->standleftFrames;
|
|
break;
|
|
case STATE_DIVERTER_TR:
|
|
e->draw = e->standrightGfx[e->animFrame];
|
|
max = e->standrightFrames;
|
|
break;
|
|
|
|
//
|
|
// DOLLY states
|
|
// angry[4] = standright[4]
|
|
// kissright[4]/kissleft[4] = standleft[8]
|
|
// panic[4]/laugh[4] = standdown[8]
|
|
// dollyuseright[5] = special1[5]
|
|
//
|
|
case STATE_ANGRY:
|
|
e->draw = e->standrightGfx[e->animFrame];
|
|
max = 2;
|
|
break;
|
|
case STATE_KISSRIGHT:
|
|
e->draw = e->standleftGfx[e->animFrame];
|
|
max = 4;
|
|
break;
|
|
case STATE_KISSLEFT:
|
|
e->draw = e->standleftGfx[e->animFrame + 4];
|
|
max = 4;
|
|
break;
|
|
case STATE_PANIC:
|
|
e->draw = e->standdownGfx[e->animFrame];
|
|
max = 2;
|
|
break;
|
|
case STATE_LAUGH:
|
|
e->draw = e->standdownGfx[e->animFrame + 4];
|
|
max = 2;
|
|
break;
|
|
case STATE_DOLLYUSERIGHT:
|
|
e->draw = e->special1Gfx[e->animFrame];
|
|
max = e->special1Frames;
|
|
break;
|
|
|
|
// SARGE yelling
|
|
case STATE_YELL:
|
|
e->draw = e->special1Gfx[e->animFrame];
|
|
max = e->special1Frames;
|
|
break;
|
|
default:
|
|
debug(9, "AI-FUNCS: animEntFrames: Unintended State for entity %s", AIType2Str(e->type));
|
|
break;
|
|
}
|
|
|
|
// Cycle animation frames
|
|
if (e->animDelay-- > 0)
|
|
return;
|
|
|
|
e->animDelay = e->animCycle;
|
|
e->animFrame++;
|
|
if (e->animFrame == max)
|
|
e->animFrame = 0;
|
|
}
|
|
|
|
void AI::drawEnts(int x, int y, int w, int h) {
|
|
static int stunAnim = 0;
|
|
static uint32 stunTimer = g_hdb->getTimeSlice();
|
|
|
|
int debugFlag = g_hdb->getDebug();
|
|
|
|
// Draw Floating Entities
|
|
for (uint i = 0; i < _floats->size(); i++) {
|
|
AIEntity *e = _floats->operator[](i);
|
|
if (e->aiDraw)
|
|
e->aiDraw(e, x, y);
|
|
|
|
if ((e->x > x - kTileWidth) && (e->x < x + w) && (e->y > y - kTileHeight) && (e->y < y + h)) {
|
|
e->draw->drawMasked(e->x - x + e->drawXOff, e->y - y + e->drawYOff);
|
|
e->onScreen = 1;
|
|
} else
|
|
e->onScreen = 0;
|
|
}
|
|
|
|
// Draw all other Ents
|
|
_numLevel2Ents = 0;
|
|
|
|
for (uint i = 0; i < _ents->size(); i++) {
|
|
AIEntity *e = _ents->operator[](i);
|
|
debugN(5, "AI::drawEnts: enity %s(%d) state %s(%d)...", AIType2Str(e->type), e->type, AIState2Str(e->state), e->state);
|
|
|
|
if (e->type == AI_LASER || e->type == AI_DIVERTER) {
|
|
if (e->aiDraw) {
|
|
if (e->level == 2 && _numLevel2Ents < kMaxLevel2Ents) {
|
|
_entsLevel2[_numLevel2Ents].aiDraw = e->aiDraw;
|
|
_entsLevel2[_numLevel2Ents].x = x;
|
|
_entsLevel2[_numLevel2Ents].y = y;
|
|
_entsLevel2[_numLevel2Ents].e = e;
|
|
_entsLevel2[_numLevel2Ents].stunnedWait = 0;
|
|
_numLevel2Ents++;
|
|
debugN(5, "not drawing1...");
|
|
} else {
|
|
e->aiDraw(e, x, y);
|
|
debugN(5, "drawing1...");
|
|
}
|
|
}
|
|
}
|
|
|
|
if ((e->x > x - kTileWidth) && (e->x < x + w) && (e->y > y - kTileHeight) && (e->y < y + h)) {
|
|
// If extra drawing func is present, call it
|
|
if (e->aiDraw && e->type != AI_LASER && e->type != AI_DIVERTER) {
|
|
if (e->level == 2 && _numLevel2Ents < kMaxLevel2Ents) {
|
|
_entsLevel2[_numLevel2Ents].aiDraw = e->aiDraw;
|
|
_entsLevel2[_numLevel2Ents].draw = e->draw;
|
|
_entsLevel2[_numLevel2Ents].x = x;
|
|
_entsLevel2[_numLevel2Ents].y = y;
|
|
_entsLevel2[_numLevel2Ents].e = e;
|
|
_entsLevel2[_numLevel2Ents].stunnedWait = 0;
|
|
_numLevel2Ents++;
|
|
debugN(5, "not drawing2...");
|
|
} else {
|
|
e->aiDraw(e, x, y);
|
|
debugN(5, "drawing2...");
|
|
}
|
|
}
|
|
|
|
switch (e->type) {
|
|
case AI_VORTEXIAN:
|
|
if (e->draw)
|
|
e->draw->drawMasked(e->x - x + e->drawXOff, e->y - y + e->drawYOff, e->value2 & 0xff);
|
|
break;
|
|
case AI_GUY: // Draw Player Last
|
|
break;
|
|
default:
|
|
if (e->level == 2 && _numLevel2Ents < kMaxLevel2Ents) {
|
|
_entsLevel2[_numLevel2Ents].aiDraw = nullptr;
|
|
_entsLevel2[_numLevel2Ents].draw = e->draw;
|
|
_entsLevel2[_numLevel2Ents].x = e->x - x + e->drawXOff;
|
|
_entsLevel2[_numLevel2Ents].y = e->y - y + e->drawYOff;
|
|
_entsLevel2[_numLevel2Ents].e = nullptr;
|
|
_entsLevel2[_numLevel2Ents].stunnedWait = e->stunnedWait;
|
|
_numLevel2Ents++;
|
|
debugN(5, "not trying to draw...");
|
|
} else {
|
|
debugN(5, "trying to draw...");
|
|
|
|
if (e->draw) {
|
|
debugN(5, "at %d %d", e->x, e->y);
|
|
|
|
e->draw->drawMasked(e->x - x + e->drawXOff, e->y - y + e->drawYOff);
|
|
} else if (debugFlag)
|
|
_debugQMark->drawMasked(e->x - x, e->y - y);
|
|
else
|
|
debugN(5, "no draw function");
|
|
|
|
if (e->stunnedWait)
|
|
g_hdb->_ai->_stunnedGfx[stunAnim]->drawMasked(e->x - x, e->y - y);
|
|
}
|
|
break;
|
|
}
|
|
e->onScreen = 1;
|
|
} else {
|
|
e->onScreen = 0;
|
|
debugN(5, "not on screen");
|
|
}
|
|
debug(5, "%s", ""); // newline
|
|
}
|
|
|
|
if (stunTimer < g_hdb->getTimeSlice()) {
|
|
stunAnim = (stunAnim + 1) & 3;
|
|
stunTimer = g_hdb->getTimeSlice();
|
|
}
|
|
|
|
// Draw player last
|
|
if (_player && _player->level < 2 && !_playerInvisible && _player->draw)
|
|
_player->draw->drawMasked(_player->x - x + _player->drawXOff, _player->y - y + _player->drawYOff);
|
|
}
|
|
|
|
void AI::drawLevel2Ents() {
|
|
int debugFlag = g_hdb->getDebug();
|
|
|
|
for (int i = 0; i < _numLevel2Ents; i++) {
|
|
// call custom drawing code?
|
|
if (_entsLevel2[i].aiDraw) {
|
|
debug(5, "AI::drawLevel2Ents: entity %s(%d) at %d,%d", AIType2Str(_entsLevel2[i].e->type), _entsLevel2[i].e->type, _entsLevel2[i].x, _entsLevel2[i].y);
|
|
|
|
_entsLevel2[i].aiDraw(_entsLevel2[i].e, _entsLevel2[i].x, _entsLevel2[i].y);
|
|
} else if (_entsLevel2[i].draw) {
|
|
debug(5, "AI::drawLevel2Ents: tile '%s' at %d,%d", _entsLevel2[i].draw->getName(), _entsLevel2[i].x, _entsLevel2[i].y);
|
|
|
|
_entsLevel2[i].draw->drawMasked(_entsLevel2[i].x, _entsLevel2[i].y);
|
|
} else if (debugFlag)
|
|
_debugQMark->drawMasked(_entsLevel2[i].x, _entsLevel2[i].y );
|
|
|
|
if (_entsLevel2[i].stunnedWait)
|
|
g_hdb->_ai->_stunnedGfx[_stunAnim]->drawMasked(_entsLevel2[i].x , _entsLevel2[i].y);
|
|
}
|
|
|
|
// always draw the player last
|
|
if (_player && _player->level == 2 && !_playerInvisible) {
|
|
int x, y;
|
|
g_hdb->_map->getMapXY(&x, &y);
|
|
|
|
if (_player->draw)
|
|
_player->draw->drawMasked((_player->x - x) + _player->drawXOff, (_player->y - y) + _player->drawYOff);
|
|
}
|
|
|
|
if (_stunTimer < g_system->getMillis()) {
|
|
_stunAnim = (_stunAnim + 1) & 3;
|
|
_stunTimer = g_system->getMillis() + 100;
|
|
}
|
|
}
|
|
|
|
void AI::animGrabbing() {
|
|
if (_player->state == STATE_GRABUP ||
|
|
_player->state == STATE_GRABDOWN ||
|
|
_player->state == STATE_GRABLEFT ||
|
|
_player->state == STATE_GRABRIGHT)
|
|
return;
|
|
|
|
AIState s = STATE_NONE;
|
|
|
|
switch (_player->dir) {
|
|
case DIR_UP:
|
|
s = STATE_GRABUP;
|
|
_player->draw = _getGfx[DIR_UP];
|
|
break;
|
|
case DIR_DOWN:
|
|
s = STATE_GRABDOWN;
|
|
_player->draw = _getGfx[DIR_DOWN];
|
|
break;
|
|
case DIR_LEFT:
|
|
s = STATE_GRABLEFT;
|
|
_player->draw = _getGfx[DIR_LEFT];
|
|
break;
|
|
case DIR_RIGHT:
|
|
s = STATE_GRABRIGHT;
|
|
_player->draw = _getGfx[DIR_RIGHT];
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
_player->state = s;
|
|
_player->animFrame = 5;
|
|
}
|
|
|
|
void AI::entityFace(const char *luaName, int dir) {
|
|
AIEntity *e = locateEntity(luaName);
|
|
e->dir = (AIDir)dir;
|
|
|
|
switch (e->dir) {
|
|
case DIR_UP:
|
|
e->state = STATE_STANDUP;
|
|
break;
|
|
case DIR_DOWN:
|
|
e->state = STATE_STANDDOWN;
|
|
break;
|
|
case DIR_LEFT:
|
|
e->state = STATE_STANDLEFT;
|
|
break;
|
|
case DIR_RIGHT:
|
|
e->state = STATE_STANDRIGHT;
|
|
break;
|
|
case DIR_NONE:
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
void AI::moveEnts() {
|
|
static int frameDelay = kAnimFrameDelay;
|
|
static bool startLaserSound = false;
|
|
|
|
if (frameDelay-- > 0)
|
|
return;
|
|
|
|
frameDelay = kAnimFrameDelay;
|
|
|
|
// Call aiAction for Floating Entities
|
|
for (Common::Array<AIEntity *>::iterator it = _floats->begin(); it != _floats->end(); ++it) {
|
|
if ((*it)->aiAction)
|
|
(*it)->aiAction((*it));
|
|
}
|
|
|
|
// Call aiAction for all other Entities
|
|
for (uint i = 0; i < _ents->size(); i++) {
|
|
AIEntity *e = _ents->operator[](i);
|
|
if (e->aiAction) {
|
|
// NPC Touchplate Counter
|
|
if (e != _player && e->touchpWait) {
|
|
e->touchpWait--;
|
|
if (!e->touchpWait) {
|
|
if (e->tileX == e->touchpX && e->tileY == e->touchpY && onEvenTile(e->x, e->y))
|
|
e->touchpWait = 1;
|
|
else {
|
|
checkActionList(e, e->touchpX, e->touchpY, false);
|
|
g_hdb->_map->setMapBGTileIndex(e->touchpX, e->touchpY, e->touchpTile);
|
|
e->touchpX = e->touchpY = e->touchpTile = 0;
|
|
}
|
|
}
|
|
}
|
|
// Stunned Entity Timer
|
|
if (!e->stunnedWait)
|
|
e->aiAction(e);
|
|
else if (e->stunnedWait < (int32)g_hdb->getTimeSlice())
|
|
e->stunnedWait = 0;
|
|
}
|
|
}
|
|
|
|
// if lasers need to rescan, do it here only
|
|
if (_laserRescan) {
|
|
_laserRescan = false;
|
|
laserScan();
|
|
}
|
|
|
|
// handle the constant laser looping sound channel
|
|
if (_laserOnScreen)
|
|
startLaserSound = true;
|
|
if (!_laserOnScreen && startLaserSound) {
|
|
startLaserSound = false;
|
|
g_hdb->_sound->stopChannel(kLaserChannel);
|
|
}
|
|
}
|
|
|
|
bool AI::findPath(AIEntity *e) {
|
|
// Initial Pointing Direction to search in
|
|
int x = e->tileX;
|
|
int y = e->tileY;
|
|
ArrowPath *here = findArrowPath(x, y);
|
|
// Only look for GO arrows at this first location
|
|
if (here && here->type == 1)
|
|
e->dir = here->dir;
|
|
|
|
int xv = 0, yv = 0;
|
|
switch (e->dir) {
|
|
case DIR_UP:
|
|
yv = -1;
|
|
break;
|
|
case DIR_DOWN:
|
|
yv = 1;
|
|
break;
|
|
case DIR_LEFT:
|
|
xv = -1;
|
|
break;
|
|
case DIR_RIGHT:
|
|
xv = 1;
|
|
break;
|
|
case DIR_NONE:
|
|
default:
|
|
break;
|
|
}
|
|
|
|
int max;
|
|
if (xv)
|
|
max = g_hdb->_map->_width;
|
|
else
|
|
max = g_hdb->_map->_height;
|
|
|
|
while (max--) {
|
|
ArrowPath *arrowPath = findArrowPath(x + xv, y + yv);
|
|
if (arrowPath) {
|
|
setEntityGoal(e, arrowPath->tileX, arrowPath->tileY);
|
|
return true;
|
|
} else {
|
|
uint32 flags = g_hdb->_map->getMapBGTileFlags(x + xv, y + yv);
|
|
if (flags & kFlagSolid)
|
|
return false;
|
|
}
|
|
x += xv;
|
|
y += yv;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
AIEntity *AI::legalMove(int tileX, int tileY, int level, int *result) {
|
|
uint32 bgFlags = g_hdb->_map->getMapBGTileFlags(tileX, tileY);
|
|
uint32 fgFlags = g_hdb->_map->getMapFGTileFlags(tileX, tileY);
|
|
AIEntity *hit = findEntity(tileX, tileY);
|
|
|
|
if (hit && hit->state != STATE_FLOATING) {
|
|
// If player and entity are not at the same level, are they on stairs?
|
|
if (hit->level != level) {
|
|
if (level == 1 && !(bgFlags & kFlagStairTop))
|
|
hit = nullptr;
|
|
else if (level == 2 && !(bgFlags & kFlagStairBot))
|
|
hit = nullptr;
|
|
}
|
|
}
|
|
|
|
if (level == 1) {
|
|
if (bgFlags & kFlagSolid) {
|
|
*result = 0;
|
|
return hit;
|
|
}
|
|
|
|
if (bgFlags & (kFlagWater | kFlagSlime)) {
|
|
if (hit && hit->state == STATE_FLOATING) {
|
|
*result = 1;
|
|
return nullptr;
|
|
} else
|
|
*result = 0;
|
|
return hit;
|
|
} else
|
|
*result = 1;
|
|
} else {
|
|
if (fgFlags & kFlagSolid) {
|
|
*result = 0;
|
|
return hit;
|
|
} else if (fgFlags & kFlagGrating) {
|
|
*result = 1;
|
|
return hit;
|
|
} else if (bgFlags & kFlagSolid) {
|
|
*result = 0;
|
|
return hit;
|
|
}
|
|
|
|
if (bgFlags & (kFlagWater | kFlagSlime | kFlagPlummet)) {
|
|
if (hit && hit->state == STATE_FLOATING) {
|
|
*result = 1;
|
|
return nullptr;
|
|
} else
|
|
*result = 0;
|
|
return hit;
|
|
} else
|
|
*result = 1;
|
|
}
|
|
return hit;
|
|
}
|
|
|
|
AIEntity *AI::legalMoveOverWater(int tileX, int tileY, int level, int *result) {
|
|
uint32 bgFlags = g_hdb->_map->getMapBGTileFlags(tileX, tileY);
|
|
uint32 fgFlags = g_hdb->_map->getMapFGTileFlags(tileX, tileY);
|
|
AIEntity *hit = findEntity(tileX, tileY);
|
|
|
|
if (level == 1 ? (bgFlags & kFlagMonsterBlock) : (!(fgFlags &kFlagGrating) && ((fgFlags & kFlagSolid) || (bgFlags & kFlagMonsterBlock))))
|
|
*result = 0;
|
|
else
|
|
*result = 1;
|
|
|
|
return hit;
|
|
}
|
|
|
|
AIEntity *AI::legalMoveOverWaterIgnore(int tileX, int tileY, int level, int *result, AIEntity *ignore) {
|
|
uint32 bgFlags = g_hdb->_map->getMapBGTileFlags(tileX, tileY);
|
|
uint32 fgFlags = g_hdb->_map->getMapFGTileFlags(tileX, tileY);
|
|
AIEntity *hit = findEntityIgnore(tileX, tileY, ignore);
|
|
|
|
if (level == 1 ? (bgFlags & kFlagMonsterBlock) : (!(fgFlags &kFlagGrating) && ((fgFlags & kFlagSolid) || (bgFlags & kFlagMonsterBlock))))
|
|
*result = 0;
|
|
else
|
|
*result = 1;
|
|
|
|
return hit;
|
|
}
|
|
|
|
AIEntity *AI::playerCollision(int topBorder, int bottomBorder, int leftBorder, int rightBorder) {
|
|
for (Common::Array<AIEntity *>::iterator it = _ents->begin(); it != _ents->end(); ++it) {
|
|
AIEntity *e = *it;
|
|
if (e == _player || !e->onScreen)
|
|
continue;
|
|
|
|
if (e->x > (_player->x - 32 - leftBorder) && e->x < (_player->x + 32 + rightBorder) && e->y >(_player->y - 32 - topBorder) && e->y < (_player->y + 32 + bottomBorder))
|
|
return e;
|
|
}
|
|
return nullptr;
|
|
}
|
|
|
|
bool AI::checkPlayerTileCollision(int x, int y) {
|
|
if (g_hdb->getDebug() == 2)
|
|
return false;
|
|
|
|
return (_player->tileX == x && _player->tileY == y);
|
|
}
|
|
|
|
bool AI::checkPlayerCollision(int x, int y, int border) {
|
|
if (g_hdb->getDebug() == 2 || !_player)
|
|
return false;
|
|
|
|
return (x > (_player->x - 32 + border) && x < (_player->x + 32 - border) &&
|
|
y > (_player->y - 32 + border) && y < (_player->y + 32 - border));
|
|
}
|
|
|
|
void AI::clearDiverters() {
|
|
for (uint i = 0; i < _ents->size(); i++) {
|
|
AIEntity *e = _ents->operator[](i);
|
|
if (e->type == AI_DIVERTER)
|
|
e->value1 = e->value2 = 0;
|
|
}
|
|
}
|
|
|
|
void AI::laserScan() {
|
|
clearDiverters();
|
|
g_hdb->_map->clearLaserBeams();
|
|
|
|
for (uint i = 0; i < _ents->size(); i++) {
|
|
AIEntity *e = _ents->operator[](i);
|
|
if (e->type == AI_LASER)
|
|
aiLaserAction(e);
|
|
}
|
|
}
|
|
|
|
void AI::floatEntity(AIEntity *e, AIState state) {
|
|
for (Common::Array<AIEntity *>::iterator it = _ents->begin(); it != _ents->end(); ++it) {
|
|
if (e == *it) {
|
|
_floats->push_back(*it);
|
|
_ents->erase(it);
|
|
break;
|
|
}
|
|
}
|
|
e->state = state;
|
|
e->level = 1;
|
|
}
|
|
|
|
bool AI::checkFloating(int x, int y) {
|
|
for (Common::Array<AIEntity *>::iterator it = _floats->begin(); it != _floats->end(); ++it) {
|
|
if ((*it)->tileX == x && (*it)->tileY == y)
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
// Check to see if we can get this entity
|
|
bool AI::getTableEnt(AIType type) {
|
|
switch (type) {
|
|
case ITEM_CELL:
|
|
case ITEM_ENV_WHITE:
|
|
case ITEM_ENV_RED:
|
|
case ITEM_ENV_BLUE:
|
|
case ITEM_ENV_GREEN:
|
|
case ITEM_TRANSCEIVER:
|
|
case ITEM_CLUB:
|
|
case ITEM_ROBOSTUNNER:
|
|
case ITEM_SLUGSLINGER:
|
|
case ITEM_MONKEYSTONE:
|
|
case ITEM_GOO_CUP:
|
|
case ITEM_TEACUP:
|
|
case ITEM_BURGER:
|
|
case ITEM_PDA:
|
|
case ITEM_BOOK:
|
|
case ITEM_CLIPBOARD:
|
|
case ITEM_NOTE:
|
|
case ITEM_KEYCARD_WHITE:
|
|
case ITEM_KEYCARD_BLUE:
|
|
case ITEM_KEYCARD_RED:
|
|
case ITEM_KEYCARD_GREEN:
|
|
case ITEM_KEYCARD_PURPLE:
|
|
case ITEM_KEYCARD_BLACK:
|
|
case ITEM_SEED:
|
|
case ITEM_SODA:
|
|
case ITEM_SLICER:
|
|
case ITEM_DOLLYTOOL1:
|
|
case ITEM_DOLLYTOOL2:
|
|
case ITEM_DOLLYTOOL3:
|
|
case ITEM_DOLLYTOOL4:
|
|
return true;
|
|
default:
|
|
return false;
|
|
}
|
|
}
|
|
|
|
// Check to see if it's okay to move through this entity
|
|
bool AI::walkThroughEnt(AIType type) {
|
|
switch (type) {
|
|
case AI_VORTEXIAN:
|
|
case AI_MEERKAT:
|
|
case AI_GOODFAIRY:
|
|
case AI_BADFAIRY:
|
|
case AI_GATEPUDDLE:
|
|
case AI_BUZZFLY:
|
|
case AI_OMNIBOT:
|
|
case AI_PUSHBOT:
|
|
case AI_TURNBOT:
|
|
case AI_RIGHTBOT:
|
|
|
|
case ITEM_GEM_WHITE:
|
|
case ITEM_GEM_BLUE:
|
|
case ITEM_GEM_RED:
|
|
case ITEM_GEM_GREEN:
|
|
return true;
|
|
default:
|
|
return getTableEnt(type);
|
|
}
|
|
}
|
|
|
|
// Play special sound for every item you get
|
|
void AI::getItemSound(AIType type) {
|
|
switch (type) {
|
|
case ITEM_GOO_CUP:
|
|
g_hdb->_sound->playSound(SND_GET_GOO);
|
|
break;
|
|
case ITEM_GEM_WHITE:
|
|
case ITEM_GEM_BLUE:
|
|
case ITEM_GEM_RED:
|
|
case ITEM_GEM_GREEN:
|
|
g_hdb->_sound->playSound(SND_GET_GEM);
|
|
break;
|
|
case ITEM_CLUB:
|
|
g_hdb->_sound->playSound(SND_GET_CLUB);
|
|
break;
|
|
case ITEM_SLUGSLINGER:
|
|
g_hdb->_sound->playSound(SND_GET_SLUG);
|
|
break;
|
|
case ITEM_ROBOSTUNNER:
|
|
g_hdb->_sound->playSound(SND_GET_STUNNER);
|
|
break;
|
|
case ITEM_CELL:
|
|
case ITEM_TRANSCEIVER:
|
|
case ITEM_TEACUP:
|
|
case ITEM_COOKIE:
|
|
case ITEM_BURGER:
|
|
case ITEM_PDA:
|
|
case ITEM_BOOK:
|
|
case ITEM_CLIPBOARD:
|
|
case ITEM_NOTE:
|
|
case ITEM_CABKEY:
|
|
case ITEM_DOLLYTOOL1:
|
|
case ITEM_DOLLYTOOL2:
|
|
case ITEM_DOLLYTOOL3:
|
|
case ITEM_DOLLYTOOL4:
|
|
case ITEM_SEED:
|
|
case ITEM_SODA:
|
|
case ITEM_ROUTER:
|
|
case ITEM_SLICER:
|
|
case ITEM_CHICKEN:
|
|
case ITEM_PACKAGE:
|
|
case ITEM_ENV_RED:
|
|
case ITEM_ENV_BLUE:
|
|
case ITEM_ENV_GREEN:
|
|
if (!g_hdb->isPPC()) {
|
|
if (g_hdb->_sound->getVoiceStatus())
|
|
g_hdb->_sound->playVoice(GUY_GOT_SOMETHING, 1);
|
|
else
|
|
g_hdb->_sound->playSound(SND_GET_THING);
|
|
break;
|
|
}
|
|
// fall through
|
|
default:
|
|
g_hdb->_sound->playSound(SND_GET_THING);
|
|
}
|
|
}
|
|
|
|
void AI::lookAtEntity(AIEntity *e) {
|
|
lookAtXY(e->tileX, e->tileY);
|
|
}
|
|
|
|
// Change player direction to XY
|
|
void AI::lookAtXY(int x, int y) {
|
|
int distX = abs(_player->tileX - x);
|
|
int distY = abs(_player->tileY - y);
|
|
|
|
if (distX > distY) {
|
|
// X takes precedence
|
|
if (x < _player->tileX)
|
|
_player->dir = DIR_LEFT;
|
|
else if (x > _player->tileX)
|
|
_player->dir = DIR_RIGHT;
|
|
else if (y < _player->tileY)
|
|
_player->dir = DIR_UP;
|
|
else
|
|
_player->dir = DIR_DOWN;
|
|
} else {
|
|
// Y takes precedence
|
|
if (y < _player->tileY)
|
|
_player->dir = DIR_UP;
|
|
else if (y > _player->tileY)
|
|
_player->dir = DIR_DOWN;
|
|
else if (x < _player->tileX)
|
|
_player->dir = DIR_LEFT;
|
|
else
|
|
_player->dir = DIR_RIGHT;
|
|
}
|
|
|
|
switch (_player->dir) {
|
|
case DIR_UP:
|
|
_player->state = STATE_STANDUP;
|
|
_player->draw = _getGfx[DIR_UP];
|
|
break;
|
|
case DIR_DOWN:
|
|
_player->state = STATE_STANDDOWN;
|
|
_player->draw = _getGfx[DIR_DOWN];
|
|
break;
|
|
case DIR_LEFT:
|
|
_player->state = STATE_STANDLEFT;
|
|
_player->draw = _getGfx[DIR_LEFT];
|
|
break;
|
|
case DIR_RIGHT:
|
|
_player->state = STATE_STANDRIGHT;
|
|
_player->draw = _getGfx[DIR_RIGHT];
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
void AI::movePlayer(uint16 buttons) {
|
|
static const AIState stateList[] = {
|
|
STATE_ATK_CLUB_UP, STATE_ATK_CLUB_DOWN, STATE_ATK_CLUB_LEFT, STATE_ATK_CLUB_RIGHT,
|
|
STATE_ATK_STUN_UP, STATE_ATK_STUN_DOWN, STATE_ATK_STUN_LEFT, STATE_ATK_STUN_RIGHT,
|
|
STATE_ATK_SLUG_UP, STATE_ATK_SLUG_DOWN, STATE_ATK_SLUG_LEFT, STATE_ATK_SLUG_RIGHT,
|
|
STATE_PUSHUP, STATE_PUSHDOWN, STATE_PUSHLEFT, STATE_PUSHRIGHT,
|
|
STATE_GRABUP, STATE_GRABDOWN, STATE_GRABLEFT, STATE_GRABRIGHT
|
|
};
|
|
|
|
static const int xva[5] = {9, 0, 0,-1, 1};
|
|
static const int yva[5] = {9,-1, 1, 0, 0};
|
|
|
|
if (!_player)
|
|
return;
|
|
|
|
// If we're already attacking, don't do anything else
|
|
for (int i = 0; i < 20; i++) {
|
|
if (_player->state == stateList[i])
|
|
return;
|
|
}
|
|
|
|
// Just trying to put away a dialog?
|
|
if (buttons & kButtonB) {
|
|
if (g_hdb->isPPC()) {
|
|
if (g_hdb->_window->deliveriesActive()) {
|
|
g_hdb->_window->closeDlvs();
|
|
return;
|
|
} else if (g_hdb->_window->inventoryActive()) {
|
|
g_hdb->_window->closeInv();
|
|
return;
|
|
}
|
|
}
|
|
if (g_hdb->_window->dialogActive()) {
|
|
g_hdb->_window->closeDialog();
|
|
return;
|
|
} else if (g_hdb->_window->dialogChoiceActive()) {
|
|
g_hdb->_window->closeDialogChoice();
|
|
return;
|
|
} else if (g_hdb->_window->msgBarActive()) {
|
|
g_hdb->_window->closeMsg();
|
|
return;
|
|
}
|
|
|
|
if (cinematicsActive() || _playerLock)
|
|
return;
|
|
|
|
// Are we trying to use something? An ACTION, AUTO, LUA?
|
|
int nx = _player->tileX + xva[_player->dir];
|
|
int ny = _player->tileY + yva[_player->dir];
|
|
AIEntity *hit = findEntity(nx, ny);
|
|
|
|
// the reason to check for no entity or an AI_NONE is because
|
|
// there's a possibility that an actual entity and a LUA entity
|
|
// can share the same spot, so we need to be able to deal with
|
|
// the real entity first, then the LUA entity.
|
|
if (!hit || hit->type == AI_NONE) {
|
|
switch (_player->state) {
|
|
case STATE_STANDUP:
|
|
case STATE_STANDDOWN:
|
|
case STATE_STANDLEFT:
|
|
case STATE_STANDRIGHT:
|
|
if (checkForTouchplate(nx, ny))
|
|
break;
|
|
if (checkActionList(_player, nx, ny, true))
|
|
return;
|
|
if (checkAutoList(_player, nx, ny))
|
|
return;
|
|
if (checkLuaList(_player, nx, ny))
|
|
return;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
// Attackable Entity? (we're right up on it)
|
|
int amt = getGemAmount();
|
|
bool attackable = false;
|
|
if (hit)
|
|
switch (hit->type) {
|
|
case AI_OMNIBOT:
|
|
case AI_TURNBOT:
|
|
case AI_SHOCKBOT:
|
|
case AI_RIGHTBOT:
|
|
case AI_PUSHBOT:
|
|
case AI_LISTENBOT:
|
|
case AI_MAINTBOT:
|
|
case AI_DEADEYE:
|
|
case AI_MEERKAT:
|
|
case AI_FATFROG:
|
|
case AI_GOODFAIRY:
|
|
case AI_BADFAIRY:
|
|
case AI_ICEPUFF:
|
|
case AI_BUZZFLY:
|
|
case AI_DRAGON:
|
|
case AI_NONE:
|
|
attackable = true;
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
if (g_hdb->getActionMode() && ((hit && attackable) || !hit)) {
|
|
// Attack
|
|
if (_weaponSelected != AI_NONE && onEvenTile(_player->x, _player->y)) {
|
|
switch (_weaponSelected) {
|
|
case ITEM_CLUB: {
|
|
AIState club[5] = {STATE_NONE, STATE_ATK_CLUB_UP, STATE_ATK_CLUB_DOWN, STATE_ATK_CLUB_LEFT, STATE_ATK_CLUB_RIGHT};
|
|
_player->state = club[_player->dir];
|
|
_player->animFrame = 0;
|
|
_player->animDelay = _player->animCycle;
|
|
g_hdb->_sound->playSound(SND_CLUB_MISS);
|
|
}
|
|
break;
|
|
|
|
case ITEM_ROBOSTUNNER: {
|
|
// it costs 1 gem to attack!
|
|
if (!amt) {
|
|
g_hdb->_sound->playSound(SND_CELLHOLDER_USE_REJECT);
|
|
g_hdb->_window->openMessageBar("Recharging...", 1);
|
|
setGemAmount(1);
|
|
return;
|
|
}
|
|
setGemAmount(amt - 1);
|
|
|
|
AIState stun[5] = {STATE_NONE, STATE_ATK_STUN_UP, STATE_ATK_STUN_DOWN, STATE_ATK_STUN_LEFT, STATE_ATK_STUN_RIGHT};
|
|
_player->state = stun[_player->dir];
|
|
_player->animFrame = 0;
|
|
_player->animDelay = _player->animCycle;
|
|
_player->sequence = 1;
|
|
}
|
|
break;
|
|
|
|
case ITEM_SLUGSLINGER: {
|
|
// it costs 1 gem to attack!
|
|
if (!amt) {
|
|
g_hdb->_sound->playSound(SND_CELLHOLDER_USE_REJECT);
|
|
g_hdb->_window->openMessageBar("Recharging...", 1);
|
|
setGemAmount(1);
|
|
return;
|
|
}
|
|
setGemAmount(amt - 1);
|
|
|
|
AIState slug[5] = {STATE_NONE, STATE_ATK_SLUG_UP, STATE_ATK_SLUG_DOWN, STATE_ATK_SLUG_LEFT, STATE_ATK_SLUG_RIGHT};
|
|
_player->state = slug[_player->dir];
|
|
_player->animFrame = 0;
|
|
_player->animDelay = _player->animCycle;
|
|
spawn(AI_SLUG_ATTACK, _player->dir, _player->tileX, _player->tileY,
|
|
nullptr, nullptr, nullptr, DIR_NONE, _player->level, 0, 0, 1);
|
|
}
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
} // switch
|
|
}
|
|
return;
|
|
}
|
|
|
|
// Puzzle Mode - throw a gem
|
|
// If this is the last gem, throw it and signal that it should come back
|
|
|
|
if (amt && (attackable || !hit)) {
|
|
int xv = xva[_player->dir];
|
|
int yv = yva[_player->dir];
|
|
nx = _player->tileX + xv;
|
|
ny = _player->tileY + yv;
|
|
|
|
spawn(AI_GEM_ATTACK, _player->dir, nx, ny, nullptr, nullptr, nullptr, DIR_NONE, _player->level, amt == 1, 0, 1);
|
|
setGemAmount(amt - 1);
|
|
animGrabbing();
|
|
return;
|
|
}
|
|
|
|
// Are we trying to use this entity?
|
|
if (hit) {
|
|
g_hdb->useEntity(hit);
|
|
return;
|
|
}
|
|
} // if kButtonB
|
|
|
|
if (!onEvenTile(_player->x, _player->y))
|
|
return;
|
|
|
|
// Is the player dead or move-locked?
|
|
if (_player->state == STATE_DEAD || _playerLock || _playerEmerging)
|
|
return;
|
|
|
|
// Are we on a touchplate and trying to move within the waiting period
|
|
if (_player->touchpWait > kPlayerTouchPWait / 4)
|
|
return;
|
|
|
|
if (g_hdb->isPPC()) {
|
|
// Are the Deliveries active?
|
|
if (g_hdb->_window->deliveriesActive())
|
|
if (!g_hdb->_ai->cinematicsActive())
|
|
return;
|
|
}
|
|
|
|
// Is a dialog active?
|
|
if (g_hdb->_window->dialogActive()) {
|
|
if (!cinematicsActive())
|
|
return;
|
|
}
|
|
|
|
// is a choice dialog active?
|
|
if (g_hdb->_window->dialogChoiceActive()) {
|
|
if (!cinematicsActive())
|
|
return;
|
|
}
|
|
|
|
if (g_hdb->isPPC()) {
|
|
// Is the Inventory active?
|
|
if (g_hdb->_window->inventoryActive())
|
|
if (!g_hdb->_ai->cinematicsActive())
|
|
return;
|
|
}
|
|
|
|
// In a cinematic?
|
|
if (_playerLock || _numWaypoints)
|
|
return;
|
|
|
|
int xv = 0, yv = 0;
|
|
if (buttons & kButtonUp)
|
|
yv = -1;
|
|
else if (buttons & kButtonDown)
|
|
yv = 1;
|
|
else if (buttons & kButtonLeft)
|
|
xv = -1;
|
|
else if (buttons & kButtonRight)
|
|
xv = 1;
|
|
else if (buttons & kButtonB) {
|
|
playerUse();
|
|
return;
|
|
}
|
|
|
|
// Check if we can move there
|
|
int nx = _player->tileX + xv;
|
|
if (!nx) // Don't allow moving to X-coordinate 0
|
|
return;
|
|
int ny = _player->tileY + yv;
|
|
|
|
int moveOK;
|
|
AIEntity *hit = legalMove(nx, ny, _player->level, &moveOK);
|
|
if (hit && walkThroughEnt(hit->type))
|
|
hit = nullptr;
|
|
|
|
if (hit || !moveOK) {
|
|
lookAtXY(nx, ny);
|
|
stopEntity(_player);
|
|
return;
|
|
}
|
|
|
|
// Walk into Lua Entity?
|
|
if (checkLuaList(_player, nx, ny))
|
|
return;
|
|
|
|
if (buttons & (kButtonUp | kButtonDown | kButtonLeft | kButtonRight)) {
|
|
int temp = _player->animFrame;
|
|
if (_player->state != STATE_MOVELEFT && _player->state != STATE_MOVERIGHT && _player->state != STATE_MOVEUP && _player->state != STATE_MOVEDOWN)
|
|
temp = 0;
|
|
setEntityGoal(_player, nx, ny);
|
|
_player->animFrame = temp;
|
|
} else
|
|
setEntityGoal(_player, nx, ny);
|
|
}
|
|
|
|
void AI::playerUse() {
|
|
static const int xv[5] = {9, 0, 0,-1, 1};
|
|
static const int yv[5] = {9,-1, 1, 0, 0};
|
|
|
|
g_hdb->setTargetXY(kTileWidth * (_player->tileX + xv[_player->dir]), kTileWidth * (_player->tileY + yv[_player->dir]));
|
|
}
|
|
} // End of Namespace
|