1824 lines
35 KiB
C++
1824 lines
35 KiB
C++
/*=========================================================================
|
|
|
|
npc.cpp
|
|
|
|
Author: CRB
|
|
Created:
|
|
Project: Spongebob
|
|
Purpose:
|
|
|
|
Copyright (c) 2000 Climax Development Ltd
|
|
|
|
===========================================================================*/
|
|
|
|
#include "enemy\npc.h"
|
|
|
|
#ifndef __PAD_VIBE_H__
|
|
#include "pad\vibe.h"
|
|
#endif
|
|
|
|
#ifndef __LEVEL_LEVEL_H__
|
|
#include "level\level.h"
|
|
#endif
|
|
|
|
#ifndef __FILE_EQUATES_H__
|
|
#include <biglump.h>
|
|
#endif
|
|
|
|
#ifndef __GAME_GAME_H__
|
|
#include "game\game.h"
|
|
#endif
|
|
|
|
#ifndef __PLAYER_PLAYER_H__
|
|
#include "player\player.h"
|
|
#endif
|
|
|
|
#ifndef __ENEMY_NPCPATH_H__
|
|
#include "enemy\npcpath.h"
|
|
#endif
|
|
|
|
#ifndef __UTILS_HEADER__
|
|
#include "utils\utils.h"
|
|
#endif
|
|
|
|
#ifndef __GAME_CONVO_H__
|
|
#include "game\convo.h"
|
|
#endif
|
|
|
|
#include "Gfx\actor.h"
|
|
|
|
#ifndef __VID_HEADER_
|
|
#include "system\vid.h"
|
|
#endif
|
|
|
|
#ifndef __PROJECTL_PROJECTL_H__
|
|
#include "projectl\projectl.h"
|
|
#endif
|
|
|
|
#ifndef __PROJECTL_PRNPC_H__
|
|
#include "projectl\prnpc.h"
|
|
#endif
|
|
|
|
#include "fx\fx.h"
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
#ifndef __ENEMY_NSJFISH_H__
|
|
#include "enemy\nsjfish.h"
|
|
#endif
|
|
|
|
#ifndef __ENEMY_NHCRAB_H__
|
|
#include "enemy\nhcrab.h"
|
|
#endif
|
|
|
|
#ifndef __ENEMY_NSCRAB_H__
|
|
#include "enemy\nscrab.h"
|
|
#endif
|
|
|
|
#ifndef __ENEMY_NGEN_H__
|
|
#include "enemy\ngen.h"
|
|
#endif
|
|
|
|
#ifndef __ENEMY_NANEMONE_H__
|
|
#include "enemy\nanemone.h"
|
|
#endif
|
|
|
|
#ifndef __ENEMY_NCLAM_H__
|
|
#include "enemy\nclam.h"
|
|
#endif
|
|
|
|
#ifndef __ENEMY_NOCTO_H__
|
|
#include "enemy\nocto.h"
|
|
#endif
|
|
|
|
#ifndef __ENEMY_NFFOLK_H__
|
|
#include "enemy\nffolk.h"
|
|
#endif
|
|
|
|
#ifndef __ENEMY_NBBLOB_H__
|
|
#include "enemy\nbblob.h"
|
|
#endif
|
|
|
|
#ifndef __ENEMY_NPUFFA_H__
|
|
#include "enemy\npuffa.h"
|
|
#endif
|
|
|
|
#ifndef __ENEMY_NSHRKMAN_H__
|
|
#include "enemy\nshrkman.h"
|
|
#endif
|
|
|
|
#ifndef __ENEMY_NSKLFISH_H__
|
|
#include "enemy\nsklfish.h"
|
|
#endif
|
|
|
|
#ifndef __ENEMY_NEYEBALL_H__
|
|
#include "enemy\neyeball.h"
|
|
#endif
|
|
|
|
#ifndef __ENEMY_NFSKULL_H__
|
|
#include "enemy\nfskull.h"
|
|
#endif
|
|
|
|
#ifndef __ENEMY_NSSTOMP_H__
|
|
#include "enemy\nsstomp.h"
|
|
#endif
|
|
|
|
#ifndef __ENEMY_NBOOGER_H__
|
|
#include "enemy\nbooger.h"
|
|
#endif
|
|
|
|
#ifndef __ENEMY_NMJFISH_H__
|
|
#include "enemy\nmjfish.h"
|
|
#endif
|
|
|
|
#ifndef __ENEMY_NSSHARK_H__
|
|
#include "enemy\nsshark.h"
|
|
#endif
|
|
|
|
#ifndef __ENEMY_NWORM_H__
|
|
#include "enemy\nworm.h"
|
|
#endif
|
|
|
|
#ifndef __ENEMY_NFDUTCH_H__
|
|
#include "enemy\nfdutch.h"
|
|
#endif
|
|
|
|
#ifndef __ENEMY_NDOGFISH_H__
|
|
#include "enemy\ndogfish.h"
|
|
#endif
|
|
|
|
#ifndef __ENEMY_NDUSTDEV_H__
|
|
#include "enemy\ndustdev.h"
|
|
#endif
|
|
|
|
#ifndef __ENEMY_NSDART_H__
|
|
#include "enemy\nsdart.h"
|
|
#endif
|
|
|
|
#ifndef __ENEMY_NPBUG_H__
|
|
#include "enemy\npbug.h"
|
|
#endif
|
|
|
|
#ifndef __ENEMY_NSSNAKE_H__
|
|
#include "enemy\nssnake.h"
|
|
#endif
|
|
|
|
#ifndef __ENEMY_NANGLER_H__
|
|
#include "enemy\nangler.h"
|
|
#endif
|
|
|
|
#ifndef __ENEMY_NGHOST_H__
|
|
#include "enemy\nghost.h"
|
|
#endif
|
|
|
|
#ifndef __ENEMY_NSHELL_H__
|
|
#include "enemy\nshell.h"
|
|
#endif
|
|
|
|
#ifndef __ENEMY_NMJBACK_H__
|
|
#include "enemy\nmjback.h"
|
|
#endif
|
|
|
|
#ifndef __ENEMY_NSJBACK_H__
|
|
#include "enemy\nsjback.h"
|
|
#endif
|
|
|
|
#ifndef __ENEMY_NBUTTFLY_H__
|
|
#include "enemy\nbuttfly.h"
|
|
#endif
|
|
|
|
#ifndef __ENEMY_NPROJJF_H__
|
|
#include "enemy\nprojjf.h"
|
|
#endif
|
|
|
|
#ifndef __ENEMY_NBOSS_H__
|
|
#include "enemy\nboss.h"
|
|
#endif
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
// Enemy NPCs
|
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
s32 CNpcEnemy::playerXDist;
|
|
s32 CNpcEnemy::playerYDist;
|
|
s32 CNpcEnemy::playerXDistSqr;
|
|
s32 CNpcEnemy::playerYDistSqr;
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
CNpcEnemy::NPC_UNIT_TYPE CNpcEnemy::getTypeFromMapEdit( u16 newType )
|
|
{
|
|
return( mapEditConvertTable[newType - NPC_ENEMY_MAPEDIT_OFFSET] );
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
CNpcEnemy *CNpcEnemy::Create(int enemyType)
|
|
{
|
|
switch( enemyType )
|
|
{
|
|
case CNpcEnemy::NPC_MOTHER_JELLYFISH:
|
|
case CNpcEnemy::NPC_SUB_SHARK:
|
|
case CNpcEnemy::NPC_FLYING_DUTCHMAN:
|
|
case CNpcEnemy::NPC_IRON_DOGFISH:
|
|
{
|
|
if ( CLevel::getIsBossRespawn() )
|
|
{
|
|
if ( CLevel::getBossHealth() <= 0 )
|
|
{
|
|
return( NULL );
|
|
}
|
|
}
|
|
}
|
|
|
|
case CNpcEnemy::NPC_SEA_SNAKE:
|
|
{
|
|
if ( CLevel::getIsBossRespawn() )
|
|
{
|
|
if ( CLevel::getBossHealth() < 0 )
|
|
{
|
|
return( NULL );
|
|
}
|
|
}
|
|
}
|
|
|
|
default:
|
|
break;
|
|
}
|
|
|
|
CNpcEnemy *enemy;
|
|
|
|
enemy = (CNpcEnemy*)CThingManager::GetThing(TYPE_ENEMY,enemyType);
|
|
if (!enemy)
|
|
switch( enemyType )
|
|
{
|
|
case CNpcEnemy::NPC_SMALL_JELLYFISH_1:
|
|
case CNpcEnemy::NPC_SMALL_JELLYFISH_2:
|
|
{
|
|
enemy = new ("small jellyfish") CNpcSmallJellyfishEnemy;
|
|
break;
|
|
}
|
|
|
|
case CNpcEnemy::NPC_HERMIT_CRAB:
|
|
{
|
|
enemy = new ("hermit crab") CNpcHermitCrabEnemy;
|
|
break;
|
|
}
|
|
|
|
case CNpcEnemy::NPC_SPIDER_CRAB:
|
|
{
|
|
enemy = new ("spider crab") CNpcSpiderCrabEnemy;
|
|
break;
|
|
}
|
|
|
|
case CNpcEnemy::NPC_SPIDER_CRAB_SPAWNER:
|
|
{
|
|
enemy = new ("spider crab spawner") CNpcEnemyGenerator;
|
|
break;
|
|
}
|
|
|
|
case CNpcEnemy::NPC_ANEMONE_1:
|
|
{
|
|
enemy = new ("anemone 1") CNpcAnemone1Enemy;
|
|
break;
|
|
}
|
|
|
|
case CNpcEnemy::NPC_ANEMONE_2:
|
|
{
|
|
enemy = new ("anemone 2") CNpcAnemone2Enemy;
|
|
break;
|
|
}
|
|
|
|
case CNpcEnemy::NPC_ANEMONE_3:
|
|
{
|
|
enemy = new ("anemone 3") CNpcAnemone3Enemy;
|
|
break;
|
|
}
|
|
|
|
case CNpcEnemy::NPC_CLAM_JUMP:
|
|
{
|
|
enemy = new ("jumping clam") CNpcJumpingClamEnemy;
|
|
break;
|
|
}
|
|
|
|
case CNpcEnemy::NPC_CLAM_STATIC:
|
|
{
|
|
enemy = new ("static clam") CNpcStaticClamEnemy;
|
|
break;
|
|
}
|
|
|
|
case CNpcEnemy::NPC_BABY_OCTOPUS:
|
|
{
|
|
enemy = new ("baby octopus") CNpcBabyOctopusEnemy;
|
|
break;
|
|
}
|
|
|
|
case CNpcEnemy::NPC_FISH_FOLK:
|
|
case CNpcEnemy::NPC_ZOMBIE_FISH_FOLK:
|
|
{
|
|
enemy = new ("fish folk") CNpcFishFolk;
|
|
break;
|
|
}
|
|
|
|
case CNpcEnemy::NPC_BALL_BLOB:
|
|
{
|
|
enemy = new ("ball blob") CNpcBallBlobEnemy;
|
|
break;
|
|
}
|
|
|
|
case CNpcEnemy::NPC_PUFFA_FISH:
|
|
{
|
|
enemy = new ("puffa fish") CNpcPuffaFishEnemy;
|
|
break;
|
|
}
|
|
|
|
case CNpcEnemy::NPC_SHARK_MAN:
|
|
{
|
|
enemy = new ("shark man") CNpcSharkManEnemy;
|
|
break;
|
|
}
|
|
|
|
case CNpcEnemy::NPC_SKELETAL_FISH:
|
|
{
|
|
enemy = new ("skeletal fish") CNpcSkeletalFishEnemy;
|
|
break;
|
|
}
|
|
|
|
case CNpcEnemy::NPC_EYEBALL:
|
|
{
|
|
enemy = new ("eyeball") CNpcEyeballEnemy;
|
|
break;
|
|
}
|
|
|
|
case CNpcEnemy::NPC_FLAMING_SKULL:
|
|
{
|
|
enemy = new ("flaming skull") CNpcFlamingSkullEnemy;
|
|
break;
|
|
}
|
|
|
|
case CNpcEnemy::NPC_SKULL_STOMPER:
|
|
{
|
|
enemy = new ("skull stomper") CNpcSkullStomperEnemy;
|
|
break;
|
|
}
|
|
|
|
case CNpcEnemy::NPC_BOOGER_MONSTER:
|
|
{
|
|
enemy = new ("booger monster") CNpcBoogerMonsterEnemy;
|
|
break;
|
|
}
|
|
|
|
case CNpcEnemy::NPC_MOTHER_JELLYFISH:
|
|
{
|
|
enemy = new ("mother jellyfish") CNpcMotherJellyfishEnemy;
|
|
break;
|
|
}
|
|
|
|
case CNpcEnemy::NPC_SUB_SHARK:
|
|
{
|
|
enemy = new ("sub shark") CNpcSubSharkEnemy;
|
|
break;
|
|
}
|
|
|
|
case CNpcEnemy::NPC_PARASITIC_WORM:
|
|
{
|
|
enemy = new ("parasitic worm") CNpcParasiticWormEnemy;
|
|
break;
|
|
}
|
|
|
|
case CNpcEnemy::NPC_FLYING_DUTCHMAN:
|
|
{
|
|
enemy = new ("flying dutchman") CNpcFlyingDutchmanEnemy;
|
|
break;
|
|
}
|
|
|
|
case CNpcEnemy::NPC_IRON_DOGFISH:
|
|
{
|
|
enemy = new ("iron dogfish") CNpcIronDogfishEnemy;
|
|
break;
|
|
}
|
|
|
|
case CNpcEnemy::NPC_DUST_DEVIL:
|
|
{
|
|
enemy = new ("dust devil") CNpcDustDevilEnemy;
|
|
break;
|
|
}
|
|
|
|
case CNpcEnemy::NPC_SQUID_DART:
|
|
{
|
|
enemy = new ("squid dart") CNpcSquidDartEnemy;
|
|
break;
|
|
}
|
|
|
|
case CNpcEnemy::NPC_PRICKLY_BUG:
|
|
{
|
|
enemy = new ("prickly bug") CNpcPricklyBugEnemy;
|
|
break;
|
|
}
|
|
|
|
case CNpcEnemy::NPC_SEA_SNAKE:
|
|
{
|
|
enemy = new ("sea snake") CNpcSeaSnakeEnemy;
|
|
break;
|
|
}
|
|
|
|
case CNpcEnemy::NPC_ANGLER_FISH:
|
|
{
|
|
enemy = new ("angler fish") CNpcAnglerFish;
|
|
break;
|
|
}
|
|
|
|
case CNpcEnemy::NPC_GHOST:
|
|
{
|
|
enemy = new ("ghost") CNpcGhostEnemy;
|
|
break;
|
|
}
|
|
|
|
case CNpcEnemy::NPC_SHELL:
|
|
{
|
|
enemy = new ("shell") CNpcShellEnemy;
|
|
break;
|
|
}
|
|
|
|
case CNpcEnemy::NPC_SMALL_JELLYFISH_BACKGROUND:
|
|
{
|
|
enemy = new ("small jellyfish background") CNpcSmallJellyfishBackgroundEnemy;
|
|
break;
|
|
}
|
|
|
|
case CNpcEnemy::NPC_BUTTERFLY_BACKGROUND:
|
|
{
|
|
enemy = new ("butterfly background") CNpcButterflyBackgroundEnemy;
|
|
break;
|
|
}
|
|
|
|
case CNpcEnemy::NPC_PROJECTILE_JELLYFISH:
|
|
{
|
|
enemy = new ("projectile jellyfish") CNpcSmallJellyfishProjectileEnemy;
|
|
break;
|
|
}
|
|
|
|
default:
|
|
{
|
|
SYSTEM_DBGMSG("UNKNOWN %i\n",enemyType);
|
|
enemy = new ("npc enemy") CNpcEnemy;
|
|
ASSERT(0);
|
|
break;
|
|
}
|
|
}
|
|
|
|
ASSERT(enemy);
|
|
|
|
enemy->setType( (NPC_UNIT_TYPE)enemyType );
|
|
enemy->setThingSubType(enemyType);
|
|
|
|
enemy->init();
|
|
|
|
return(enemy);
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
CNpcEnemy *CNpcEnemy::Create(sThingActor *ThisActor)
|
|
{
|
|
CNpcEnemy *enemy;
|
|
|
|
NPC_UNIT_TYPE enemyType = CNpcEnemy::getTypeFromMapEdit( ThisActor->Type );
|
|
|
|
enemy = (CNpcEnemy*)Create(enemyType);
|
|
|
|
return(enemy);
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
void CNpcEnemy::setupWaypoints( sThingActor *ThisActor )
|
|
{
|
|
u16 *PntList=(u16*)MakePtr(ThisActor,sizeof(sThingActor));
|
|
|
|
u16 newXPos, newYPos;
|
|
|
|
m_npcPath.setWaypointCount( ThisActor->PointCount - 1 );
|
|
|
|
newXPos = (u16) *PntList;
|
|
setWaypointPtr( PntList );
|
|
PntList++;
|
|
newYPos = (u16) *PntList;
|
|
PntList++;
|
|
|
|
setStartPos( newXPos, newYPos );
|
|
|
|
if ( ThisActor->PointCount > 1 )
|
|
{
|
|
newXPos = (u16) *PntList;
|
|
PntList++;
|
|
newYPos = (u16) *PntList;
|
|
PntList++;
|
|
|
|
setHeading( newXPos, newYPos );
|
|
}
|
|
|
|
setThinkArea();
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
void CNpcEnemy::setThinkArea()
|
|
{
|
|
s32 minX, maxX, minY, maxY;
|
|
|
|
m_npcPath.getPathXExtents( &minX, &maxX );
|
|
m_npcPath.getPathYExtents( &minY, &maxY );
|
|
|
|
m_thinkArea.x1 = minX;
|
|
m_thinkArea.x2 = maxX;
|
|
m_thinkArea.y1 = minY;
|
|
m_thinkArea.y2 = maxY;
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
void CNpcEnemy::setStartPos( s32 xPos, s32 yPos )
|
|
{
|
|
Pos.vx = ( xPos << 4 ) + 8;
|
|
Pos.vy = ( yPos << 4 ) + 16;
|
|
|
|
m_initPos = m_base = Pos;
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
void CNpcEnemy::setStartPosHighRes( s32 xPos, s32 yPos )
|
|
{
|
|
Pos.vx = xPos;
|
|
Pos.vy = yPos;
|
|
|
|
m_initPos = m_base = Pos;
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
void CNpcEnemy::setHeading( s32 xPos, s32 yPos )
|
|
{
|
|
m_heading = ( ratan2( ( ( yPos << 4 ) + 16 ) - Pos.vy, ( ( xPos << 4 ) + 8 ) - Pos.vx ) ) & 4095;
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
void CNpcEnemy::init()
|
|
{
|
|
CEnemyThing::init();
|
|
if (m_data[m_type].skelType)
|
|
{
|
|
m_actorGfx=CActorPool::GetActor((FileEquate)m_data[m_type].skelType);
|
|
}
|
|
else
|
|
{
|
|
m_actorGfx=0;
|
|
}
|
|
|
|
m_animPlaying = true;
|
|
m_animNo = m_data[m_type].initAnim;
|
|
m_frame = 0;
|
|
|
|
m_movementTimer = 0;
|
|
m_timerTimer = 0;
|
|
m_velocity = 0;
|
|
m_extension = 0;
|
|
m_reversed = false;
|
|
m_isActive = true;
|
|
m_isDying = false;
|
|
|
|
m_health = m_data[this->m_type].initHealth;
|
|
|
|
m_extendDir = EXTEND_RIGHT;
|
|
|
|
m_timerFunc = m_data[this->m_type].timerFunc;
|
|
m_sensorFunc = m_data[this->m_type].sensorFunc;
|
|
m_movementFunc = m_data[this->m_type].movementFunc;
|
|
|
|
m_controlFunc = NPC_CONTROL_MOVEMENT;
|
|
|
|
m_npcPath.initPath();
|
|
|
|
m_drawRotation = 0;
|
|
m_isCaught = false;
|
|
m_isBlowerOn = false;
|
|
m_speed = m_data[m_type].speed;
|
|
m_heading = 0;
|
|
m_RGB = 0;
|
|
|
|
m_soundId = (int) NOT_PLAYING;
|
|
|
|
updateCollisionArea();
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
void CNpcEnemy::setSpeed( s16 newSpeed )
|
|
{
|
|
switch( m_type )
|
|
{
|
|
case CNpcEnemy::NPC_MOTHER_JELLYFISH:
|
|
case CNpcEnemy::NPC_SUB_SHARK:
|
|
case CNpcEnemy::NPC_FLYING_DUTCHMAN:
|
|
case CNpcEnemy::NPC_IRON_DOGFISH:
|
|
case CNpcEnemy::NPC_SEA_SNAKE:
|
|
{
|
|
if ( CLevel::getIsBossRespawn() )
|
|
{
|
|
break;
|
|
}
|
|
else
|
|
{
|
|
m_speed = newSpeed;
|
|
break;
|
|
}
|
|
}
|
|
|
|
default:
|
|
{
|
|
m_speed = newSpeed;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
void CNpcEnemy::postInit()
|
|
{
|
|
m_npcPath.setPathType( CNpcPath::PONG_PATH );
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
void CNpcEnemy::reinit()
|
|
{
|
|
m_animPlaying = true;
|
|
m_animNo = m_data[m_type].initAnim;
|
|
m_frame = 0;
|
|
|
|
m_movementTimer = 0;
|
|
m_timerTimer = 0;
|
|
m_velocity = 0;
|
|
m_extension = 0;
|
|
m_reversed = false;
|
|
m_isActive = true;
|
|
m_isDying = false;
|
|
|
|
m_health = m_data[this->m_type].initHealth;
|
|
|
|
m_extendDir = EXTEND_RIGHT;
|
|
|
|
m_timerFunc = m_data[this->m_type].timerFunc;
|
|
m_sensorFunc = m_data[this->m_type].sensorFunc;
|
|
m_movementFunc = m_data[this->m_type].movementFunc;
|
|
|
|
m_controlFunc = NPC_CONTROL_MOVEMENT;
|
|
|
|
Pos = m_initPos;
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
void CNpcEnemy::shutdown()
|
|
{
|
|
if ( m_soundId != NOT_PLAYING )
|
|
{
|
|
CSoundMediator::stopAndUnlockSfx( (xmPlayingId) m_soundId );
|
|
}
|
|
|
|
if (m_actorGfx) delete m_actorGfx;
|
|
|
|
CEnemyThing::shutdown();
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
void CNpcEnemy::leftThinkZone(int _frames)
|
|
{
|
|
if ( m_soundId != NOT_PLAYING )
|
|
{
|
|
CSoundMediator::stopAndUnlockSfx( (xmPlayingId) m_soundId );
|
|
m_soundId = NOT_PLAYING;
|
|
}
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
int CNpcEnemy::getFrameCount()
|
|
{
|
|
return( m_actorGfx->getFrameCount( m_animNo ) );
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
void CNpcEnemy::processGraphicFlipping()
|
|
{
|
|
if ( m_heading > 1024 && m_heading < 3072 )
|
|
{
|
|
m_reversed = true;
|
|
}
|
|
else
|
|
{
|
|
m_reversed = false;
|
|
}
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
s32 CNpcEnemy::getFrameShift( int _frames )
|
|
{
|
|
return( _frames << 8 );
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
void CNpcEnemy::processAnimFrames( int _frames )
|
|
{
|
|
if ( m_animPlaying && ( ( m_isDying && m_data[m_type].playDeathAnim ) || !m_isDying ) )
|
|
{
|
|
s32 frameCount;
|
|
|
|
frameCount = getFrameCount();
|
|
|
|
s32 frameShift = getFrameShift( _frames );
|
|
|
|
if ( ( frameCount << 8 ) - m_frame > frameShift )
|
|
{
|
|
m_frame += frameShift;
|
|
}
|
|
else
|
|
{
|
|
m_frame = ( frameCount - 1 ) << 8;
|
|
m_animPlaying = false;
|
|
}
|
|
}
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
void CNpcEnemy::think(int _frames)
|
|
{
|
|
int moveFrames = _frames;
|
|
|
|
if ( moveFrames > 2 )
|
|
{
|
|
// make sure enemies don't go berserk if too many frames are dropped
|
|
|
|
moveFrames = 2;
|
|
}
|
|
|
|
CEnemyThing::think(_frames);
|
|
|
|
processGenericGetUserDist( moveFrames, &playerXDist, &playerYDist );
|
|
playerXDistSqr = playerXDist * playerXDist;
|
|
playerYDistSqr = playerYDist * playerYDist;
|
|
|
|
if ( m_soundId != NOT_PLAYING )
|
|
{
|
|
if( !CSoundMediator::isSfxStillPlaying( (xmPlayingId) m_soundId ) )
|
|
{
|
|
// unlock sound if it has finished
|
|
|
|
CSoundMediator::stopAndUnlockSfx( (xmPlayingId) m_soundId );
|
|
m_soundId = NOT_PLAYING;
|
|
}
|
|
}
|
|
|
|
if ( m_isCaught )
|
|
{
|
|
processCoralBlower( moveFrames );
|
|
}
|
|
else
|
|
{
|
|
if ( m_isActive )
|
|
{
|
|
processAnimFrames( moveFrames );
|
|
|
|
switch ( this->m_controlFunc )
|
|
{
|
|
case NPC_CONTROL_NONE:
|
|
return;
|
|
|
|
case NPC_CONTROL_MOVEMENT:
|
|
if ( !processSensor() )
|
|
{
|
|
processMovement( moveFrames );
|
|
}
|
|
else
|
|
{
|
|
processClose( moveFrames );
|
|
}
|
|
|
|
break;
|
|
|
|
case NPC_CONTROL_SHOT:
|
|
processShot( moveFrames );
|
|
|
|
break;
|
|
|
|
case NPC_CONTROL_CLOSE:
|
|
processClose( moveFrames );
|
|
|
|
break;
|
|
|
|
case NPC_CONTROL_COLLISION:
|
|
processCollision();
|
|
|
|
break;
|
|
}
|
|
|
|
processGraphicFlipping();
|
|
}
|
|
}
|
|
|
|
if ( !m_isCaught )
|
|
{
|
|
processTimer( moveFrames );
|
|
}
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
void CNpcEnemy::processAttackCollision()
|
|
{
|
|
//if ( m_controlFunc == NPC_CONTROL_CLOSE && m_data[m_type].closeFunc != NPC_CLOSE_NONE )
|
|
if ( m_controlFunc == NPC_CONTROL_CLOSE )
|
|
{
|
|
// only detect collision if in attack mode
|
|
|
|
m_oldControlFunc = m_controlFunc;
|
|
m_controlFunc = NPC_CONTROL_COLLISION;
|
|
}
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
void CNpcEnemy::drawAttackEffect()
|
|
{
|
|
CRECT rect;
|
|
rect = getCollisionArea();
|
|
|
|
DVECTOR thwakPos;
|
|
|
|
s32 xDist;
|
|
|
|
CPlayer *player = GameScene.getPlayer();
|
|
DVECTOR playerPos = player->getPos();
|
|
|
|
xDist = playerPos.vx - this->Pos.vx;
|
|
|
|
if ( xDist > 0 )
|
|
{
|
|
thwakPos.vx = rect.x2;
|
|
|
|
if ( rect.x1 > thwakPos.vx )
|
|
{
|
|
thwakPos.vx = rect.x1;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
thwakPos.vx = rect.x1;
|
|
|
|
if ( rect.x2 < thwakPos.vx )
|
|
{
|
|
thwakPos.vx = rect.x2;
|
|
}
|
|
}
|
|
|
|
thwakPos.vy = ( rect.y1 + rect.y2 ) >> 1;
|
|
|
|
CFX::Create( CFX::FX_TYPE_THWACK, thwakPos );
|
|
CPadVibrationManager::setVibration(0,CPadVibrationManager::VIBE_CHOP);
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
void CNpcEnemy::collidedWith( CThing *_thisThing )
|
|
{
|
|
if ( m_isActive && !m_isCaught && !m_isDying )
|
|
{
|
|
switch(_thisThing->getThingType())
|
|
{
|
|
case TYPE_PLAYER:
|
|
{
|
|
CPlayer *player = (CPlayer *) _thisThing;
|
|
|
|
ATTACK_STATE playerState = player->getAttackState();
|
|
|
|
if(playerState==ATTACK_STATE__NONE)
|
|
{
|
|
if ( !player->isRecoveringFromHit() )
|
|
{
|
|
switch( m_data[m_type].detectCollision )
|
|
{
|
|
case DETECT_NO_COLLISION:
|
|
{
|
|
// ignore
|
|
|
|
break;
|
|
}
|
|
|
|
case DETECT_ALL_COLLISION:
|
|
{
|
|
m_oldControlFunc = m_controlFunc;
|
|
m_controlFunc = NPC_CONTROL_COLLISION;
|
|
|
|
processUserCollision( _thisThing );
|
|
|
|
break;
|
|
}
|
|
|
|
case DETECT_ATTACK_COLLISION_GENERIC:
|
|
{
|
|
processAttackCollision();
|
|
processUserCollision( _thisThing );
|
|
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// player is attacking, respond appropriately
|
|
|
|
if ( m_controlFunc != NPC_CONTROL_SHOT )
|
|
{
|
|
if(playerState==ATTACK_STATE__BUTT_BOUNCE)
|
|
{
|
|
player->justButtBouncedABadGuy();
|
|
}
|
|
m_controlFunc = NPC_CONTROL_SHOT;
|
|
m_state = NPC_GENERIC_HIT_CHECK_HEALTH;
|
|
|
|
drawAttackEffect();
|
|
}
|
|
}
|
|
|
|
break;
|
|
}
|
|
|
|
case TYPE_ENEMY:
|
|
{
|
|
CNpcEnemy *enemy = (CNpcEnemy *) _thisThing;
|
|
|
|
if ( canCollideWithEnemy() && enemy->canCollideWithEnemy() )
|
|
{
|
|
processEnemyCollision( _thisThing );
|
|
}
|
|
|
|
break;
|
|
}
|
|
|
|
default:
|
|
ASSERT(0);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
bool CNpcEnemy::processSensor()
|
|
{
|
|
switch( m_sensorFunc )
|
|
{
|
|
case NPC_SENSOR_NONE:
|
|
return( false );
|
|
|
|
default:
|
|
{
|
|
switch( m_sensorFunc )
|
|
{
|
|
case NPC_SENSOR_GENERIC_USER_CLOSE:
|
|
{
|
|
if ( playerXDistSqr + playerYDistSqr < 10000 )
|
|
{
|
|
m_controlFunc = NPC_CONTROL_CLOSE;
|
|
m_velocity = m_speed;
|
|
|
|
return( true );
|
|
}
|
|
else
|
|
{
|
|
return( false );
|
|
}
|
|
}
|
|
|
|
case NPC_SENSOR_GENERIC_USER_VISIBLE:
|
|
{
|
|
s32 xDistWaypoint, yDistWaypoint;
|
|
|
|
if ( abs( playerXDist ) < 500 )
|
|
{
|
|
// within range
|
|
|
|
// make sure user is closer to npc than next waypoint
|
|
|
|
s32 xDistWaypoint, yDistWaypoint;
|
|
|
|
m_npcPath.getDistToNextWaypoint( Pos, &xDistWaypoint, &yDistWaypoint );
|
|
|
|
if ( abs( playerXDist ) < abs( xDistWaypoint ) )
|
|
{
|
|
s16 headingToPlayer = ratan2( playerYDist, playerXDist );
|
|
|
|
s16 decDir, incDir, moveDist;
|
|
|
|
s32 headingToWaypoint = ratan2( yDistWaypoint, xDistWaypoint );
|
|
|
|
// check waypoint is in the same direction as the user
|
|
|
|
decDir = headingToPlayer - headingToWaypoint;
|
|
|
|
if ( decDir < 0 )
|
|
{
|
|
decDir += ONE;
|
|
}
|
|
|
|
incDir = headingToWaypoint - headingToPlayer;
|
|
|
|
if ( incDir < 0 )
|
|
{
|
|
incDir += ONE;
|
|
}
|
|
|
|
if ( decDir < incDir )
|
|
{
|
|
moveDist = decDir;
|
|
}
|
|
else
|
|
{
|
|
moveDist = incDir;
|
|
}
|
|
|
|
if ( moveDist > 512 )
|
|
{
|
|
return( false );
|
|
}
|
|
else
|
|
{
|
|
// check if npc is facing user
|
|
|
|
decDir = m_heading - headingToPlayer;
|
|
|
|
if ( decDir < 0 )
|
|
{
|
|
decDir += ONE;
|
|
}
|
|
|
|
incDir = headingToPlayer - m_heading;
|
|
|
|
if ( incDir < 0 )
|
|
{
|
|
incDir += ONE;
|
|
}
|
|
|
|
if ( decDir < incDir )
|
|
{
|
|
moveDist = decDir;
|
|
}
|
|
else
|
|
{
|
|
moveDist = incDir;
|
|
}
|
|
|
|
if ( moveDist < 1024 )
|
|
{
|
|
m_controlFunc = NPC_CONTROL_CLOSE;
|
|
m_velocity = 8;
|
|
|
|
return( true );
|
|
}
|
|
else
|
|
{
|
|
return( false );
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
return( false );
|
|
}
|
|
}
|
|
else
|
|
{
|
|
return( false );
|
|
}
|
|
}
|
|
|
|
default:
|
|
return( false );
|
|
}
|
|
}
|
|
|
|
break;
|
|
}
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
void CNpcEnemy::processMovement(int _frames)
|
|
{
|
|
s32 moveX = 0, moveY = 0;
|
|
s32 moveVel = 0;
|
|
s32 moveDist = 0;
|
|
|
|
if ( m_soundId == NOT_PLAYING && m_data[m_type].moveSfx < CSoundMediator::NUM_SFXIDS )
|
|
{
|
|
m_soundId = (int) CSoundMediator::playSfx( m_data[m_type].moveSfx, true );
|
|
}
|
|
|
|
switch( m_movementFunc )
|
|
{
|
|
case NPC_MOVEMENT_STATIC:
|
|
{
|
|
break;
|
|
}
|
|
|
|
case NPC_MOVEMENT_STATIC_CYCLE_ANIM:
|
|
{
|
|
if ( !m_animPlaying )
|
|
{
|
|
m_animPlaying = true;
|
|
m_animNo = m_data[m_type].initAnim;
|
|
m_frame = 0;
|
|
}
|
|
|
|
break;
|
|
}
|
|
|
|
case NPC_MOVEMENT_FIXED_PATH:
|
|
{
|
|
processGenericFixedPathMove( _frames, &moveX, &moveY, &moveVel, &moveDist );
|
|
|
|
if ( !m_animPlaying )
|
|
{
|
|
m_animPlaying = true;
|
|
m_animNo = m_data[m_type].moveAnim;
|
|
m_frame = 0;
|
|
}
|
|
|
|
break;
|
|
}
|
|
|
|
case NPC_MOVEMENT_FIXED_PATH_WALK:
|
|
{
|
|
processGenericFixedPathWalk( _frames, &moveX, &moveY );
|
|
|
|
if ( !m_animPlaying )
|
|
{
|
|
m_animPlaying = true;
|
|
m_animNo = m_data[m_type].moveAnim;
|
|
m_frame = 0;
|
|
}
|
|
|
|
break;
|
|
}
|
|
|
|
default:
|
|
|
|
break;
|
|
}
|
|
|
|
processMovementModifier( _frames, moveX, moveY, moveVel, moveDist );
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
void CNpcEnemy::processMovementModifier(int _frames, s32 distX, s32 distY, s32 dist, s16 headingChange)
|
|
{
|
|
Pos.vx += distX;
|
|
Pos.vy += distY;
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
u8 CNpcEnemy::hasBeenAttacked()
|
|
{
|
|
if ( m_controlFunc != NPC_CONTROL_SHOT )
|
|
{
|
|
m_controlFunc = NPC_CONTROL_SHOT;
|
|
m_state = NPC_GENERIC_HIT_CHECK_HEALTH;
|
|
}
|
|
|
|
return( true );
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
void CNpcEnemy::processShot( int _frames )
|
|
{
|
|
switch( m_data[m_type].shotFunc )
|
|
{
|
|
case NPC_SHOT_NONE:
|
|
{
|
|
// do nothing
|
|
m_controlFunc = m_oldControlFunc;
|
|
|
|
break;
|
|
}
|
|
|
|
case NPC_SHOT_GENERIC:
|
|
{
|
|
switch ( m_state )
|
|
{
|
|
case NPC_GENERIC_HIT_CHECK_HEALTH:
|
|
{
|
|
m_health -= 3;
|
|
|
|
if ( m_health <= 0 )
|
|
{
|
|
m_state = NPC_GENERIC_HIT_DEATH_START;
|
|
m_isDying = true;
|
|
m_health = 0;
|
|
}
|
|
else
|
|
{
|
|
m_state = NPC_GENERIC_HIT_RECOIL;
|
|
|
|
m_animPlaying = true;
|
|
m_animNo = m_data[m_type].recoilAnim;
|
|
m_frame = 0;
|
|
}
|
|
|
|
break;
|
|
}
|
|
|
|
case NPC_GENERIC_HIT_RECOIL:
|
|
{
|
|
processShotRecoil( _frames );
|
|
|
|
break;
|
|
}
|
|
|
|
case NPC_GENERIC_HIT_DEATH_START:
|
|
{
|
|
processShotDeathStart( _frames );
|
|
|
|
break;
|
|
}
|
|
|
|
case NPC_GENERIC_HIT_DEATH_END:
|
|
{
|
|
processShotDeathEnd( _frames );
|
|
|
|
break;
|
|
}
|
|
}
|
|
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
void CNpcEnemy::processShotRecoil( int _frames )
|
|
{
|
|
if ( !m_animPlaying )
|
|
{
|
|
m_state = 0;
|
|
m_controlFunc = NPC_CONTROL_MOVEMENT;
|
|
}
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
void CNpcEnemy::processShotDeathStart( int _frames )
|
|
{
|
|
m_animPlaying = true;
|
|
m_animNo = m_data[m_type].dieAnim;
|
|
m_frame = 0;
|
|
m_state = NPC_GENERIC_HIT_DEATH_END;
|
|
m_isDying = true;
|
|
|
|
if ( m_data[m_type].deathSfx < CSoundMediator::NUM_SFXIDS )
|
|
{
|
|
if( m_soundId != NOT_PLAYING )
|
|
{
|
|
CSoundMediator::stopAndUnlockSfx( (xmPlayingId) m_soundId );
|
|
}
|
|
|
|
m_soundId = (int) CSoundMediator::playSfx( m_data[m_type].deathSfx, true );
|
|
}
|
|
|
|
m_speed = -5;
|
|
|
|
if (m_data[m_type].skelType)
|
|
{
|
|
m_actorGfx->SetOtPos( 0 );
|
|
}
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
void CNpcEnemy::processShotDeathEnd( int _frames )
|
|
{
|
|
m_drawRotation += 64 * _frames;
|
|
m_drawRotation &= 4095;
|
|
|
|
Pos.vy += m_speed * _frames;
|
|
|
|
if ( m_speed < 5 )
|
|
{
|
|
m_speed++;
|
|
}
|
|
|
|
DVECTOR offset = CLevel::getCameraPos();
|
|
|
|
if ( Pos.vy - offset.vy > VidGetScrH() )
|
|
{
|
|
if ( m_data[m_type].respawning )
|
|
{
|
|
m_isActive = false;
|
|
|
|
m_timerFunc = NPC_TIMER_RESPAWN;
|
|
m_timerTimer = 4 * GameState::getOneSecondInFrames();
|
|
}
|
|
else
|
|
{
|
|
setToShutdown();
|
|
}
|
|
}
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
void CNpcEnemy::processClose(int _frames)
|
|
{
|
|
switch( m_data[this->m_type].closeFunc )
|
|
{
|
|
case NPC_CLOSE_GENERIC_USER_SEEK:
|
|
{
|
|
processGenericGotoTarget( _frames, playerXDist, playerYDist, m_speed );
|
|
|
|
break;
|
|
}
|
|
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
void CNpcEnemy::processCollision()
|
|
{
|
|
CPlayer *player = GameScene.getPlayer();
|
|
|
|
player->takeDamage( m_data[m_type].damageToUserType,REACT__GET_DIRECTION_FROM_THING,(CThing*)this );
|
|
processUserCollision( (CThing *) player );
|
|
|
|
m_controlFunc = m_oldControlFunc;
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
void CNpcEnemy::processTimer(int _frames)
|
|
{
|
|
if ( m_timerTimer > 0 )
|
|
{
|
|
m_timerTimer -= _frames;
|
|
}
|
|
|
|
switch( m_timerFunc )
|
|
{
|
|
case NPC_TIMER_NONE:
|
|
{
|
|
break;
|
|
}
|
|
|
|
case NPC_TIMER_EVADE_DONE:
|
|
case NPC_TIMER_ATTACK_DONE:
|
|
{
|
|
if ( m_timerTimer <= 0 )
|
|
{
|
|
m_timerFunc = NPC_TIMER_NONE;
|
|
m_sensorFunc = m_data[this->m_type].sensorFunc;
|
|
}
|
|
|
|
break;
|
|
}
|
|
|
|
case NPC_TIMER_RESPAWN:
|
|
{
|
|
if ( m_timerTimer <= 0 )
|
|
{
|
|
reinit();
|
|
}
|
|
|
|
break;
|
|
}
|
|
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
void CNpcEnemy::render()
|
|
{
|
|
SprFrame = NULL;
|
|
|
|
if ( m_isActive )
|
|
{
|
|
CEnemyThing::render();
|
|
|
|
if (canRender())
|
|
{
|
|
DVECTOR &renderPos=getRenderPos();
|
|
|
|
SprFrame = m_actorGfx->Render(renderPos,m_animNo,( m_frame >> 8 ),m_reversed);
|
|
m_actorGfx->RotateScale( SprFrame, renderPos, m_drawRotation, 4096, 4096 );
|
|
|
|
sBBox boundingBox = m_actorGfx->GetBBox();
|
|
setCollisionSize( ( boundingBox.XMax - boundingBox.XMin ), ( boundingBox.YMax - boundingBox.YMin ) );
|
|
setCollisionCentreOffset( ( boundingBox.XMax + boundingBox.XMin ) >> 1, ( boundingBox.YMax + boundingBox.YMin ) >> 1 );
|
|
}
|
|
}
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
void CNpcEnemy::processEvent( GAME_EVENT evt, CThing *sourceThing )
|
|
{
|
|
switch( evt )
|
|
{
|
|
case PROJECTILE_RETURNED_TO_SOURCE_EVENT:
|
|
{
|
|
m_controlFunc = NPC_CONTROL_MOVEMENT;
|
|
m_timerFunc = NPC_TIMER_ATTACK_DONE;
|
|
m_timerTimer = GameState::getOneSecondInFrames();
|
|
m_sensorFunc = NPC_SENSOR_NONE;
|
|
|
|
//removeChild( sourceThing );
|
|
//sourceThing->shutdown();
|
|
//delete sourceThing;
|
|
|
|
CProjectile *projectile;
|
|
projectile = (CProjectile *) sourceThing;
|
|
projectile->setMovementType( CProjectile::PROJECTILE_FIXED );
|
|
projectile->setPosition( Pos );
|
|
|
|
break;
|
|
}
|
|
|
|
case BOSS_FOUND_EVENT:
|
|
{
|
|
addHealthMeter();
|
|
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
u8 CNpcEnemy::canBeCaughtByNet()
|
|
{
|
|
return( m_isActive && !m_isDying && m_data[m_type].canBeNetted );
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
void CNpcEnemy::caughtWithNet()
|
|
{
|
|
if ( m_data[m_type].respawning )
|
|
{
|
|
if ( m_isActive )
|
|
{
|
|
m_isActive = false;
|
|
|
|
m_timerFunc = NPC_TIMER_RESPAWN;
|
|
m_timerTimer = 4 * GameState::getOneSecondInFrames();
|
|
}
|
|
}
|
|
else
|
|
{
|
|
setToShutdown();
|
|
}
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
int CNpcEnemy::canCollide()
|
|
{
|
|
return( m_isActive && !m_isDying );
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
void CNpcEnemy::processEnemyCollision( CThing *thisThing )
|
|
{
|
|
DVECTOR otherPos = thisThing->getPos();
|
|
DVECTOR otherDelta = thisThing->getPosDelta();
|
|
|
|
s32 xDist = Pos.vx - otherPos.vx;
|
|
s32 yDist = Pos.vy - otherPos.vy;
|
|
|
|
s16 headingFromTarget = ratan2( yDist, xDist );
|
|
|
|
if ( ( xDist > 0 && otherDelta.vx < 0 ) || ( xDist < 0 && otherDelta.vx > 0 ) )
|
|
{
|
|
otherDelta.vx = -otherDelta.vx;
|
|
}
|
|
|
|
if ( ( yDist > 0 && otherDelta.vy < 0 ) || ( yDist < 0 && otherDelta.vy > 0 ) )
|
|
{
|
|
otherDelta.vy = -otherDelta.vy;
|
|
}
|
|
|
|
Pos.vx += otherDelta.vx;
|
|
|
|
if ( m_npcPath.getWaypointCount() )
|
|
{
|
|
s32 minX, maxX;
|
|
m_npcPath.getPathXExtents( &minX, &maxX );
|
|
|
|
if ( Pos.vx < minX )
|
|
{
|
|
Pos.vx = minX;
|
|
}
|
|
else if ( Pos.vx > maxX )
|
|
{
|
|
Pos.vx = maxX;
|
|
}
|
|
}
|
|
|
|
s32 groundHeight = CGameScene::getCollision()->getHeightFromGround( Pos.vx, Pos.vy, 16 );
|
|
|
|
if ( groundHeight < 8 )
|
|
{
|
|
Pos.vy += groundHeight;
|
|
}
|
|
else
|
|
{
|
|
Pos.vy += otherDelta.vy;
|
|
}
|
|
|
|
m_heading = headingFromTarget;
|
|
|
|
s32 waypointXDist;
|
|
s32 waypointYDist;
|
|
|
|
m_npcPath.getDistToNextWaypoint( Pos, &waypointXDist, &waypointYDist );
|
|
|
|
if ( ( xDist > 0 && waypointXDist < 0 ) || ( xDist < 0 && waypointXDist > 0 ) )
|
|
{
|
|
// try next waypoint to get around other enemy
|
|
|
|
m_npcPath.incPath();
|
|
}
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
void CNpcEnemy::processUserCollision( CThing *thisThing )
|
|
{
|
|
DVECTOR otherPos = thisThing->getPos();
|
|
DVECTOR otherDelta = thisThing->getPosDelta();
|
|
|
|
s32 xDist = Pos.vx - otherPos.vx;
|
|
s32 yDist = Pos.vy - otherPos.vy;
|
|
|
|
s16 headingFromTarget = ratan2( yDist, xDist );
|
|
|
|
/*if ( ( xDist > 0 && otherDelta.vx < 0 ) || ( xDist < 0 && otherDelta.vx > 0 ) )
|
|
{
|
|
otherDelta.vx = -otherDelta.vx;
|
|
}
|
|
|
|
if ( ( yDist > 0 && otherDelta.vy < 0 ) || ( yDist < 0 && otherDelta.vy > 0 ) )
|
|
{
|
|
otherDelta.vy = -otherDelta.vy;
|
|
}
|
|
|
|
Pos.vx += otherDelta.vx;
|
|
|
|
s32 groundHeight = CGameScene::getCollision()->getHeightFromGround( Pos.vx, Pos.vy, 16 );
|
|
|
|
if ( groundHeight < 8 )
|
|
{
|
|
Pos.vy += groundHeight;
|
|
}
|
|
else
|
|
{
|
|
Pos.vy += otherDelta.vy;
|
|
}*/
|
|
|
|
m_heading = headingFromTarget;
|
|
|
|
s32 waypointXDist;
|
|
s32 waypointYDist;
|
|
|
|
m_npcPath.getDistToNextWaypoint( Pos, &waypointXDist, &waypointYDist );
|
|
|
|
if ( ( xDist > 0 && waypointXDist < 0 ) || ( xDist < 0 && waypointXDist > 0 ) )
|
|
{
|
|
// try next waypoint to get around other enemy
|
|
|
|
m_npcPath.incPath();
|
|
}
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
bool CNpcEnemy::processCoralBlowerMovement( int _frames, s32 xDist, s32 yDist, u8 destroyAtTarget )
|
|
{
|
|
s32 moveX, moveY;
|
|
s16 headingToTarget;
|
|
|
|
headingToTarget = ratan2( yDist, xDist );
|
|
|
|
s32 preShiftX = _frames * 5 * rcos( headingToTarget );
|
|
s32 preShiftY = _frames * 5 * rsin( headingToTarget );
|
|
|
|
moveX = preShiftX >> 12;
|
|
if ( !moveX && preShiftX )
|
|
{
|
|
moveX = preShiftX / abs( preShiftX );
|
|
}
|
|
|
|
if ( xDist > 0 )
|
|
{
|
|
if ( moveX > xDist )
|
|
{
|
|
moveX = xDist;
|
|
}
|
|
}
|
|
else if ( xDist < 0 )
|
|
{
|
|
if ( moveX < xDist )
|
|
{
|
|
moveX = xDist;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
moveX = 0;
|
|
}
|
|
|
|
moveY = preShiftY >> 12;
|
|
if ( !moveY && preShiftY )
|
|
{
|
|
moveY = preShiftY / abs( preShiftY );
|
|
}
|
|
|
|
if ( yDist > 0 )
|
|
{
|
|
if ( moveY > yDist )
|
|
{
|
|
moveY = yDist;
|
|
}
|
|
}
|
|
else if ( yDist < 0 )
|
|
{
|
|
if ( moveY < yDist )
|
|
{
|
|
moveY = yDist;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
moveY = 0;
|
|
}
|
|
|
|
Pos.vx += moveX;
|
|
Pos.vy += moveY;
|
|
|
|
if ( moveX || moveY )
|
|
{
|
|
return( false );
|
|
}
|
|
else
|
|
{
|
|
// has reached target point
|
|
|
|
if ( destroyAtTarget )
|
|
{
|
|
if ( m_data[m_type].respawning )
|
|
{
|
|
if ( m_isActive )
|
|
{
|
|
m_isCaught = false;
|
|
m_isActive = false;
|
|
|
|
m_timerFunc = NPC_TIMER_RESPAWN;
|
|
m_timerTimer = 1 * GameState::getOneSecondInFrames();
|
|
}
|
|
}
|
|
else
|
|
{
|
|
setToShutdown();
|
|
}
|
|
}
|
|
|
|
return( true );
|
|
}
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
bool CNpcEnemy::canBeSuckedUp()
|
|
{
|
|
return( m_data[m_type].canBeSuckedUp );
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
bool CNpcEnemy::suckUp( DVECTOR *suckPos, int _frames )
|
|
{
|
|
m_isCaught = true;
|
|
m_isBlowerOn = true;
|
|
|
|
bool returnVal = false;
|
|
|
|
switch( m_state )
|
|
{
|
|
case NPC_CORAL_BLOWER_SUCK:
|
|
{
|
|
// go to user
|
|
|
|
s32 targetXDist = suckPos->vx - Pos.vx;
|
|
s32 targetYDist = suckPos->vy - Pos.vy;
|
|
|
|
returnVal = processCoralBlowerMovement( _frames, targetXDist, targetYDist, true );
|
|
|
|
break;
|
|
}
|
|
|
|
case NPC_CORAL_BLOWER_RETURN:
|
|
{
|
|
m_state = NPC_CORAL_BLOWER_SUCK;
|
|
|
|
break;
|
|
}
|
|
|
|
default:
|
|
{
|
|
m_state = NPC_CORAL_BLOWER_SUCK;
|
|
m_oldState = m_state;
|
|
m_caughtPos = Pos;
|
|
|
|
break;
|
|
}
|
|
}
|
|
|
|
return( returnVal );
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
void CNpcEnemy::processCoralBlower( int _frames )
|
|
{
|
|
s32 targetXDist, targetYDist;
|
|
|
|
switch( m_state )
|
|
{
|
|
case NPC_CORAL_BLOWER_SUCK:
|
|
{
|
|
if ( !m_isBlowerOn )
|
|
{
|
|
m_state = NPC_CORAL_BLOWER_RETURN;
|
|
}
|
|
|
|
break;
|
|
}
|
|
|
|
case NPC_CORAL_BLOWER_RETURN:
|
|
{
|
|
// go to original position
|
|
|
|
targetXDist = m_caughtPos.vx - Pos.vx;
|
|
targetYDist = m_caughtPos.vy - Pos.vy;
|
|
|
|
processCoralBlowerMovement( _frames, targetXDist, targetYDist, false );
|
|
|
|
if ( !targetXDist && !targetYDist )
|
|
{
|
|
m_state = m_oldState;
|
|
m_isCaught = false;
|
|
}
|
|
|
|
break;
|
|
}
|
|
|
|
default:
|
|
{
|
|
m_oldState = m_state;
|
|
m_state = NPC_CORAL_BLOWER_SUCK;
|
|
m_caughtPos = Pos;
|
|
|
|
break;
|
|
}
|
|
}
|
|
|
|
m_isBlowerOn = false;
|
|
} |