mirror of
https://github.com/libretro/scummvm.git
synced 2025-01-27 05:32:45 +00:00
cb949044db
These warnings were emitted by GCC when -Wcast-function-type was passed. This fixes these by refactoring so the engine only uses a single function pointer type with the "extra" parameters always present.
1772 lines
49 KiB
C++
1772 lines
49 KiB
C++
/* ScummVM - Graphic Adventure Engine
|
|
*
|
|
* ScummVM is the legal property of its developers, whose names
|
|
* are too numerous to list here. Please refer to the COPYRIGHT
|
|
* file distributed with this source distribution.
|
|
*
|
|
* This program is free software: you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
*
|
|
*/
|
|
|
|
#include "common/random.h"
|
|
|
|
#include "hdb/hdb.h"
|
|
#include "hdb/ai.h"
|
|
#include "hdb/ai-player.h"
|
|
#include "hdb/file-manager.h"
|
|
#include "hdb/gfx.h"
|
|
#include "hdb/sound.h"
|
|
#include "hdb/menu.h"
|
|
#include "hdb/lua-script.h"
|
|
#include "hdb/map.h"
|
|
#include "hdb/mpc.h"
|
|
#include "hdb/window.h"
|
|
|
|
namespace HDB {
|
|
|
|
void aiPlayerInit(AIEntity *e, int mx, int my) {
|
|
g_hdb->_ai->clearInventory();
|
|
e->aiAction = aiPlayerAction;
|
|
e->draw = g_hdb->_ai->getStandFrameDir(e);
|
|
|
|
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;
|
|
}
|
|
|
|
e->moveSpeed = kPlayerMoveSpeed;
|
|
Common::strlcpy(e->entityName, "player", 32);
|
|
g_hdb->_ai->assignPlayer(e);
|
|
}
|
|
|
|
void aiPlayerInit2(AIEntity *e, int mx, int my) {
|
|
if (!g_hdb->_ai->_clubUpGfx[0]) {
|
|
g_hdb->_ai->_weaponSelGfx = g_hdb->_gfx->loadTile(TILE_WEAPON_EQUIPPED);
|
|
g_hdb->_ai->_clubUpGfx[0] = g_hdb->_gfx->getPicGfx(CLUBUP1, -1);
|
|
g_hdb->_ai->_clubUpGfx[1] = g_hdb->_gfx->getPicGfx(CLUBUP2, -1);
|
|
g_hdb->_ai->_clubUpGfx[2] = g_hdb->_gfx->getPicGfx(CLUBUP3, -1);
|
|
g_hdb->_ai->_clubUpGfx[3] = g_hdb->_gfx->getPicGfx(CLUBUP3, -1);
|
|
|
|
g_hdb->_ai->_clubDownGfx[0] = g_hdb->_gfx->getPicGfx(CLUBDOWN1, -1);
|
|
g_hdb->_ai->_clubDownGfx[1] = g_hdb->_gfx->getPicGfx(CLUBDOWN2, -1);
|
|
g_hdb->_ai->_clubDownGfx[2] = g_hdb->_gfx->getPicGfx(CLUBDOWN3, -1);
|
|
g_hdb->_ai->_clubDownGfx[3] = g_hdb->_gfx->getPicGfx(CLUBDOWN3, -1);
|
|
|
|
g_hdb->_ai->_clubLeftGfx[0] = g_hdb->_gfx->getPicGfx(CLUBLEFT1, -1);
|
|
g_hdb->_ai->_clubLeftGfx[1] = g_hdb->_gfx->getPicGfx(CLUBLEFT2, -1);
|
|
g_hdb->_ai->_clubLeftGfx[2] = g_hdb->_gfx->getPicGfx(CLUBLEFT3, -1);
|
|
g_hdb->_ai->_clubLeftGfx[3] = g_hdb->_gfx->getPicGfx(CLUBLEFT3, -1);
|
|
|
|
g_hdb->_ai->_clubRightGfx[0] = g_hdb->_gfx->getPicGfx(CLUBRIGHT1, -1);
|
|
g_hdb->_ai->_clubRightGfx[1] = g_hdb->_gfx->getPicGfx(CLUBRIGHT2, -1);
|
|
g_hdb->_ai->_clubRightGfx[2] = g_hdb->_gfx->getPicGfx(CLUBRIGHT3, -1);
|
|
g_hdb->_ai->_clubRightGfx[3] = g_hdb->_gfx->getPicGfx(CLUBRIGHT3, -1);
|
|
|
|
g_hdb->_ai->_clubUpFrames = g_hdb->_ai->_clubDownFrames =
|
|
g_hdb->_ai->_clubLeftFrames = g_hdb->_ai->_clubRightFrames = 4;
|
|
|
|
g_hdb->_ai->_slugAttackGfx[0] = g_hdb->_gfx->loadPic(SLUG_SHOT1);
|
|
g_hdb->_ai->_slugAttackGfx[1] = g_hdb->_gfx->loadPic(SLUG_SHOT2);
|
|
g_hdb->_ai->_slugAttackGfx[2] = g_hdb->_gfx->loadPic(SLUG_SHOT3);
|
|
g_hdb->_ai->_slugAttackGfx[3] = g_hdb->_gfx->loadPic(SLUG_SHOT4);
|
|
|
|
int32 size = g_hdb->_fileMan->getLength("shock_spark_sit01", TYPE_TILE32);
|
|
g_hdb->_ai->_stunLightningGfx[0] = g_hdb->_gfx->getTileGfx("shock_spark_sit01", size);
|
|
size = g_hdb->_fileMan->getLength("shock_spark_sit02", TYPE_TILE32);
|
|
g_hdb->_ai->_stunLightningGfx[1] = g_hdb->_gfx->getTileGfx("shock_spark_sit02", size);
|
|
size = g_hdb->_fileMan->getLength("shock_spark_sit03", TYPE_TILE32);
|
|
g_hdb->_ai->_stunLightningGfx[2] = g_hdb->_gfx->getTileGfx("shock_spark_sit03", size);
|
|
size = g_hdb->_fileMan->getLength("shock_spark_sit04", TYPE_TILE32);
|
|
g_hdb->_ai->_stunLightningGfx[3] = g_hdb->_gfx->getTileGfx("shock_spark_sit04", size);
|
|
|
|
size = g_hdb->_fileMan->getLength("starstun_sit01", TYPE_TILE32);
|
|
g_hdb->_ai->_stunnedGfx[0] = g_hdb->_gfx->getTileGfx("starstun_sit01", size);
|
|
size = g_hdb->_fileMan->getLength("starstun_sit02", TYPE_TILE32);
|
|
g_hdb->_ai->_stunnedGfx[1] = g_hdb->_gfx->getTileGfx("starstun_sit02", size);
|
|
size = g_hdb->_fileMan->getLength("starstun_sit03", TYPE_TILE32);
|
|
g_hdb->_ai->_stunnedGfx[2] = g_hdb->_gfx->getTileGfx("starstun_sit03", size);
|
|
size = g_hdb->_fileMan->getLength("starstun_sit04", TYPE_TILE32);
|
|
g_hdb->_ai->_stunnedGfx[3] = g_hdb->_gfx->getTileGfx("starstun_sit04", size);
|
|
}
|
|
|
|
e->draw = g_hdb->_ai->getStandFrameDir(e);
|
|
}
|
|
|
|
void aiPlayerAction(AIEntity *e, int mx, int my) {
|
|
static const AIState stand[5] = {STATE_NONE, STATE_STANDUP, STATE_STANDDOWN, STATE_STANDLEFT, STATE_STANDRIGHT};
|
|
static const int xvAhead[5] = {9, 0, 0, -1, 1};
|
|
static const int yvAhead[5] = {9, -1, 1, 0, 0};
|
|
|
|
AIEntity *hit = nullptr;
|
|
|
|
// Draw the STUN lightning if it exists
|
|
if (e->sequence) {
|
|
e->aiDraw = aiPlayerDraw;
|
|
g_hdb->_sound->playSound(SND_STUNNER_FIRE);
|
|
hit = g_hdb->_ai->findEntity(e->tileX + xvAhead[e->dir], e->tileY + yvAhead[e->dir]);
|
|
if (hit)
|
|
switch (hit->type) {
|
|
case AI_MEERKAT:
|
|
if (hit->sequence > 2)
|
|
g_hdb->_ai->stunEnemy(hit, 8);
|
|
break;
|
|
case AI_ICEPUFF:
|
|
if (hit->state == STATE_ICEP_APPEAR || hit->state == STATE_ICEP_THROWDOWN || hit->state == STATE_ICEP_THROWLEFT || hit->state == STATE_ICEP_THROWRIGHT) {
|
|
g_hdb->_ai->addAnimateTarget(hit->x, hit->y, 0, 3, ANIM_NORMAL, false, false, GROUP_STEAM_PUFF_SIT);
|
|
g_hdb->_ai->stunEnemy(hit, 8);
|
|
}
|
|
break;
|
|
case AI_BADFAIRY:
|
|
case AI_GOODFAIRY:
|
|
case AI_CHICKEN:
|
|
case AI_OMNIBOT:
|
|
case AI_TURNBOT:
|
|
case AI_PUSHBOT:
|
|
case AI_DEADEYE:
|
|
case AI_FATFROG:
|
|
case AI_BUZZFLY:
|
|
case AI_MAINTBOT:
|
|
case AI_RIGHTBOT:
|
|
case AI_GATEPUDDLE:
|
|
g_hdb->_ai->stunEnemy(hit, 8);
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
hit = g_hdb->_ai->findEntity(e->tileX + (xvAhead[e->dir] << 1), e->tileY + (yvAhead[e->dir] << 1));
|
|
if (hit)
|
|
switch (hit->type) {
|
|
case AI_MEERKAT:
|
|
if (hit->sequence > 2)
|
|
g_hdb->_ai->stunEnemy(hit, 8);
|
|
break;
|
|
case AI_ICEPUFF:
|
|
if (hit->state == STATE_ICEP_APPEAR || hit->state == STATE_ICEP_THROWDOWN || hit->state == STATE_ICEP_THROWLEFT || hit->state == STATE_ICEP_THROWRIGHT) {
|
|
g_hdb->_ai->addAnimateTarget(hit->x, hit->y, 0, 3, ANIM_NORMAL, false, false, GROUP_STEAM_PUFF_SIT);
|
|
g_hdb->_ai->stunEnemy(hit, 8);
|
|
}
|
|
break;
|
|
case AI_BADFAIRY:
|
|
case AI_GOODFAIRY:
|
|
case AI_CHICKEN:
|
|
case AI_OMNIBOT:
|
|
case AI_TURNBOT:
|
|
case AI_PUSHBOT:
|
|
case AI_DEADEYE:
|
|
case AI_FATFROG:
|
|
case AI_BUZZFLY:
|
|
case AI_MAINTBOT:
|
|
case AI_RIGHTBOT:
|
|
case AI_GATEPUDDLE:
|
|
g_hdb->_ai->stunEnemy(hit, 8);
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
int xOff[] = {0, 0, -8,-16};
|
|
int yOff[] = {-8,-24,-16,-16};
|
|
// If the player is supposed to animate for abit, check for it here
|
|
switch (e->state) {
|
|
case STATE_GRABUP:
|
|
case STATE_GRABDOWN:
|
|
case STATE_GRABLEFT:
|
|
case STATE_GRABRIGHT:
|
|
if (!e->animFrame--) {
|
|
// Done with the Grabbing Animation, switch to standing
|
|
switch (e->state) {
|
|
case STATE_GRABUP:
|
|
e->draw = e->standupGfx[0];
|
|
e->state = STATE_STANDUP;
|
|
break;
|
|
case STATE_GRABDOWN:
|
|
e->draw = e->standdownGfx[0];
|
|
e->state = STATE_STANDDOWN;
|
|
break;
|
|
case STATE_GRABLEFT:
|
|
e->draw = e->standleftGfx[0];
|
|
e->state = STATE_STANDLEFT;
|
|
break;
|
|
case STATE_GRABRIGHT:
|
|
e->draw = e->standrightGfx[0];
|
|
e->state = STATE_STANDRIGHT;
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
e->animDelay = 1;
|
|
e->animCycle = 1;
|
|
}
|
|
break;
|
|
case STATE_ATK_CLUB_UP:
|
|
case STATE_ATK_CLUB_DOWN:
|
|
case STATE_ATK_CLUB_LEFT:
|
|
case STATE_ATK_CLUB_RIGHT:
|
|
g_hdb->_ai->setPlayerInvisible(true);
|
|
e->aiDraw = aiPlayerDraw;
|
|
e->drawXOff = xOff[e->state - STATE_ATK_CLUB_UP];
|
|
e->drawYOff = yOff[e->state - STATE_ATK_CLUB_UP];
|
|
switch (e->state) {
|
|
case STATE_ATK_CLUB_UP:
|
|
cycleFrames(e, g_hdb->_ai->_clubUpFrames);
|
|
break;
|
|
case STATE_ATK_CLUB_DOWN:
|
|
cycleFrames(e, g_hdb->_ai->_clubDownFrames);
|
|
break;
|
|
case STATE_ATK_CLUB_LEFT:
|
|
cycleFrames(e, g_hdb->_ai->_clubLeftFrames);
|
|
break;
|
|
case STATE_ATK_CLUB_RIGHT:
|
|
cycleFrames(e, g_hdb->_ai->_clubRightFrames);
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
// Whack!
|
|
if ((e->animFrame >= 1) && (e->animDelay == e->animCycle)) {
|
|
switch (e->dir) {
|
|
case DIR_UP:
|
|
hit = g_hdb->_ai->playerCollision(32, 0, 16, 16);
|
|
break;
|
|
case DIR_DOWN:
|
|
hit = g_hdb->_ai->playerCollision(0, 32, 16, 16);
|
|
break;
|
|
case DIR_LEFT:
|
|
hit = g_hdb->_ai->playerCollision(16, 16, 32, 0);
|
|
break;
|
|
case DIR_RIGHT:
|
|
hit = g_hdb->_ai->playerCollision(16, 16, 0, 32);
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
if (hit && hit->level == e->level && !hit->stunnedWait) {
|
|
switch (hit->type) {
|
|
case AI_MEERKAT:
|
|
if (hit->sequence > 2) // out of the ground?
|
|
g_hdb->_ai->stunEnemy(hit, 2);
|
|
break;
|
|
case AI_ICEPUFF:
|
|
if (hit->state == STATE_ICEP_APPEAR ||
|
|
hit->state == STATE_ICEP_THROWDOWN ||
|
|
hit->state == STATE_ICEP_THROWLEFT ||
|
|
hit->state == STATE_ICEP_THROWRIGHT)
|
|
g_hdb->_ai->stunEnemy(hit, 2);
|
|
break;
|
|
case AI_CHICKEN:
|
|
g_hdb->_ai->addAnimateTarget(hit->x, hit->y, 0, 3, ANIM_NORMAL, false, false, GROUP_STEAM_PUFF_SIT);
|
|
g_hdb->_ai->removeEntity(hit);
|
|
break;
|
|
case AI_BADFAIRY:
|
|
case AI_GOODFAIRY:
|
|
case AI_OMNIBOT:
|
|
case AI_TURNBOT:
|
|
case AI_PUSHBOT:
|
|
case AI_DEADEYE:
|
|
case AI_FATFROG:
|
|
case AI_BUZZFLY:
|
|
case AI_MAINTBOT:
|
|
case AI_RIGHTBOT:
|
|
case AI_SHOCKBOT:
|
|
case AI_GATEPUDDLE:
|
|
g_hdb->_ai->stunEnemy(hit, 2);
|
|
g_hdb->_sound->playSound(g_hdb->_ai->metalOrFleshSND(hit));
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
if ((!e->animFrame) && (e->animDelay == e->animCycle)) {
|
|
e->state = stand[e->dir];
|
|
e->aiDraw = nullptr;
|
|
switch (e->state) {
|
|
case STATE_ATK_CLUB_UP:
|
|
e->draw = e->standupGfx[0];
|
|
break;
|
|
case STATE_ATK_CLUB_DOWN:
|
|
e->draw = e->standdownGfx[0];
|
|
break;
|
|
case STATE_ATK_CLUB_LEFT:
|
|
e->draw = e->standleftGfx[0];
|
|
break;
|
|
case STATE_ATK_CLUB_RIGHT:
|
|
e->draw = e->standrightGfx[0];
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
g_hdb->_ai->setPlayerInvisible(false);
|
|
e->drawXOff = e->drawYOff = 0;
|
|
}
|
|
return;
|
|
case STATE_ATK_STUN_UP:
|
|
e->draw = g_hdb->_ai->_stunUpGfx[e->animFrame];
|
|
cycleFrames(e, g_hdb->_ai->_stunUpFrames);
|
|
if (!e->animFrame && e->animDelay == e->animCycle) {
|
|
e->state = stand[e->dir];
|
|
e->aiDraw = nullptr;
|
|
e->sequence = 0;
|
|
}
|
|
return;
|
|
case STATE_ATK_STUN_DOWN:
|
|
e->draw = g_hdb->_ai->_stunDownGfx[e->animFrame];
|
|
cycleFrames(e, g_hdb->_ai->_stunDownFrames);
|
|
if (!e->animFrame && e->animDelay == e->animCycle) {
|
|
e->state = stand[e->dir];
|
|
e->aiDraw = nullptr;
|
|
e->sequence = 0;
|
|
}
|
|
return;
|
|
case STATE_ATK_STUN_LEFT:
|
|
e->draw = g_hdb->_ai->_stunLeftGfx[e->animFrame];
|
|
cycleFrames(e, g_hdb->_ai->_stunLeftFrames);
|
|
if (!e->animFrame && e->animDelay == e->animCycle) {
|
|
e->state = stand[e->dir];
|
|
e->aiDraw = nullptr;
|
|
e->sequence = 0;
|
|
}
|
|
return;
|
|
case STATE_ATK_STUN_RIGHT:
|
|
e->draw = g_hdb->_ai->_stunRightGfx[e->animFrame];
|
|
cycleFrames(e, g_hdb->_ai->_stunRightFrames);
|
|
if (!e->animFrame && e->animDelay == e->animCycle) {
|
|
e->state = stand[e->dir];
|
|
e->aiDraw = nullptr;
|
|
e->sequence = 0;
|
|
}
|
|
return;
|
|
case STATE_ATK_SLUG_UP:
|
|
e->draw = g_hdb->_ai->_slugUpGfx[e->animFrame];
|
|
cycleFrames(e, g_hdb->_ai->_slugUpFrames);
|
|
if (!e->animFrame && e->animDelay == e->animCycle) {
|
|
e->state = stand[e->dir];
|
|
e->sequence = 0;
|
|
}
|
|
return;
|
|
case STATE_ATK_SLUG_DOWN:
|
|
e->draw = g_hdb->_ai->_slugDownGfx[e->animFrame];
|
|
cycleFrames(e, g_hdb->_ai->_slugDownFrames);
|
|
if (!e->animFrame && e->animDelay == e->animCycle) {
|
|
e->state = stand[e->dir];
|
|
e->sequence = 0;
|
|
}
|
|
return;
|
|
case STATE_ATK_SLUG_LEFT:
|
|
e->draw = g_hdb->_ai->_slugLeftGfx[e->animFrame];
|
|
cycleFrames(e, g_hdb->_ai->_slugLeftFrames);
|
|
if (!e->animFrame && e->animDelay == e->animCycle) {
|
|
e->state = stand[e->dir];
|
|
e->sequence = 0;
|
|
}
|
|
return;
|
|
case STATE_ATK_SLUG_RIGHT:
|
|
e->draw = g_hdb->_ai->_slugRightGfx[e->animFrame];
|
|
cycleFrames(e, g_hdb->_ai->_slugRightFrames);
|
|
if (!e->animFrame && e->animDelay == e->animCycle) {
|
|
e->state = stand[e->dir];
|
|
e->sequence = 0;
|
|
}
|
|
return;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
// If the touchplate wait is on, keep it timing
|
|
if (e->touchpWait) {
|
|
e->touchpWait--;
|
|
if (!e->touchpWait)
|
|
e->touchpTile = -e->touchpTile;
|
|
} else if (e->touchpTile < 0 && (e->touchpX != e->tileX || e->touchpY != e->tileY)) {
|
|
g_hdb->_ai->checkActionList(e, e->touchpX, e->touchpY, false);
|
|
g_hdb->_map->setMapBGTileIndex(e->touchpX, e->touchpY, -e->touchpTile);
|
|
e->touchpX = e->touchpY = e->touchpTile = 0;
|
|
}
|
|
|
|
// If the player is moving somewhere, animate him
|
|
int bgFlags, fgFlags;
|
|
if (e->goalX) {
|
|
if (onEvenTile(e->x, e->y)) {
|
|
g_hdb->_ai->playerOnIce() ? g_hdb->_sound->playSound(SND_STEPS_ICE) : g_hdb->_sound->playSound(SND_FOOTSTEPS);
|
|
|
|
// Did we just fall down a PLUMMET?
|
|
bgFlags = g_hdb->_map->getMapBGTileFlags(e->tileX, e->tileY);
|
|
fgFlags = g_hdb->_map->getMapFGTileFlags(e->tileX, e->tileY);
|
|
if ((bgFlags & kFlagPlummet) && !(fgFlags & kFlagGrating) && !g_hdb->_ai->playerDead()) {
|
|
g_hdb->_ai->killPlayer(DEATH_PLUMMET);
|
|
g_hdb->_ai->animEntFrames(e);
|
|
return;
|
|
}
|
|
}
|
|
g_hdb->_ai->animateEntity(e);
|
|
} else {
|
|
// Sometimes the fading stays black
|
|
if (!g_hdb->_ai->cinematicsActive() && g_hdb->_gfx->isFadeStaying())
|
|
g_hdb->_gfx->turnOffFade();
|
|
|
|
// Did we just fall down a PLUMMET?
|
|
bgFlags = g_hdb->_map->getMapBGTileFlags(e->tileX, e->tileY);
|
|
fgFlags = g_hdb->_map->getMapFGTileFlags(e->tileX, e->tileY);
|
|
if ((bgFlags & kFlagPlummet) && !(fgFlags & kFlagGrating) && !g_hdb->_ai->playerDead()) {
|
|
g_hdb->_ai->killPlayer(DEATH_PLUMMET);
|
|
g_hdb->_ai->animEntFrames(e);
|
|
return;
|
|
}
|
|
|
|
// Standing on a TouchPlate will activate something WHILE standing on it
|
|
int bgTile = g_hdb->_ai->checkForTouchplate(e->tileX, e->tileY);
|
|
if (bgTile && !e->touchpWait && !e->touchpTile) {
|
|
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;
|
|
g_hdb->_ai->stopEntity(e);
|
|
}
|
|
}
|
|
g_hdb->_ai->animEntFrames(e);
|
|
}
|
|
}
|
|
|
|
void aiPlayerDraw(AIEntity *e, int mx, int my) {
|
|
switch (e->state) {
|
|
case STATE_ATK_CLUB_UP:
|
|
g_hdb->_ai->_clubUpGfx[e->animFrame]->drawMasked(e->x + e->drawXOff - mx, e->y + e->drawYOff - my);
|
|
break;
|
|
case STATE_ATK_CLUB_DOWN:
|
|
g_hdb->_ai->_clubDownGfx[e->animFrame]->drawMasked(e->x + e->drawXOff - mx, e->y + e->drawYOff - my);
|
|
break;
|
|
case STATE_ATK_CLUB_LEFT:
|
|
g_hdb->_ai->_clubLeftGfx[e->animFrame]->drawMasked(e->x + e->drawXOff - mx, e->y + e->drawYOff - my);
|
|
break;
|
|
case STATE_ATK_CLUB_RIGHT:
|
|
g_hdb->_ai->_clubRightGfx[e->animFrame]->drawMasked(e->x + e->drawXOff - mx, e->y + e->drawYOff - my);
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
if (e->sequence) {
|
|
static int frame = 0;
|
|
switch (e->dir) {
|
|
case DIR_UP:
|
|
g_hdb->_ai->_stunLightningGfx[frame]->drawMasked(e->x - mx, e->y - 32 - my);
|
|
g_hdb->_ai->_stunLightningGfx[frame]->drawMasked(e->x - mx, e->y - 64 - my);
|
|
break;
|
|
case DIR_DOWN:
|
|
g_hdb->_ai->_stunLightningGfx[frame]->drawMasked(e->x - mx, e->y + 32 - my);
|
|
g_hdb->_ai->_stunLightningGfx[frame]->drawMasked(e->x - mx, e->y + 64 - my);
|
|
break;
|
|
case DIR_LEFT:
|
|
g_hdb->_ai->_stunLightningGfx[frame]->drawMasked(e->x - 32 - mx, e->y - my);
|
|
g_hdb->_ai->_stunLightningGfx[frame]->drawMasked(e->x - 64 - mx, e->y - my);
|
|
break;
|
|
case DIR_RIGHT:
|
|
g_hdb->_ai->_stunLightningGfx[frame]->drawMasked(e->x + 32 - mx, e->y - my);
|
|
g_hdb->_ai->_stunLightningGfx[frame]->drawMasked(e->x + 64 - mx, e->y - my);
|
|
break;
|
|
case DIR_NONE:
|
|
default:
|
|
break;
|
|
}
|
|
|
|
frame = (frame + 1) & 3;
|
|
}
|
|
}
|
|
|
|
void aiGemAttackInit(AIEntity *e, int mx, int my) {
|
|
static const int xv[5] = {9, 0, 0, -1, 1};
|
|
static const int yv[5] = {9, -1, 1, 0, 0};
|
|
|
|
e->moveSpeed = kPlayerMoveSpeed << 1;
|
|
g_hdb->_ai->setEntityGoal(e, e->tileX + xv[e->dir], e->tileY + yv[e->dir]);
|
|
e->state = STATE_MOVEDOWN; // so it will draw & animate
|
|
e->sequence = 0; // flying out at something
|
|
e->aiAction = aiGemAttackAction;
|
|
e->draw = e->movedownGfx[0];
|
|
g_hdb->_sound->playSound(SND_GEM_THROW);
|
|
}
|
|
|
|
void aiGemAttackAction(AIEntity *e, int mx, int my) {
|
|
static const int xv[5] = {9, 0, 0, -1, 1};
|
|
static const int yv[5] = {9, -1, 1, 0, 0};
|
|
|
|
switch (e->sequence) {
|
|
// flying out at something
|
|
case 0:
|
|
if (e->goalX)
|
|
g_hdb->_ai->animateEntity(e);
|
|
else {
|
|
g_hdb->_ai->checkActionList(e, e->tileX, e->tileY, false);
|
|
g_hdb->_ai->checkAutoList(e, e->tileX, e->tileY);
|
|
|
|
AIEntity *hit = g_hdb->_ai->findEntityIgnore(e->tileX, e->tileY, e);
|
|
uint32 bgFlags = g_hdb->_map->getMapBGTileFlags(e->tileX, e->tileY);
|
|
uint32 fgFlags = g_hdb->_map->getMapFGTileFlags(e->tileX, e->tileY);
|
|
int result = (e->level == 1 ? (bgFlags & (kFlagSolid)) : !(fgFlags & kFlagGrating) && (bgFlags & (kFlagSolid)));
|
|
if (hit) {
|
|
switch (hit->type) {
|
|
case AI_CHICKEN:
|
|
g_hdb->_ai->addAnimateTarget(hit->x, hit->y, 0, 3, ANIM_NORMAL, false, false, GROUP_STEAM_PUFF_SIT);
|
|
g_hdb->_ai->removeEntity(hit);
|
|
g_hdb->_sound->playSound(SND_CHICKEN_BAGAWK);
|
|
break;
|
|
case AI_BADFAIRY:
|
|
g_hdb->_ai->stunEnemy(hit, 2);
|
|
g_hdb->_ai->addAnimateTarget(hit->x, hit->y, 0, 3, ANIM_NORMAL, false, false, GEM_FLASH);
|
|
break;
|
|
case AI_NONE:
|
|
if (hit->value1 == (int)AI_DRAGON) {
|
|
// pull dragon's coords out of "lua_func_use" string.
|
|
char num1[4], num2[4];
|
|
memset(num1, 0, 4);
|
|
memset(num2, 0, 4);
|
|
memcpy(num1, hit->luaFuncUse, 3);
|
|
memcpy(num2, hit->luaFuncUse + 3, 3);
|
|
|
|
g_hdb->_sound->playSound(SND_CLUB_HIT_FLESH);
|
|
AIEntity *found = g_hdb->_ai->findEntity(atoi(num1), atoi(num2));
|
|
if (found)
|
|
aiDragonWake(found, 0, 0);
|
|
}
|
|
g_hdb->_ai->addAnimateTarget(e->x, e->y, 0, 3, ANIM_NORMAL, false, false, GEM_FLASH);
|
|
g_hdb->_sound->playSound(SND_INV_SELECT);
|
|
break;
|
|
case AI_DRAGON:
|
|
g_hdb->_sound->playSound(SND_CLUB_HIT_FLESH);
|
|
aiDragonWake(hit, 0, 0);
|
|
// fallthrough
|
|
default:
|
|
g_hdb->_ai->addAnimateTarget(e->x, e->y, 0, 3, ANIM_NORMAL, false, false, GEM_FLASH);
|
|
g_hdb->_sound->playSound(SND_CLUB_HIT_FLESH);
|
|
}
|
|
if (e->value1)
|
|
e->sequence = 1;
|
|
else
|
|
g_hdb->_ai->removeEntity(e); // bye bye!
|
|
return;
|
|
} else if (result) { // hit a wall
|
|
g_hdb->_ai->addAnimateTarget(e->x, e->y, 0, 3, ANIM_NORMAL, false, false, GEM_FLASH);
|
|
g_hdb->_sound->playSound(SND_INV_SELECT);
|
|
// come back to daddy?
|
|
if (e->value1)
|
|
e->sequence = 1;
|
|
else {
|
|
g_hdb->_ai->removeEntity(e);
|
|
return;
|
|
}
|
|
} else {
|
|
g_hdb->_ai->setEntityGoal(e, e->tileX + xv[e->dir], e->tileY + yv[e->dir]);
|
|
e->state = STATE_MOVEDOWN; // so it will draw & animate
|
|
}
|
|
g_hdb->_ai->animateEntity(e);
|
|
}
|
|
break;
|
|
// coming back to daddy?
|
|
case 1:
|
|
{
|
|
AIEntity *p = g_hdb->_ai->getPlayer();
|
|
if (e->x < p->x)
|
|
e->x++;
|
|
else
|
|
e->x--;
|
|
|
|
if (e->y < p->y)
|
|
e->y++;
|
|
else
|
|
e->y--;
|
|
|
|
if (abs(e->x - p->x) < 4 && abs(e->y - p->y) < 4)
|
|
{
|
|
int amt = g_hdb->_ai->getGemAmount();
|
|
g_hdb->_ai->setGemAmount(amt + 1);
|
|
g_hdb->_ai->addAnimateTarget(e->x, e->y, 0, 3, ANIM_NORMAL, false, false, GEM_FLASH);
|
|
g_hdb->_ai->removeEntity(e);
|
|
g_hdb->_sound->playSound(SND_GET_GEM);
|
|
}
|
|
}
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
void aiChickenAction(AIEntity *e, int mx, int my) {
|
|
static int delay = 64;
|
|
|
|
if (g_hdb->_map->checkEntOnScreen(e) && !delay) {
|
|
g_hdb->_sound->playSound(SND_CHICKEN_AMBIENT);
|
|
delay = g_hdb->_rnd->getRandomNumber(127) + 160;
|
|
aiChickenUse(e, 0, 0);
|
|
}
|
|
|
|
if (delay)
|
|
delay--;
|
|
|
|
if (e->goalX)
|
|
g_hdb->_ai->animateEntity(e);
|
|
else
|
|
g_hdb->_ai->animEntFrames(e);
|
|
}
|
|
|
|
void aiChickenUse(AIEntity *e, int mx, int my) {
|
|
g_hdb->_sound->playSound(SND_CHICKEN_BAGAWK);
|
|
}
|
|
|
|
void aiChickenInit(AIEntity *e, int mx, int my) {
|
|
e->aiUse = aiChickenUse;
|
|
e->aiAction = aiChickenAction;
|
|
}
|
|
|
|
void aiChickenInit2(AIEntity *e, int mx, int my) {
|
|
e->draw = e->standdownGfx[0];
|
|
}
|
|
|
|
void aiDollyInit(AIEntity *e, int mx, int my) {
|
|
e->moveSpeed = kPlayerMoveSpeed >> 1;
|
|
e->aiAction = aiGenericAction;
|
|
}
|
|
|
|
void aiDollyInit2(AIEntity *e, int mx, int my) {
|
|
e->draw = e->movedownGfx[0];
|
|
}
|
|
|
|
void aiSergeantInit(AIEntity *e, int mx, int my) {
|
|
e->moveSpeed = kPlayerMoveSpeed >> 1;
|
|
if (e->value1)
|
|
e->aiAction = aiSergeantAction;
|
|
}
|
|
|
|
void aiSergeantInit2(AIEntity *e, int mx, int my) {
|
|
e->draw = g_hdb->_ai->getStandFrameDir(e);
|
|
}
|
|
|
|
void aiSergeantAction(AIEntity *e, int mx, int my) {
|
|
if (e->goalX) {
|
|
g_hdb->_sound->playSound(SND_FOOTSTEPS);
|
|
g_hdb->_ai->animateEntity(e);
|
|
} else
|
|
g_hdb->_ai->animEntFrames(e);
|
|
}
|
|
|
|
void aiSpacedudeInit(AIEntity *e, int mx, int my) {
|
|
e->moveSpeed = kPlayerMoveSpeed >> 1;
|
|
if (e->value1)
|
|
e->aiAction = aiGenericAction;
|
|
}
|
|
|
|
void aiSpacedudeInit2(AIEntity *e, int mx, int my) {
|
|
e->standdownFrames = 1;
|
|
e->standdownGfx[0] = e->movedownGfx[0];
|
|
e->standupFrames = 1;
|
|
e->standupGfx[0] = e->moveupGfx[0];
|
|
e->standleftFrames = 1;
|
|
e->standleftGfx[0] = e->moveleftGfx[0];
|
|
e->standrightFrames = 1;
|
|
e->standrightGfx[0] = e->moverightGfx[0];
|
|
e->draw = g_hdb->_ai->getStandFrameDir(e);
|
|
}
|
|
|
|
void aiCrateAction(AIEntity *e, int mx, int my) {
|
|
// if crate isn't moving somewhere, don't move it
|
|
if (!e->goalX) {
|
|
// crate is stopped in the water... should it continue downstream?
|
|
// not if it's marked by the Number of the Beast!
|
|
if (e->state == STATE_FLOATING) {
|
|
if (e->value1 != 0x666) {
|
|
int flags = g_hdb->_map->getMapBGTileFlags(e->tileX, e->tileY);
|
|
if (flags & (kFlagPushRight | kFlagPushLeft | kFlagPushUp | kFlagPushDown)) {
|
|
g_hdb->_ai->setEntityGoal(e, e->tileX, e->tileY);
|
|
g_hdb->_ai->animateEntity(e);
|
|
} else
|
|
g_hdb->_ai->animEntFrames(e);
|
|
} else
|
|
g_hdb->_ai->animEntFrames(e);
|
|
}
|
|
return;
|
|
}
|
|
|
|
g_hdb->_ai->animateEntity(e);
|
|
}
|
|
|
|
void aiCrateInit2(AIEntity *e, int mx, int my) {
|
|
// point all crate move frames to the standing one
|
|
e->movedownFrames =
|
|
e->moveleftFrames =
|
|
e->moverightFrames =
|
|
e->moveupFrames = 1;
|
|
|
|
e->movedownGfx[0] =
|
|
e->moveupGfx[0] =
|
|
e->moveleftGfx[0] =
|
|
e->moverightGfx[0] = e->standdownGfx[0];
|
|
|
|
e->draw = e->standdownGfx[0]; // standing frame - doesn't move
|
|
}
|
|
|
|
void aiCrateInit(AIEntity *e, int mx, int my) {
|
|
e->moveSpeed = kPushMoveSpeed;
|
|
e->aiAction = aiCrateAction;
|
|
e->value1 = 0;
|
|
}
|
|
|
|
void aiBarrelLightAction(AIEntity *e, int mx, int my) {
|
|
if (!e->goalX) {
|
|
if (e->state == STATE_FLOATING)
|
|
g_hdb->_ai->animEntFrames(e);
|
|
return;
|
|
}
|
|
|
|
g_hdb->_ai->animateEntity(e);
|
|
}
|
|
|
|
void aiBarrelLightInit2(AIEntity *e, int mx, int my) {
|
|
// point all light barrel move frames to the standing one
|
|
e->movedownFrames =
|
|
e->moveleftFrames =
|
|
e->moverightFrames =
|
|
e->moveupFrames = 1;
|
|
|
|
e->movedownGfx[0] =
|
|
e->moveupGfx[0] =
|
|
e->moveleftGfx[0] =
|
|
e->moverightGfx[0] = e->standdownGfx[0];
|
|
|
|
e->draw = e->standdownGfx[0]; // standing frame - doesn't move
|
|
}
|
|
|
|
void aiBarrelLightInit(AIEntity *e, int mx, int my) {
|
|
e->moveSpeed = kPushMoveSpeed;
|
|
e->aiAction = aiBarrelLightAction;
|
|
}
|
|
|
|
void aiBarrelHeavyAction(AIEntity *e, int mx, int my) {
|
|
if (!e->goalX) {
|
|
if (e->state == STATE_FLOATING)
|
|
g_hdb->_ai->animEntFrames(e);
|
|
return;
|
|
}
|
|
|
|
g_hdb->_ai->animateEntity(e);
|
|
}
|
|
|
|
void aiBarrelHeavyInit2(AIEntity *e, int mx, int my) {
|
|
// point all heavy barrel move frames to the standing one
|
|
e->movedownFrames =
|
|
e->moveleftFrames =
|
|
e->moverightFrames =
|
|
e->moveupFrames = 1;
|
|
|
|
e->movedownGfx[0] =
|
|
e->moveupGfx[0] =
|
|
e->moveleftGfx[0] =
|
|
e->moverightGfx[0] = e->standdownGfx[0];
|
|
|
|
e->draw = e->standdownGfx[0]; // standing frame - doesn't move
|
|
}
|
|
|
|
void aiBarrelHeavyInit(AIEntity *e, int mx, int my) {
|
|
e->moveSpeed = kPushMoveSpeed;
|
|
e->aiAction = aiBarrelHeavyAction;
|
|
}
|
|
|
|
void aiBarrelExplode(AIEntity *e, int mx, int my) {
|
|
e->state = STATE_EXPLODING;
|
|
e->animDelay = e->animCycle;
|
|
e->animFrame = 0;
|
|
|
|
if (!g_hdb->isDemo())
|
|
g_hdb->_sound->playSound(SND_BARREL_EXPLODE);
|
|
|
|
g_hdb->_map->setBoomBarrel(e->tileX, e->tileY, 0);
|
|
}
|
|
|
|
void aiBarrelExplodeInit(AIEntity *e, int mx, int my) {
|
|
e->moveSpeed = kPushMoveSpeed;
|
|
e->aiAction = aiBarrelExplodeAction;
|
|
g_hdb->_map->setBoomBarrel(e->tileX, e->tileY, 1);
|
|
}
|
|
|
|
void aiBarrelExplodeInit2(AIEntity *e, int mx, int my) {
|
|
// point all exploding barrel MOVE frames to the standing one
|
|
e->blinkFrames =
|
|
e->movedownFrames =
|
|
e->moveleftFrames =
|
|
e->moverightFrames =
|
|
e->moveupFrames = 1;
|
|
|
|
e->blinkGfx[0] =
|
|
e->movedownGfx[0] =
|
|
e->moveupGfx[0] =
|
|
e->moveleftGfx[0] =
|
|
e->moverightGfx[0] = e->standdownGfx[0];
|
|
|
|
e->draw = e->standdownGfx[0]; // standing frame - doesn't move
|
|
}
|
|
|
|
void aiBarrelExplodeAction(AIEntity *e, int mx, int my) {
|
|
if (e->goalX)
|
|
g_hdb->_ai->animateEntity(e);
|
|
else if (e->state == STATE_EXPLODING)
|
|
g_hdb->_ai->animEntFrames(e);
|
|
}
|
|
|
|
void aiBarrelExplodeSpread(AIEntity *e, int mx, int my) {
|
|
static const int xv1[4] = {-1, 1, -1, 0};
|
|
static const int yv1[4] = {-1, -1, 0, -1};
|
|
static const int xv2[4] = {1, 0, 1, -1};
|
|
static const int yv2[4] = {0, 1, 1, 1};
|
|
|
|
int x = e->tileX;
|
|
int y = e->tileY;
|
|
int index = e->animFrame;
|
|
|
|
// are we just starting an explosion ring?
|
|
if (e->animDelay != e->animCycle)
|
|
return;
|
|
|
|
// the animation frame is the index into which set of 2 explosions to spawn
|
|
int xv = xv1[index];
|
|
int yv = yv1[index];
|
|
|
|
// explosion 1: check to see if we can explode (non-solid tile)
|
|
// if so, spawn it and mark it in the explosion matrix
|
|
if (!(g_hdb->_map->getMapBGTileFlags(x + xv, y + yv) & kFlagSolid) && !g_hdb->_map->explosionExist(x + xv, y + yv)) {
|
|
aiBarrelBlowup(e, x + xv, y + yv);
|
|
// are we blowing up on another BOOMBARREL? if so, start it exploding.
|
|
AIEntity *e2 = g_hdb->_ai->findEntity(x + xv, y + yv);
|
|
if (e2 && e2->state != STATE_EXPLODING) {
|
|
switch (e2->type) {
|
|
case AI_GUY:
|
|
g_hdb->_ai->killPlayer(DEATH_FRIED);
|
|
break;
|
|
case AI_BOOMBARREL:
|
|
aiBarrelExplode(e2, 0, 0);
|
|
break;
|
|
case AI_OMNIBOT:
|
|
case AI_TURNBOT:
|
|
case AI_SHOCKBOT:
|
|
case AI_RIGHTBOT:
|
|
case AI_PUSHBOT:
|
|
case AI_RAILRIDER:
|
|
case AI_MAINTBOT:
|
|
case AI_DEADEYE:
|
|
case AI_FATFROG:
|
|
case AI_ICEPUFF:
|
|
case AI_MEERKAT:
|
|
case AI_BUZZFLY:
|
|
case AI_GOODFAIRY:
|
|
case AI_GATEPUDDLE:
|
|
case AI_BADFAIRY:
|
|
g_hdb->_ai->addAnimateTarget(x * kTileWidth,
|
|
y * kTileHeight, 0, 3, ANIM_NORMAL, false, false, GROUP_EXPLOSION_BOOM_SIT);
|
|
if (e2->type != AI_LASERBEAM)
|
|
g_hdb->_ai->removeEntity(e2);
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
xv = xv2[index];
|
|
yv = yv2[index];
|
|
|
|
// explosion 2: check to see if we can explode (non-solid tile)
|
|
// if so, spawn it and mark it in the explosion matrix
|
|
|
|
if (!(g_hdb->_map->getMapBGTileFlags(x + xv, y + yv) & kFlagSolid) && !g_hdb->_map->explosionExist(x + xv, y + yv)) {
|
|
aiBarrelBlowup(e, x + xv, y + yv);
|
|
// are we blowing up on another BOOMBARREL? if so, start it exploding.
|
|
AIEntity *e2 = g_hdb->_ai->findEntity(x + xv, y + yv);
|
|
if (e2 && e2->state != STATE_EXPLODING) {
|
|
switch (e2->type) {
|
|
case AI_GUY:
|
|
g_hdb->_ai->killPlayer(DEATH_FRIED);
|
|
break;
|
|
case AI_BOOMBARREL:
|
|
aiBarrelExplode(e2, 0, 0);
|
|
break;
|
|
case AI_OMNIBOT:
|
|
case AI_TURNBOT:
|
|
case AI_SHOCKBOT:
|
|
case AI_RIGHTBOT:
|
|
case AI_PUSHBOT:
|
|
case AI_RAILRIDER:
|
|
case AI_MAINTBOT:
|
|
case AI_DEADEYE:
|
|
case AI_FATFROG:
|
|
case AI_ICEPUFF:
|
|
case AI_MEERKAT:
|
|
case AI_BUZZFLY:
|
|
case AI_GOODFAIRY:
|
|
case AI_GATEPUDDLE:
|
|
case AI_BADFAIRY:
|
|
g_hdb->_ai->addAnimateTarget(x * kTileWidth,
|
|
y * kTileHeight, 0, 3, ANIM_NORMAL, false, false, GROUP_EXPLOSION_BOOM_SIT);
|
|
if (e2->type != AI_LASERBEAM)
|
|
g_hdb->_ai->removeEntity(e2);
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void callbackAiBarrelExplosionEnd(int x, int y) {
|
|
g_hdb->_map->setExplosion(x, y, 0);
|
|
}
|
|
|
|
void aiBarrelBlowup(AIEntity *e, int x, int y) {
|
|
g_hdb->_ai->addAnimateTarget(x * kTileWidth,
|
|
y * kTileHeight, 0, 3, ANIM_NORMAL, false, false, GROUP_EXPLOSION_BOOM_SIT);
|
|
g_hdb->_map->setExplosion(x, y, 1);
|
|
g_hdb->_ai->addCallback(AI_BARREL_EXPLOSION_END, x, y, e->animCycle * 4);
|
|
}
|
|
|
|
void aiScientistInit(AIEntity *e, int x, int y) {
|
|
e->moveSpeed = kPlayerMoveSpeed >> 1;
|
|
if (g_hdb->_ai->findPath(e))
|
|
e->aiAction = aiGenericAction;
|
|
else if (e->value1)
|
|
e->aiAction = aiGenericAction;
|
|
}
|
|
|
|
void aiScientistInit2(AIEntity *e, int x, int y) {
|
|
e->draw = g_hdb->_ai->getStandFrameDir(e);
|
|
}
|
|
|
|
void aiSlugAttackAction(AIEntity *e, int x, int y) {
|
|
static const int xv[5] = {9, 0, 0, -1, 1};
|
|
static const int yv[5] = {9, -1, 1, 0, 0};
|
|
|
|
if (e->goalX)
|
|
g_hdb->_ai->animateEntity(e);
|
|
|
|
g_hdb->_ai->checkActionList(e, e->tileX, e->tileY, false);
|
|
g_hdb->_ai->checkAutoList(e, e->tileX, e->tileY);
|
|
|
|
AIEntity *hit = g_hdb->_ai->findEntityIgnore(e->tileX, e->tileY, e);
|
|
if (hit && hit->type == AI_GUY)
|
|
hit = nullptr;
|
|
|
|
// don't hit anything you can walk through...
|
|
if (hit && true == g_hdb->_ai->getTableEnt(hit->type))
|
|
hit = nullptr;
|
|
|
|
// don't hit floating stuff
|
|
if (hit && hit->state == STATE_FLOATING)
|
|
hit = nullptr;
|
|
|
|
uint32 bg_flags = g_hdb->_map->getMapBGTileFlags(e->tileX, e->tileY);
|
|
uint32 fg_flags = g_hdb->_map->getMapFGTileFlags(e->tileX, e->tileY);
|
|
int result = (e->level == 1 ? (bg_flags & (kFlagSolid)) : !(fg_flags & kFlagGrating) && (bg_flags & (kFlagSolid)));
|
|
if (hit) {
|
|
g_hdb->_sound->playSound(SND_SLUG_HIT);
|
|
g_hdb->_sound->playSound(g_hdb->_ai->metalOrFleshSND(hit));
|
|
switch (hit->type) {
|
|
case AI_MEERKAT:
|
|
if (hit->sequence > 2) { // out of the ground?
|
|
g_hdb->_ai->addAnimateTarget(hit->x, hit->y, 0, 3, ANIM_NORMAL, false, false, GROUP_STEAM_PUFF_SIT);
|
|
g_hdb->_ai->stunEnemy(hit, 8);
|
|
} else {
|
|
g_hdb->_ai->setEntityGoal(e, e->tileX + xv[e->dir], e->tileY + yv[e->dir]);
|
|
e->state = STATE_MOVEDOWN; // so it will draw & animate
|
|
g_hdb->_ai->animateEntity(e);
|
|
return;
|
|
}
|
|
break;
|
|
case AI_ICEPUFF:
|
|
if (hit->state == STATE_ICEP_APPEAR ||
|
|
hit->state == STATE_ICEP_THROWDOWN ||
|
|
hit->state == STATE_ICEP_THROWLEFT ||
|
|
hit->state == STATE_ICEP_THROWRIGHT) {
|
|
g_hdb->_ai->addAnimateTarget(hit->x, hit->y, 0, 3, ANIM_NORMAL, false, false, GROUP_STEAM_PUFF_SIT);
|
|
g_hdb->_ai->stunEnemy(hit, 8);
|
|
} else {
|
|
g_hdb->_ai->setEntityGoal(e, e->tileX + xv[e->dir], e->tileY + yv[e->dir]);
|
|
e->state = STATE_MOVEDOWN; // so it will draw & animate
|
|
g_hdb->_ai->animateEntity(e);
|
|
return;
|
|
}
|
|
break;
|
|
case AI_OMNIBOT:
|
|
case AI_TURNBOT:
|
|
case AI_SHOCKBOT:
|
|
case AI_RIGHTBOT:
|
|
case AI_PUSHBOT:
|
|
case AI_LISTENBOT:
|
|
case AI_MAINTBOT:
|
|
case AI_FATFROG:
|
|
case AI_BADFAIRY:
|
|
case AI_BUZZFLY:
|
|
g_hdb->_ai->addAnimateTarget(e->x, e->y, 0, 3, ANIM_NORMAL, false, false, GROUP_STEAM_PUFF_SIT);
|
|
g_hdb->_ai->stunEnemy(hit, 8);
|
|
break;
|
|
|
|
case AI_CHICKEN:
|
|
g_hdb->_ai->addAnimateTarget(hit->x, hit->y, 0, 3, ANIM_NORMAL, false, false, GROUP_STEAM_PUFF_SIT);
|
|
g_hdb->_ai->removeEntity(hit);
|
|
break;
|
|
case AI_BOOMBARREL:
|
|
g_hdb->_sound->playSound(SND_CLUB_HIT_METAL);
|
|
aiBarrelExplode(hit, 0, 0);
|
|
aiBarrelBlowup(hit, hit->tileX, hit->tileY);
|
|
break;
|
|
// ACTION MODE entities go away - except the FOURFIRER
|
|
case AI_GATEPUDDLE:
|
|
g_hdb->_ai->addAnimateTarget(e->x, e->y, 0, 7, ANIM_NORMAL, false, false, TELEPORT_FLASH);
|
|
g_hdb->_ai->removeEntity(hit);
|
|
g_hdb->_sound->playSound(SND_TELEPORT);
|
|
break;
|
|
case AI_DEADEYE:
|
|
g_hdb->_ai->addAnimateTarget(e->tileX * kTileWidth,
|
|
e->tileY * kTileHeight, 0, 3, ANIM_NORMAL, false, false, GROUP_EXPLOSION_BOOM_SIT);
|
|
g_hdb->_ai->removeEntity(hit);
|
|
g_hdb->_sound->playSound(SND_BARREL_EXPLODE);
|
|
break;
|
|
|
|
case AI_NONE:
|
|
if (hit->value1 == (int)AI_DRAGON) {
|
|
// pull dragon's coords out of "lua_func_use" string.
|
|
char num1[4], num2[4];
|
|
memset(num1, 0, 4);
|
|
memset(num2, 0, 4);
|
|
memcpy(num1, hit->luaFuncUse, 3);
|
|
memcpy(num2, hit->luaFuncUse + 3, 3);
|
|
|
|
g_hdb->_sound->playSound(SND_CLUB_HIT_FLESH);
|
|
AIEntity *found = g_hdb->_ai->findEntity(atoi(num1), atoi(num2));
|
|
if (found)
|
|
aiDragonWake(found, 0, 0);
|
|
g_hdb->_ai->addAnimateTarget(e->x, e->y, 0, 3, ANIM_NORMAL, false, false, GEM_FLASH);
|
|
}
|
|
break;
|
|
case AI_DRAGON:
|
|
aiDragonWake(hit, 0, 0);
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
g_hdb->_ai->removeEntity(e); // bye bye!
|
|
return;
|
|
} else if (result) { // hit a wall
|
|
g_hdb->_sound->playSound(SND_SLUG_HIT);
|
|
g_hdb->_ai->addAnimateTarget(e->x, e->y, 0, 3, ANIM_NORMAL, false, false, GROUP_STEAM_PUFF_SIT);
|
|
g_hdb->_ai->removeEntity(e);
|
|
} else {
|
|
g_hdb->_ai->setEntityGoal(e, e->tileX + xv[e->dir], e->tileY + yv[e->dir]);
|
|
e->state = STATE_MOVEDOWN; // so it will draw & animate
|
|
g_hdb->_ai->animateEntity(e);
|
|
}
|
|
}
|
|
|
|
void aiSlugAttackDraw(AIEntity *e, int mx, int my) {
|
|
g_hdb->_ai->_slugAttackGfx[e->animFrame]->drawMasked(e->x - mx + 8, e->y - my + 8);
|
|
}
|
|
|
|
void aiSlugAttackInit(AIEntity *e, int mx, int my) {
|
|
static const int xv[5] = {9, 0, 0, -1, 1};
|
|
static const int yv[5] = {9, -1, 1, 0, 0};
|
|
|
|
if (g_hdb->isDemo())
|
|
return;
|
|
|
|
e->moveSpeed = kPlayerMoveSpeed << 1;
|
|
g_hdb->_ai->setEntityGoal(e, e->tileX + xv[e->dir], e->tileY + yv[e->dir]);
|
|
e->draw = nullptr; // use custom draw function
|
|
e->aiDraw = aiSlugAttackDraw;
|
|
e->state = STATE_MOVEDOWN; // so it will draw & animate
|
|
e->aiAction = aiSlugAttackAction;
|
|
g_hdb->_sound->playSound(SND_SLUG_FIRE);
|
|
}
|
|
|
|
void aiSlugAttackInit2(AIEntity *e, int mx, int my) {
|
|
e->movedownFrames = 4;
|
|
}
|
|
|
|
void aiDeadWorkerInit(AIEntity *e, int mx, int my) {
|
|
}
|
|
|
|
void aiDeadWorkerInit2(AIEntity *e, int mx, int my) {
|
|
e->draw = e->standdownGfx[0];
|
|
}
|
|
|
|
void aiWorkerInit(AIEntity *e, int mx, int my) {
|
|
if (e->value1)
|
|
e->aiAction = aiGenericAction;
|
|
e->moveSpeed = kPlayerMoveSpeed >> 1;
|
|
}
|
|
|
|
void aiWorkerInit2(AIEntity *e, int mx, int my) {
|
|
e->draw = g_hdb->_ai->getStandFrameDir(e);
|
|
}
|
|
|
|
void aiAccountantInit(AIEntity *e, int mx, int my) {
|
|
}
|
|
|
|
void aiAccountantInit2(AIEntity *e, int mx, int my) {
|
|
e->draw = g_hdb->_ai->getStandFrameDir(e);
|
|
}
|
|
|
|
void aiFrogStatueInit(AIEntity *e, int mx, int my) {
|
|
e->moveSpeed = kPushMoveSpeed;
|
|
e->aiAction = aiFrogStatueAction;
|
|
}
|
|
|
|
void aiFrogStatueInit2(AIEntity *e, int mx, int my) {
|
|
// point all frog statue MOVE frames to the standing one
|
|
e->blinkFrames =
|
|
e->movedownFrames =
|
|
e->moveleftFrames =
|
|
e->moverightFrames =
|
|
e->moveupFrames = 1;
|
|
|
|
e->blinkGfx[0] =
|
|
e->movedownGfx[0] =
|
|
e->moveupGfx[0] =
|
|
e->moveleftGfx[0] =
|
|
e->moverightGfx[0] = e->standdownGfx[0];
|
|
|
|
e->draw = e->standdownGfx[0]; // standing frame - doesn't move
|
|
}
|
|
|
|
void aiFrogStatueAction(AIEntity *e, int mx, int my) {
|
|
// if frog statue isn't moving somewhere, don't move it
|
|
if (!e->goalX)
|
|
return;
|
|
|
|
g_hdb->_ai->animateEntity(e);
|
|
}
|
|
|
|
void aiRoboStunnerAction(AIEntity *e, int mx, int my) {
|
|
aiAnimateStanddown(e, 1);
|
|
aiGetItemAction(e, 0, 0);
|
|
}
|
|
|
|
void aiRoboStunnerInit(AIEntity *e, int mx, int my) {
|
|
e->aiAction = aiRoboStunnerAction;
|
|
Common::strlcpy(e->printedName, "Robostunner", 32);
|
|
}
|
|
|
|
void aiRoboStunnerInit2(AIEntity *e, int mx, int my) {
|
|
e->draw = e->standdownGfx[0];
|
|
}
|
|
|
|
void aiClubInit(AIEntity *e, int mx, int my) {
|
|
Common::strlcpy(e->printedName, "Creature Clubber", 32);
|
|
e->aiAction = aiGetItemAction;
|
|
}
|
|
|
|
void aiClubInit2(AIEntity *e, int mx, int my) {
|
|
e->draw = e->standdownGfx[0];
|
|
}
|
|
|
|
void aiSlugSlingerInit(AIEntity *e, int mx, int my) {
|
|
Common::strlcpy(e->printedName, "Slugslinger", 32);
|
|
e->aiAction = aiGetItemAction;
|
|
}
|
|
|
|
void aiSlugSlingerInit2(AIEntity *e, int mx, int my) {
|
|
e->draw = e->standdownGfx[0];
|
|
}
|
|
|
|
void aiEnvelopeGreenInit(AIEntity *e, int mx, int my) {
|
|
Common::strlcpy(e->printedName, "Green envelope", 32);
|
|
e->aiAction = aiGetItemAction;
|
|
}
|
|
|
|
void aiEnvelopeGreenInit2(AIEntity *e, int mx, int my) {
|
|
e->draw = e->standdownGfx[0];
|
|
}
|
|
|
|
void aiGemBlueInit(AIEntity *e, int mx, int my) {
|
|
e->aiAction = aiGemAction;
|
|
}
|
|
|
|
void aiGemBlueInit2(AIEntity *e, int mx, int my) {
|
|
e->draw = e->standdownGfx[0];
|
|
}
|
|
|
|
void aiGemRedInit(AIEntity *e, int mx, int my) {
|
|
e->aiAction = aiGemAction;
|
|
}
|
|
|
|
void aiGemRedInit2(AIEntity *e, int mx, int my) {
|
|
e->draw = e->standdownGfx[0];
|
|
}
|
|
|
|
void aiGemGreenInit(AIEntity *e, int mx, int my) {
|
|
e->aiAction = aiGemAction;
|
|
}
|
|
|
|
void aiGemGreenInit2(AIEntity *e, int mx, int my) {
|
|
e->draw = e->standdownGfx[0];
|
|
}
|
|
|
|
void aiTeaCupInit(AIEntity *e, int mx, int my) {
|
|
Common::strlcpy(e->printedName, "a Teacup", 32);
|
|
e->aiAction = aiGetItemAction;
|
|
}
|
|
|
|
void aiTeaCupInit2(AIEntity *e, int mx, int my) {
|
|
e->draw = e->standdownGfx[0];
|
|
}
|
|
|
|
void aiCookieInit(AIEntity *e, int mx, int my) {
|
|
Common::strlcpy(e->printedName, "a Cookie", 32);
|
|
e->aiAction = aiGetItemAction;
|
|
}
|
|
|
|
void aiCookieInit2(AIEntity *e, int mx, int my) {
|
|
e->draw = e->standdownGfx[0];
|
|
}
|
|
|
|
void aiBurgerInit(AIEntity *e, int mx, int my) {
|
|
Common::strlcpy(e->printedName, "a Burger", 32);
|
|
e->aiAction = aiGetItemAction;
|
|
}
|
|
|
|
void aiBurgerInit2(AIEntity *e, int mx, int my) {
|
|
e->draw = e->standdownGfx[0];
|
|
}
|
|
|
|
void aiBookInit(AIEntity *e, int mx, int my) {
|
|
Common::strlcpy(e->printedName, "a Book", 32);
|
|
e->aiAction = aiGetItemAction;
|
|
}
|
|
|
|
void aiBookInit2(AIEntity *e, int mx, int my) {
|
|
e->draw = e->standdownGfx[0];
|
|
}
|
|
|
|
void aiClipboardInit(AIEntity *e, int mx, int my) {
|
|
Common::strlcpy(e->printedName, "a Clipboard", 32);
|
|
e->aiAction = aiGetItemAction;
|
|
}
|
|
|
|
void aiClipboardInit2(AIEntity *e, int mx, int my) {
|
|
e->draw = e->standdownGfx[0];
|
|
}
|
|
|
|
void aiNoteInit(AIEntity *e, int mx, int my) {
|
|
Common::strlcpy(e->printedName, "a Note", 32);
|
|
e->aiAction = aiGetItemAction;
|
|
}
|
|
|
|
void aiNoteInit2(AIEntity *e, int mx, int my) {
|
|
e->draw = e->standdownGfx[0];
|
|
}
|
|
|
|
void aiKeycardWhiteInit(AIEntity *e, int mx, int my) {
|
|
Common::strlcpy(e->printedName, "a White keycard", 32);
|
|
e->aiAction = aiGetItemAction;
|
|
}
|
|
|
|
void aiKeycardWhiteInit2(AIEntity *e, int mx, int my) {
|
|
e->draw = e->standdownGfx[0];
|
|
}
|
|
|
|
void aiKeycardBlueInit(AIEntity *e, int mx, int my) {
|
|
Common::strlcpy(e->printedName, "a Blue keycard", 32);
|
|
e->aiAction = aiGetItemAction;
|
|
}
|
|
|
|
void aiKeycardBlueInit2(AIEntity *e, int mx, int my) {
|
|
e->draw = e->standdownGfx[0];
|
|
}
|
|
|
|
void aiKeycardRedInit(AIEntity *e, int mx, int my) {
|
|
Common::strlcpy(e->printedName, "a Red keycard", 32);
|
|
e->aiAction = aiGetItemAction;
|
|
}
|
|
|
|
void aiKeycardRedInit2(AIEntity *e, int mx, int my) {
|
|
e->draw = e->standdownGfx[0];
|
|
}
|
|
|
|
void aiKeycardGreenInit(AIEntity *e, int mx, int my) {
|
|
Common::strlcpy(e->printedName, "a Green keycard", 32);
|
|
e->aiAction = aiGetItemAction;
|
|
}
|
|
|
|
void aiKeycardGreenInit2(AIEntity *e, int mx, int my) {
|
|
e->draw = e->standdownGfx[0];
|
|
}
|
|
|
|
void aiKeycardPurpleInit(AIEntity *e, int mx, int my) {
|
|
Common::strlcpy(e->printedName, "a Purple keycard", 32);
|
|
e->aiAction = aiGetItemAction;
|
|
}
|
|
|
|
void aiKeycardPurpleInit2(AIEntity *e, int mx, int my) {
|
|
e->draw = e->standdownGfx[0];
|
|
}
|
|
|
|
void aiKeycardBlackInit(AIEntity *e, int mx, int my) {
|
|
Common::strlcpy(e->printedName, "a Black keycard", 32);
|
|
e->aiAction = aiGetItemAction;
|
|
}
|
|
|
|
void aiKeycardBlackInit2(AIEntity *e, int mx, int my) {
|
|
e->draw = e->standdownGfx[0];
|
|
}
|
|
|
|
void aiSeedInit(AIEntity *e, int mx, int my) {
|
|
Common::strlcpy(e->printedName, "some Henscratch", 32);
|
|
e->aiAction = aiGetItemAction;
|
|
}
|
|
|
|
void aiSeedInit2(AIEntity *e, int mx, int my) {
|
|
e->draw = e->standdownGfx[0];
|
|
}
|
|
|
|
void aiSodaInit(AIEntity *e, int mx, int my) {
|
|
Common::strlcpy(e->printedName, "a Dr. Frostee", 32);
|
|
e->aiAction = aiGetItemAction;
|
|
}
|
|
|
|
void aiSodaInit2(AIEntity *e, int mx, int my) {
|
|
e->draw = e->standdownGfx[0];
|
|
}
|
|
|
|
void aiDollyTool1Init(AIEntity *e, int mx, int my) {
|
|
Common::strlcpy(e->printedName, "Dolly's Wrench", 32);
|
|
e->aiAction = aiGetItemAction;
|
|
}
|
|
|
|
void aiDollyTool1Init2(AIEntity *e, int mx, int my) {
|
|
e->draw = e->standdownGfx[0];
|
|
}
|
|
|
|
void aiDollyTool2Init(AIEntity *e, int mx, int my) {
|
|
Common::strlcpy(e->printedName, "Dolly's Torch", 32);
|
|
e->aiAction = aiGetItemAction;
|
|
}
|
|
|
|
void aiDollyTool2Init2(AIEntity *e, int mx, int my) {
|
|
e->draw = e->standdownGfx[0];
|
|
}
|
|
|
|
void aiDollyTool3Init(AIEntity *e, int mx, int my) {
|
|
Common::strlcpy(e->printedName, "Dolly's EMF Resonator", 32);
|
|
e->aiAction = aiGetItemAction;
|
|
}
|
|
|
|
void aiDollyTool3Init2(AIEntity *e, int mx, int my) {
|
|
e->draw = e->standdownGfx[0];
|
|
}
|
|
|
|
void aiDollyTool4Init(AIEntity *e, int mx, int my) {
|
|
Common::strlcpy(e->printedName, "Dolly's Toolbox", 32);
|
|
e->aiAction = aiGetItemAction;
|
|
}
|
|
|
|
void aiDollyTool4Init2(AIEntity *e, int mx, int my) {
|
|
e->draw = e->standdownGfx[0];
|
|
}
|
|
|
|
void aiRouterInit(AIEntity *e, int mx, int my) {
|
|
Common::strlcpy(e->printedName, "a Computer Router", 32);
|
|
e->aiAction = aiGetItemAction;
|
|
}
|
|
|
|
void aiRouterInit2(AIEntity *e, int mx, int my) {
|
|
e->draw = e->standdownGfx[0];
|
|
}
|
|
|
|
void aiSlicerInit(AIEntity *e, int mx, int my) {
|
|
Common::strlcpy(e->printedName, "a Pizza Slicer", 32);
|
|
e->aiAction = aiGetItemAction;
|
|
}
|
|
|
|
void aiSlicerInit2(AIEntity *e, int mx, int my) {
|
|
e->draw = e->standdownGfx[0];
|
|
}
|
|
|
|
void aiPackageInit(AIEntity *e, int mx, int my) {
|
|
Common::strlcpy(e->printedName, "a Package", 32);
|
|
e->aiAction = aiGetItemAction;
|
|
}
|
|
|
|
void aiPackageInit2(AIEntity *e, int mx, int my) {
|
|
e->draw = e->standdownGfx[0];
|
|
}
|
|
|
|
void aiMagicEggAction(AIEntity *e, int mx, int my) {
|
|
// if magic egg isn't moving somewhere, don't move it
|
|
if (!e->goalX)
|
|
return;
|
|
|
|
g_hdb->_ai->animateEntity(e);
|
|
}
|
|
|
|
void aiMagicEggInit(AIEntity *e, int mx, int my) {
|
|
e->moveSpeed = kPushMoveSpeed;
|
|
e->aiAction = aiMagicEggAction;
|
|
}
|
|
|
|
void aiMagicEggInit2(AIEntity *e, int mx, int my) {
|
|
// point all magic egg move frames to the standing one
|
|
e->movedownFrames =
|
|
e->moveleftFrames =
|
|
e->moverightFrames =
|
|
e->moveupFrames = 1;
|
|
|
|
e->movedownGfx[0] =
|
|
e->moveupGfx[0] =
|
|
e->moveleftGfx[0] =
|
|
e->moverightGfx[0] = e->standdownGfx[0];
|
|
|
|
e->draw = e->standdownGfx[0]; // standing frame - doesn't move
|
|
}
|
|
|
|
void aiMagicEggUse(AIEntity *e, int mx, int my) {
|
|
if (!scumm_strnicmp(e->luaFuncAction, "ai_", 3) || !scumm_strnicmp(e->luaFuncAction, "item_", 5)) {
|
|
AIEntity *spawned = nullptr;
|
|
for (int i = 0; aiEntList[i].type != END_AI_TYPES; ++i) {
|
|
if (!scumm_stricmp(aiEntList[i].luaName, e->luaFuncAction)) {
|
|
spawned = g_hdb->_ai->spawn(aiEntList[i].type, e->dir, e->tileX, e->tileY, nullptr, nullptr, nullptr, DIR_NONE, e->level, 0, 0, 1);
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (spawned) {
|
|
g_hdb->_ai->addAnimateTarget(e->tileX * kTileWidth,
|
|
e->tileY * kTileHeight, 0, 3, ANIM_NORMAL, false, false, GROUP_EXPLOSION_BOOM_SIT);
|
|
|
|
if (!g_hdb->isDemo())
|
|
g_hdb->_sound->playSound(SND_BARREL_EXPLODE);
|
|
|
|
g_hdb->_ai->removeEntity(e);
|
|
}
|
|
}
|
|
}
|
|
|
|
void aiIceBlockAction(AIEntity *e, int mx, int my) {
|
|
// if ice block isn't moving somewhere, don't move it
|
|
if (!e->goalX)
|
|
return;
|
|
|
|
g_hdb->_ai->animateEntity(e);
|
|
}
|
|
|
|
void aiIceBlockInit(AIEntity *e, int mx, int my) {
|
|
e->moveSpeed = kPushMoveSpeed;
|
|
e->aiAction = aiIceBlockAction;
|
|
}
|
|
|
|
void aiIceBlockInit2(AIEntity *e, int mx, int my) {
|
|
// point all ice block move frames to the standing one
|
|
e->movedownFrames =
|
|
e->moveleftFrames =
|
|
e->moverightFrames =
|
|
e->moveupFrames = 1;
|
|
|
|
e->movedownGfx[0] =
|
|
e->moveupGfx[0] =
|
|
e->moveleftGfx[0] =
|
|
e->moverightGfx[0] = e->standdownGfx[0];
|
|
|
|
e->draw = e->standdownGfx[0]; // standing frame - doesn't move
|
|
}
|
|
|
|
void aiCabKeyInit(AIEntity *e, int mx, int my) {
|
|
Common::strlcpy(e->printedName, "a Cabinet key", 32);
|
|
}
|
|
|
|
void aiCabKeyInit2(AIEntity *e, int mx, int my) {
|
|
e->draw = e->standdownGfx[0];
|
|
}
|
|
|
|
void aiItemChickenInit(AIEntity *e, int mx, int my) {
|
|
Common::strlcpy(e->printedName, "Cooper's chicken", 32);
|
|
}
|
|
|
|
void aiItemChickenInit2(AIEntity *e, int mx, int my) {
|
|
e->draw = e->standdownGfx[0];
|
|
}
|
|
|
|
void aiPdaInit(AIEntity *e, int mx, int my) {
|
|
Common::strlcpy(e->printedName, "a P.D.A.", 32);
|
|
e->aiAction = aiGetItemAction;
|
|
}
|
|
|
|
void aiPdaInit2(AIEntity *e, int mx, int my) {
|
|
e->draw = e->standdownGfx[0];
|
|
}
|
|
|
|
#if 0
|
|
void aiCellUse(AIEntity *e, int mx, int my) {
|
|
g_hdb->_window->openMessageBar("You got the Energy Cell!", kMsgDelay);
|
|
}
|
|
#endif
|
|
|
|
void aiCellInit2(AIEntity *e, int mx, int my) {
|
|
e->draw = e->standdownGfx[0];
|
|
}
|
|
|
|
void aiCellInit(AIEntity *e, int mx, int my) {
|
|
Common::strlcpy(e->printedName, "Energy Cell", 32);
|
|
e->aiAction = aiGetItemAction;
|
|
}
|
|
|
|
void aiEnvelopeWhiteInit(AIEntity *e, int mx, int my) {
|
|
Common::strlcpy(e->printedName, "White envelope", 32);
|
|
e->aiAction = aiGetItemAction;
|
|
}
|
|
|
|
void aiEnvelopeWhiteInit2(AIEntity *e, int mx, int my) {
|
|
e->draw = e->standdownGfx[0];
|
|
}
|
|
|
|
void aiEnvelopeBlueInit(AIEntity *e, int mx, int my) {
|
|
Common::strlcpy(e->printedName, "Blue envelope", 32);
|
|
e->aiAction = aiGetItemAction;
|
|
}
|
|
|
|
void aiEnvelopeBlueInit2(AIEntity *e, int mx, int my) {
|
|
e->draw = e->standdownGfx[0];
|
|
}
|
|
|
|
void aiEnvelopeRedInit(AIEntity *e, int mx, int my) {
|
|
Common::strlcpy(e->printedName, "Red envelope", 32);
|
|
e->aiAction = aiGetItemAction;
|
|
}
|
|
|
|
void aiEnvelopeRedInit2(AIEntity *e, int mx, int my) {
|
|
e->draw = e->standdownGfx[0];
|
|
}
|
|
|
|
void aiTransceiverInit(AIEntity *e, int mx, int my) {
|
|
e->aiAction = aiTransceiverAction;
|
|
Common::strlcpy(e->printedName, "Transceiver", 32);
|
|
}
|
|
|
|
void aiTransceiverInit2(AIEntity *e, int mx, int my) {
|
|
e->draw = e->standdownGfx[0];
|
|
}
|
|
|
|
void aiTransceiverAction(AIEntity *e, int mx, int my) {
|
|
aiAnimateStanddown(e, 5);
|
|
if (e->onScreen)
|
|
aiGetItemAction(e, 0, 0);
|
|
}
|
|
|
|
#if 0
|
|
void aiTransceiverUse(AIEntity *e, int mx, int my) {
|
|
g_hdb->_window->openMessageBar("You got the Transceiver!", kMsgDelay);
|
|
}
|
|
#endif
|
|
|
|
void aiMonkeystoneInit(AIEntity *e, int mx, int my) {
|
|
e->aiUse = aiMonkeystoneUse;
|
|
e->aiAction = aiMonkeystoneAction;
|
|
}
|
|
|
|
void aiMonkeystoneAction(AIEntity *e, int mx, int my) {
|
|
if (!e->onScreen)
|
|
return;
|
|
|
|
AIEntity *p = g_hdb->_ai->getPlayer();
|
|
if (abs(p->x - e->x) < 16 && abs(p->y - e->y) < 16 && e->level == p->level) {
|
|
if (e->luaFuncUse[0])
|
|
g_hdb->_lua->callFunction(e->luaFuncUse, 0);
|
|
|
|
g_hdb->_ai->addToInventory(e);
|
|
aiMonkeystoneUse(nullptr, 0, 0);
|
|
}
|
|
}
|
|
|
|
void aiMonkeystoneInit2(AIEntity *e, int mx, int my) {
|
|
e->draw = e->standdownGfx[0];
|
|
}
|
|
|
|
void aiMonkeystoneUse(AIEntity *e, int mx, int my) {
|
|
int val = g_hdb->_ai->getMonkeystoneAmount();
|
|
Common::String monkString = Common::String::format("You have %d Monkeystone%s!", val, (val > 1) ? "s" : "");
|
|
g_hdb->_sound->playSound(SND_GET_MONKEYSTONE);
|
|
g_hdb->_window->openMessageBar(monkString.c_str(), kMsgDelay);
|
|
|
|
// have we unlocked a secret star(tm)???
|
|
if (val == 7) {
|
|
g_hdb->_window->openMessageBar("Red Star is Ready!", kMsgDelay * 2);
|
|
g_hdb->setStarsMonkeystone7(STARS_MONKEYSTONE_7);
|
|
g_hdb->_menu->writeConfig();
|
|
g_hdb->_gfx->turnOnBonusStars(0);
|
|
}
|
|
if (val == 14) {
|
|
g_hdb->_window->openMessageBar("Green Star is GO!", kMsgDelay * 2);
|
|
g_hdb->setStarsMonkeystone14(STARS_MONKEYSTONE_14);
|
|
g_hdb->_menu->writeConfig();
|
|
g_hdb->_gfx->turnOnBonusStars(1);
|
|
}
|
|
if (val == 21) {
|
|
g_hdb->_window->openMessageBar("Blue Star is Born!", kMsgDelay * 2);
|
|
g_hdb->setStarsMonkeystone21(STARS_MONKEYSTONE_21);
|
|
g_hdb->_menu->writeConfig();
|
|
g_hdb->_gfx->turnOnBonusStars(2);
|
|
}
|
|
}
|
|
|
|
void aiGemAction(AIEntity *e, int mx, int my) {
|
|
e->animFrame++;
|
|
if (e->animFrame >= e->standdownFrames) {
|
|
e->animFrame = 0;
|
|
|
|
// every 4th frame, check for player collision &
|
|
// add to inventory if it happens
|
|
AIEntity *p = g_hdb->_ai->getPlayer();
|
|
int tolerance = 16;
|
|
if (g_hdb->_ai->playerRunning())
|
|
tolerance = 24;
|
|
|
|
if (e->onScreen && abs(p->x - e->x) < tolerance && abs(p->y - e->y) < tolerance && e->level == p->level) {
|
|
g_hdb->_ai->addAnimateTarget(e->x, e->y, 0, 3, ANIM_NORMAL, false, false, GEM_FLASH);
|
|
g_hdb->_ai->addToInventory(e);
|
|
g_hdb->_sound->playSound(SND_GET_GEM);
|
|
return;
|
|
}
|
|
}
|
|
e->draw = e->standdownGfx[e->animFrame];
|
|
}
|
|
|
|
void aiGemWhiteInit(AIEntity *e, int mx, int my) {
|
|
e->aiAction = aiGemAction;
|
|
}
|
|
|
|
void aiGemWhiteInit2(AIEntity *e, int mx, int my) {
|
|
e->draw = e->standdownGfx[0];
|
|
}
|
|
|
|
void aiGooCupUse(AIEntity *e, int mx, int my) {
|
|
g_hdb->_window->openMessageBar("Got a... cup of goo.", kMsgDelay);
|
|
}
|
|
|
|
void aiGooCupInit(AIEntity *e, int mx, int my) {
|
|
e->aiUse = aiGooCupUse;
|
|
e->aiAction = aiGetItemAction;
|
|
}
|
|
|
|
void aiGooCupInit2(AIEntity *e, int mx, int my) {
|
|
e->draw = e->standdownGfx[0];
|
|
}
|
|
|
|
void aiVortexianAction(AIEntity *e, int mx, int my) {
|
|
// anim the alpha blending : down to 32, up to 180, back down...
|
|
e->value2 += e->value1;
|
|
if ((e->value2 & 0xff) > 128) {
|
|
e->value2 = (e->value2 & 0xff00) | 128;
|
|
e->value1 = -e->value1;
|
|
} else if ((e->value2 & 0xff) < 32) {
|
|
e->value2 = (e->value2 & 0xff00) | 32;
|
|
e->value1 = -e->value1;
|
|
}
|
|
|
|
// anim the shape
|
|
e->animFrame++;
|
|
if (e->animFrame >= e->standdownFrames) {
|
|
e->animFrame = 0;
|
|
|
|
// every 4th frame, check for player collision &
|
|
// do an autosave
|
|
AIEntity *p = g_hdb->_ai->getPlayer();
|
|
if (abs(p->x - e->x) < 4 && abs(p->y - e->y) < 4) {
|
|
if (!(e->value2 & 0xff00)) {
|
|
// let's make sure we don't autosave every frikken second!
|
|
e->value2 |= 0xff00;
|
|
|
|
g_hdb->saveWhenReady(kAutoSaveSlot);
|
|
g_hdb->_window->openMessageBar("Saving progress at Vortexian...", 1);
|
|
}
|
|
} else
|
|
e->value2 &= 0x00ff;
|
|
}
|
|
e->draw = e->standdownGfx[e->animFrame];
|
|
}
|
|
|
|
void aiVortexianUse(AIEntity *e, int mx, int my) {
|
|
}
|
|
|
|
void aiVortexianInit(AIEntity *e, int mx, int my) {
|
|
e->aiUse = aiVortexianUse;
|
|
e->aiAction = aiVortexianAction;
|
|
e->value1 = 5;
|
|
e->value2 = 128;
|
|
}
|
|
|
|
void aiVortexianInit2(AIEntity *e, int mx, int my) {
|
|
e->draw = e->standdownGfx[0];
|
|
}
|
|
|
|
void aiNoneInit(AIEntity *e, int mx, int my) {
|
|
}
|
|
|
|
// Utility Functions
|
|
void aiAnimateStanddown(AIEntity *e, int speed) {
|
|
if (e->value2-- > 0)
|
|
return;
|
|
e->value2 = speed;
|
|
|
|
if (e->type == AI_GUY && e->animFrame > 0)
|
|
e->value2 = 0;
|
|
e->draw = e->standdownGfx[e->animFrame];
|
|
e->animFrame++;
|
|
if (e->animFrame >= e->standdownFrames)
|
|
e->animFrame = 0;
|
|
}
|
|
|
|
void aiGenericAction(AIEntity *e, int mx, int my) {
|
|
if (!e->goalX)
|
|
g_hdb->_ai->findPath(e);
|
|
else if (onEvenTile(e->x, e->y))
|
|
g_hdb->_sound->playSound(SND_FOOTSTEPS);
|
|
g_hdb->_ai->animateEntity(e);
|
|
}
|
|
|
|
void aiGetItemAction(AIEntity *e, int mx, int my) {
|
|
if (!e->onScreen)
|
|
return;
|
|
|
|
AIEntity *p = g_hdb->_ai->getPlayer();
|
|
if (abs(p->x - e->x) < 16 && abs(p->y - e->y) < 16 && e->level == p->level) {
|
|
if (e->aiUse)
|
|
e->aiUse(e, 0, 0);
|
|
if (e->luaFuncUse[0])
|
|
g_hdb->_lua->callFunction(e->luaFuncUse, 0);
|
|
|
|
g_hdb->_ai->getItemSound(e->type);
|
|
g_hdb->_ai->addToInventory(e);
|
|
}
|
|
}
|
|
|
|
} // End of Namespace
|