2001-05-24 16:06:10 +02:00
|
|
|
/*=========================================================================
|
|
|
|
|
|
|
|
nssnake.cpp
|
|
|
|
|
|
|
|
Author: CRB
|
|
|
|
Created:
|
|
|
|
Project: Spongebob
|
|
|
|
Purpose:
|
|
|
|
|
|
|
|
Copyright (c) 2001 Climax Development Ltd
|
|
|
|
|
|
|
|
===========================================================================*/
|
|
|
|
|
|
|
|
#ifndef __ENEMY_NPC_H__
|
|
|
|
#include "enemy\npc.h"
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifndef __ENEMY_NSSNAKE_H__
|
|
|
|
#include "enemy\nssnake.h"
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifndef __GAME_GAME_H__
|
|
|
|
#include "game\game.h"
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifndef __PLAYER_PLAYER_H__
|
|
|
|
#include "player\player.h"
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifndef __VID_HEADER_
|
|
|
|
#include "system\vid.h"
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef SHOW_BBOX
|
|
|
|
#include "gfx\prim.h"
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifndef __ANIM_SEASNAKE_HEADER__
|
|
|
|
#include <ACTOR_SEASNAKE_ANIM.h>
|
|
|
|
#endif
|
|
|
|
|
2001-06-16 20:25:07 +02:00
|
|
|
#ifndef __PROJECTL_PROJECTL_H__
|
|
|
|
#include "projectl\projectl.h"
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifndef __SPR_SPRITES_H__
|
|
|
|
#include <sprites.h>
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
2001-05-24 16:06:10 +02:00
|
|
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
void CNpcSeaSnakeSegment::init()
|
|
|
|
{
|
|
|
|
m_actorGfx=CActorPool::GetActor( (FileEquate) ACTORS_SEASNAKE_SBK );
|
|
|
|
|
|
|
|
m_heading = 0;
|
|
|
|
m_nextSegment = NULL;
|
|
|
|
|
|
|
|
setCollisionSize( 20, 20 );
|
|
|
|
setCollisionCentreOffset( 10, 10 );
|
|
|
|
updateCollisionArea();
|
|
|
|
}
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
void CNpcSeaSnakeSegment::updateCollisionArea()
|
|
|
|
{
|
|
|
|
m_collisionCentre.vx=Pos.vx+m_collisionCentreOffset.vx;
|
|
|
|
m_collisionCentre.vy=Pos.vy+m_collisionCentreOffset.vy;
|
|
|
|
m_collisionArea.x1=m_collisionCentre.vx-(m_collisionSize.vx/2);
|
|
|
|
m_collisionArea.x2=m_collisionArea.x1+m_collisionSize.vx;
|
|
|
|
m_collisionArea.y1=m_collisionCentre.vy-(m_collisionSize.vy/2);
|
|
|
|
m_collisionArea.y2=m_collisionArea.y1+m_collisionSize.vy;
|
|
|
|
}
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
void CNpcSeaSnakeSegment::setCollisionSize(int _w,int _h)
|
|
|
|
{
|
|
|
|
m_collisionSize.vx=_w;
|
|
|
|
m_collisionSize.vy=_h;
|
|
|
|
if(m_collisionSize.vx>m_collisionSize.vy)
|
|
|
|
{
|
|
|
|
m_collisionRadius=m_collisionSize.vx;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
m_collisionRadius=m_collisionSize.vy;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
void CNpcSeaSnakeEnemy::postInit()
|
|
|
|
{
|
|
|
|
m_npcPath.setPathType( CNpcPath::REPEATING_PATH );
|
|
|
|
|
2001-06-14 18:07:48 +02:00
|
|
|
s16 maxArraySize = NPC_SEA_SNAKE_LENGTH * NPC_SEA_SNAKE_SPACING;
|
2001-05-24 16:06:10 +02:00
|
|
|
|
2001-06-14 18:07:48 +02:00
|
|
|
m_positionHistoryArray[0].pos = Pos;
|
|
|
|
m_positionHistoryArray[0].next = &m_positionHistoryArray[1];
|
|
|
|
m_positionHistoryArray[0].prev = &m_positionHistoryArray[maxArraySize - 1];
|
2001-05-24 16:06:10 +02:00
|
|
|
|
2001-06-14 18:07:48 +02:00
|
|
|
for ( int histLength = 1 ; histLength < maxArraySize - 1 ; histLength++ )
|
2001-05-24 16:06:10 +02:00
|
|
|
{
|
2001-06-14 18:07:48 +02:00
|
|
|
m_positionHistoryArray[histLength].pos = Pos;
|
|
|
|
m_positionHistoryArray[histLength].next = &m_positionHistoryArray[histLength + 1];
|
|
|
|
m_positionHistoryArray[histLength].prev = &m_positionHistoryArray[histLength - 1];
|
2001-05-24 16:06:10 +02:00
|
|
|
}
|
|
|
|
|
2001-06-14 18:07:48 +02:00
|
|
|
m_positionHistoryArray[maxArraySize - 1].pos = Pos;
|
|
|
|
m_positionHistoryArray[maxArraySize - 1].next = &m_positionHistoryArray[0];
|
|
|
|
m_positionHistoryArray[maxArraySize - 1].prev = &m_positionHistoryArray[maxArraySize - 2];
|
2001-05-24 16:06:10 +02:00
|
|
|
|
2001-06-14 18:07:48 +02:00
|
|
|
m_positionHistory = &m_positionHistoryArray[0];
|
2001-05-24 16:06:10 +02:00
|
|
|
|
|
|
|
u16 segScale;
|
|
|
|
int initLength = NPC_SEA_SNAKE_LENGTH / 3;
|
|
|
|
int remLength = NPC_SEA_SNAKE_LENGTH - initLength;
|
|
|
|
|
2001-07-20 16:57:08 +02:00
|
|
|
m_health = NPC_SEA_SNAKE_LENGTH + 1;
|
2001-06-16 20:25:07 +02:00
|
|
|
|
|
|
|
if ( CLevel::getIsBossRespawn() )
|
|
|
|
{
|
|
|
|
m_health = CLevel::getBossHealth();
|
2001-07-20 16:57:08 +02:00
|
|
|
m_speed = m_data[m_type].speed + ( ( 3 * ( NPC_SEA_SNAKE_LENGTH - ( m_health - 1 ) ) ) / NPC_SEA_SNAKE_LENGTH );
|
2001-06-16 20:25:07 +02:00
|
|
|
}
|
|
|
|
|
2001-05-24 16:06:10 +02:00
|
|
|
for ( int segCount = 0 ; segCount < NPC_SEA_SNAKE_LENGTH ; segCount++ )
|
|
|
|
{
|
2001-06-14 18:07:48 +02:00
|
|
|
m_segmentArray[segCount].init();
|
2001-05-24 16:06:10 +02:00
|
|
|
|
|
|
|
if ( segCount < initLength )
|
|
|
|
{
|
2001-07-06 00:52:06 +02:00
|
|
|
u16 sum = ONE << 1;
|
|
|
|
u16 start = ONE;
|
2001-05-24 16:06:10 +02:00
|
|
|
u16 end = sum - start;
|
|
|
|
|
|
|
|
segScale = start + ( ( end * segCount ) / initLength );
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2001-07-06 00:52:06 +02:00
|
|
|
u16 sum = ONE << 1;
|
|
|
|
u16 start = ONE >> 1;
|
2001-05-24 16:06:10 +02:00
|
|
|
u16 end = sum - start;
|
|
|
|
|
|
|
|
segScale = start + ( ( end * ( NPC_SEA_SNAKE_LENGTH - segCount ) ) / remLength );
|
|
|
|
}
|
|
|
|
|
2001-06-14 18:07:48 +02:00
|
|
|
m_segmentArray[segCount].setScale( segScale );
|
2001-05-24 16:06:10 +02:00
|
|
|
|
|
|
|
// attach snake segment
|
|
|
|
|
2001-06-14 18:07:48 +02:00
|
|
|
if ( segCount < NPC_SEA_SNAKE_LENGTH - 1 )
|
2001-05-24 16:06:10 +02:00
|
|
|
{
|
2001-06-14 18:07:48 +02:00
|
|
|
m_segmentArray[segCount].m_nextSegment = &m_segmentArray[segCount + 1];
|
2001-05-24 16:06:10 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2001-07-20 16:57:08 +02:00
|
|
|
m_segmentCount = m_health - 1;
|
2001-06-14 18:07:48 +02:00
|
|
|
|
2001-05-24 16:06:10 +02:00
|
|
|
m_movementTimer = 2 * GameState::getOneSecondInFrames();
|
|
|
|
m_collTimer = 0;
|
2001-06-21 22:10:19 +02:00
|
|
|
m_turnDir = 0;
|
2001-06-25 22:44:53 +02:00
|
|
|
m_waitTimer = 0;
|
2001-06-27 20:28:21 +02:00
|
|
|
|
2001-06-30 17:39:43 +02:00
|
|
|
CNpcBossEnemy::postInit();
|
2001-05-24 16:06:10 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
void CNpcSeaSnakeSegment::shutdown()
|
|
|
|
{
|
|
|
|
delete m_actorGfx;
|
|
|
|
}
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
void CNpcSeaSnakeEnemy::shutdown()
|
|
|
|
{
|
2001-07-06 16:55:58 +02:00
|
|
|
CLevel::setIsBossRespawn( true );
|
|
|
|
|
2001-06-16 20:25:07 +02:00
|
|
|
if ( m_state != NPC_GENERIC_HIT_DEATH_END )
|
|
|
|
{
|
|
|
|
CLevel::setBossHealth( m_health );
|
|
|
|
}
|
2001-07-06 16:55:58 +02:00
|
|
|
else
|
|
|
|
{
|
2001-07-20 16:57:08 +02:00
|
|
|
CLevel::setBossHealth( 0 );
|
2001-07-06 16:55:58 +02:00
|
|
|
}
|
2001-06-16 20:25:07 +02:00
|
|
|
|
2001-05-24 16:06:10 +02:00
|
|
|
// delete snake segments
|
|
|
|
|
2001-06-14 18:07:48 +02:00
|
|
|
for ( int segCount = 0 ; segCount < NPC_SEA_SNAKE_LENGTH ; segCount++ )
|
2001-05-24 16:06:10 +02:00
|
|
|
{
|
2001-06-14 18:07:48 +02:00
|
|
|
m_segmentArray[segCount].shutdown();
|
2001-05-24 16:06:10 +02:00
|
|
|
}
|
|
|
|
|
2001-06-30 17:39:43 +02:00
|
|
|
CNpcBossEnemy::shutdown();
|
2001-05-24 16:06:10 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
bool CNpcSeaSnakeEnemy::processSensor()
|
|
|
|
{
|
2001-06-12 21:30:01 +02:00
|
|
|
if ( m_sensorFunc == NPC_SENSOR_NONE )
|
|
|
|
{
|
|
|
|
return( false );
|
|
|
|
}
|
|
|
|
|
2001-07-05 00:02:56 +02:00
|
|
|
if ( playerXDistSqr + playerYDistSqr < 50000 )
|
2001-05-24 16:06:10 +02:00
|
|
|
{
|
|
|
|
m_controlFunc = NPC_CONTROL_CLOSE;
|
2001-07-05 00:02:56 +02:00
|
|
|
m_state = SEA_SNAKE_VERTICAL_LINEUP;
|
2001-05-24 16:06:10 +02:00
|
|
|
|
|
|
|
return( true );
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
return( false );
|
2001-06-12 21:30:01 +02:00
|
|
|
}
|
2001-05-24 16:06:10 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
2001-06-27 22:50:28 +02:00
|
|
|
u8 CNpcSeaSnakeEnemy::processPathMove( int _frames, s32 *moveX, s32 *moveY, s32 *moveVel, s32 *moveDist )
|
|
|
|
{
|
|
|
|
bool pathComplete;
|
|
|
|
bool waypointChange;
|
|
|
|
s32 xDist, yDist;
|
|
|
|
|
|
|
|
s16 headingToTarget = m_npcPath.think( Pos, &pathComplete, &waypointChange, &xDist, &yDist );
|
|
|
|
|
|
|
|
/*if ( waypointChange )
|
|
|
|
{
|
|
|
|
m_movementTimer = 0;
|
|
|
|
}*/
|
|
|
|
|
|
|
|
if ( !pathComplete )
|
|
|
|
{
|
|
|
|
s16 decDir, incDir;
|
|
|
|
s16 maxTurnRate = m_data[m_type].turnSpeed;
|
|
|
|
|
|
|
|
decDir = m_heading - headingToTarget;
|
|
|
|
|
|
|
|
if ( decDir < 0 )
|
|
|
|
{
|
|
|
|
decDir += ONE;
|
|
|
|
}
|
|
|
|
|
|
|
|
incDir = headingToTarget - m_heading;
|
|
|
|
|
|
|
|
if ( incDir < 0 )
|
|
|
|
{
|
|
|
|
incDir += ONE;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( decDir < incDir )
|
|
|
|
{
|
|
|
|
*moveDist = -decDir;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
*moveDist = incDir;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( *moveDist < -maxTurnRate )
|
|
|
|
{
|
|
|
|
*moveDist = -maxTurnRate;
|
|
|
|
}
|
|
|
|
else if ( *moveDist > maxTurnRate )
|
|
|
|
{
|
|
|
|
*moveDist = maxTurnRate;
|
|
|
|
}
|
|
|
|
|
|
|
|
m_heading += *moveDist;
|
|
|
|
m_heading &= 4095;
|
|
|
|
|
|
|
|
s32 speed = m_speed;
|
|
|
|
|
|
|
|
if ( abs( *moveDist ) > 10 )
|
|
|
|
{
|
|
|
|
speed = 3;
|
|
|
|
}
|
|
|
|
|
|
|
|
s32 preShiftX = _frames * speed * rcos( m_heading );
|
|
|
|
s32 preShiftY = _frames * speed * rsin( m_heading );
|
|
|
|
|
|
|
|
*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;
|
|
|
|
}
|
|
|
|
|
|
|
|
*moveVel = ( _frames * m_speed ) << 8;
|
|
|
|
|
|
|
|
//processGroundCollisionReverse( moveX, moveY );
|
|
|
|
}
|
|
|
|
|
|
|
|
return( waypointChange );
|
|
|
|
}
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
2001-05-24 16:06:10 +02:00
|
|
|
void CNpcSeaSnakeEnemy::processMovement( int _frames )
|
|
|
|
{
|
|
|
|
s32 moveX = 0, moveY = 0;
|
|
|
|
s32 moveVel = 0;
|
|
|
|
s32 moveDist = 0;
|
|
|
|
DVECTOR oldPos = Pos;
|
|
|
|
|
2001-06-12 21:30:01 +02:00
|
|
|
if ( m_snapTimer > 0 )
|
|
|
|
{
|
|
|
|
m_snapTimer -= _frames;
|
|
|
|
|
2001-06-19 23:43:18 +02:00
|
|
|
if ( m_snapTimer > 0 )
|
2001-06-12 21:30:01 +02:00
|
|
|
{
|
2001-06-19 23:43:18 +02:00
|
|
|
if ( !m_animPlaying )
|
2001-06-12 21:30:01 +02:00
|
|
|
{
|
2001-06-19 23:43:18 +02:00
|
|
|
m_animNo = ANIM_SEASNAKE_HEADSNAP;
|
2001-06-12 21:30:01 +02:00
|
|
|
m_animPlaying = true;
|
|
|
|
m_frame = 0;
|
2001-07-14 00:11:49 +02:00
|
|
|
|
|
|
|
CSoundMediator::playSfx( CSoundMediator::SFX_WORM___CHOMP );
|
2001-06-12 21:30:01 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2001-06-01 18:30:37 +02:00
|
|
|
if ( m_data[m_type].moveSfx < CSoundMediator::NUM_SFXIDS )
|
|
|
|
{
|
2001-06-18 21:06:43 +02:00
|
|
|
if ( m_soundId == NOT_PLAYING )
|
|
|
|
{
|
|
|
|
m_soundId = (int) CSoundMediator::playSfx( m_data[m_type].moveSfx, true );
|
|
|
|
}
|
2001-06-01 18:30:37 +02:00
|
|
|
}
|
|
|
|
|
2001-06-25 22:44:53 +02:00
|
|
|
if ( m_waitTimer > 0 )
|
2001-06-21 22:10:19 +02:00
|
|
|
{
|
2001-06-25 22:44:53 +02:00
|
|
|
m_waitTimer -= _frames;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
switch( m_turnDir )
|
2001-06-21 22:10:19 +02:00
|
|
|
{
|
2001-06-25 22:44:53 +02:00
|
|
|
case NPC_SEA_SNAKE_CIRCLE_CLOCKWISE:
|
2001-06-21 22:10:19 +02:00
|
|
|
{
|
2001-06-25 22:44:53 +02:00
|
|
|
m_circleHeading += m_data[m_type].turnSpeed;
|
2001-06-21 22:10:19 +02:00
|
|
|
|
2001-06-25 22:44:53 +02:00
|
|
|
if ( m_circleHeading > 4096 )
|
|
|
|
{
|
|
|
|
m_circleHeading = 0;
|
|
|
|
m_turnDir = 0;
|
|
|
|
}
|
2001-06-21 22:10:19 +02:00
|
|
|
|
2001-06-25 22:44:53 +02:00
|
|
|
m_heading = ( m_origHeading + m_circleHeading ) & 4095;
|
2001-06-21 22:10:19 +02:00
|
|
|
|
2001-06-27 22:50:28 +02:00
|
|
|
s32 preShiftX = _frames * 3 * rcos( m_heading );
|
|
|
|
s32 preShiftY = _frames * 3 * rsin( m_heading );
|
2001-06-21 22:10:19 +02:00
|
|
|
|
2001-06-25 22:44:53 +02:00
|
|
|
s32 moveX = preShiftX >> 12;
|
|
|
|
if ( !moveX && preShiftX )
|
|
|
|
{
|
|
|
|
moveX = preShiftX / abs( preShiftX );
|
|
|
|
}
|
2001-06-21 22:10:19 +02:00
|
|
|
|
2001-06-25 22:44:53 +02:00
|
|
|
s32 moveY = preShiftY >> 12;
|
|
|
|
if ( !moveY && preShiftY )
|
|
|
|
{
|
|
|
|
moveY = preShiftY / abs( preShiftY );
|
|
|
|
}
|
2001-06-21 22:10:19 +02:00
|
|
|
|
2001-06-25 22:44:53 +02:00
|
|
|
Pos.vx += moveX;
|
|
|
|
Pos.vy += moveY;
|
2001-06-21 22:10:19 +02:00
|
|
|
|
2001-06-25 22:44:53 +02:00
|
|
|
break;
|
|
|
|
}
|
2001-06-21 22:10:19 +02:00
|
|
|
|
2001-06-25 22:44:53 +02:00
|
|
|
case NPC_SEA_SNAKE_CIRCLE_ANTICLOCKWISE:
|
2001-06-21 22:10:19 +02:00
|
|
|
{
|
2001-06-25 22:44:53 +02:00
|
|
|
m_circleHeading -= m_data[m_type].turnSpeed;
|
2001-06-21 22:10:19 +02:00
|
|
|
|
2001-06-25 22:44:53 +02:00
|
|
|
if ( m_circleHeading < -4096 )
|
|
|
|
{
|
|
|
|
m_circleHeading = 0;
|
|
|
|
m_turnDir = 0;
|
|
|
|
}
|
2001-06-21 22:10:19 +02:00
|
|
|
|
2001-06-25 22:44:53 +02:00
|
|
|
m_heading = ( m_origHeading + m_circleHeading ) & 4095;
|
2001-06-21 22:10:19 +02:00
|
|
|
|
2001-06-27 22:50:28 +02:00
|
|
|
s32 preShiftX = _frames * 3 * rcos( m_heading );
|
|
|
|
s32 preShiftY = _frames * 3 * rsin( m_heading );
|
2001-06-21 22:10:19 +02:00
|
|
|
|
2001-06-25 22:44:53 +02:00
|
|
|
s32 moveX = preShiftX >> 12;
|
|
|
|
if ( !moveX && preShiftX )
|
|
|
|
{
|
|
|
|
moveX = preShiftX / abs( preShiftX );
|
|
|
|
}
|
2001-06-21 22:10:19 +02:00
|
|
|
|
2001-06-25 22:44:53 +02:00
|
|
|
s32 moveY = preShiftY >> 12;
|
|
|
|
if ( !moveY && preShiftY )
|
|
|
|
{
|
|
|
|
moveY = preShiftY / abs( preShiftY );
|
|
|
|
}
|
2001-06-21 22:10:19 +02:00
|
|
|
|
2001-06-25 22:44:53 +02:00
|
|
|
Pos.vx += moveX;
|
|
|
|
Pos.vy += moveY;
|
2001-06-21 22:10:19 +02:00
|
|
|
|
2001-06-25 22:44:53 +02:00
|
|
|
break;
|
|
|
|
}
|
2001-06-21 22:10:19 +02:00
|
|
|
|
2001-06-25 22:44:53 +02:00
|
|
|
default:
|
2001-06-21 22:10:19 +02:00
|
|
|
{
|
2001-06-25 22:44:53 +02:00
|
|
|
DVECTOR waypointPos;
|
|
|
|
m_npcPath.getCurrentWaypointPos( &waypointPos );
|
|
|
|
waypointPos.vy -= 8;
|
|
|
|
|
|
|
|
if ( CGameScene::getCollision()->getHeightFromGround( waypointPos.vx, waypointPos.vy ) < 0 )
|
|
|
|
{
|
|
|
|
// waypoint is either start or end waypoint
|
2001-06-21 22:10:19 +02:00
|
|
|
|
2001-06-25 22:44:53 +02:00
|
|
|
s32 distX, distY;
|
2001-06-21 22:10:19 +02:00
|
|
|
|
2001-06-25 22:44:53 +02:00
|
|
|
distX = waypointPos.vx - Pos.vx;
|
|
|
|
distY = waypointPos.vy - Pos.vy;
|
2001-06-21 22:10:19 +02:00
|
|
|
|
2001-06-25 22:44:53 +02:00
|
|
|
if( !distX && !distY )
|
2001-06-21 22:10:19 +02:00
|
|
|
{
|
2001-06-25 22:44:53 +02:00
|
|
|
if ( isSnakeStopped() )
|
|
|
|
{
|
|
|
|
m_npcPath.incPath();
|
2001-06-21 22:10:19 +02:00
|
|
|
|
2001-06-25 22:44:53 +02:00
|
|
|
m_npcPath.getCurrentWaypointPos( &waypointPos );
|
|
|
|
waypointPos.vy -= 8;
|
2001-06-21 22:10:19 +02:00
|
|
|
|
2001-06-25 22:44:53 +02:00
|
|
|
if ( CGameScene::getCollision()->getHeightFromGround( waypointPos.vx, waypointPos.vy ) < 0 )
|
|
|
|
{
|
|
|
|
// if next waypoint is ALSO a start/end waypoint, teleport directly to it
|
2001-06-21 22:10:19 +02:00
|
|
|
|
2001-06-25 22:44:53 +02:00
|
|
|
moveEntireSnake( waypointPos );
|
2001-07-02 21:21:09 +02:00
|
|
|
m_waitTimer = GameState::getOneSecondInFrames();
|
2001-06-25 22:44:53 +02:00
|
|
|
oldPos.vx = waypointPos.vx;
|
|
|
|
oldPos.vy = waypointPos.vy;
|
2001-06-21 22:10:19 +02:00
|
|
|
|
2001-06-25 22:44:53 +02:00
|
|
|
// increment path
|
|
|
|
m_npcPath.incPath();
|
2001-06-21 22:10:19 +02:00
|
|
|
|
2001-06-25 22:44:53 +02:00
|
|
|
// point snake in correct direction
|
|
|
|
m_npcPath.getCurrentWaypointPos( &waypointPos );
|
2001-06-21 22:10:19 +02:00
|
|
|
|
2001-06-25 22:44:53 +02:00
|
|
|
m_heading = ratan2( waypointPos.vy - Pos.vy, waypointPos.vx - Pos.vx ) & 4095;
|
|
|
|
}
|
2001-06-21 22:10:19 +02:00
|
|
|
}
|
|
|
|
}
|
2001-06-25 22:44:53 +02:00
|
|
|
else
|
|
|
|
{
|
|
|
|
processGenericGotoTarget( _frames, distX, distY, m_speed );
|
|
|
|
}
|
2001-06-21 22:10:19 +02:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2001-06-27 22:50:28 +02:00
|
|
|
if ( processPathMove( _frames, &moveX, &moveY, &moveVel, &moveDist ) )
|
2001-06-21 22:10:19 +02:00
|
|
|
{
|
2001-06-25 22:44:53 +02:00
|
|
|
// path has changed
|
2001-06-21 22:10:19 +02:00
|
|
|
|
2001-06-25 22:44:53 +02:00
|
|
|
DVECTOR newWaypointPos;
|
2001-06-21 22:10:19 +02:00
|
|
|
|
2001-06-25 22:44:53 +02:00
|
|
|
m_npcPath.getCurrentWaypointPos( &newWaypointPos );
|
|
|
|
newWaypointPos.vy -= 8;
|
2001-06-21 22:10:19 +02:00
|
|
|
|
2001-06-25 22:44:53 +02:00
|
|
|
if ( newWaypointPos.vy == waypointPos.vy )
|
|
|
|
{
|
|
|
|
int testDir = newWaypointPos.vx - waypointPos.vx;
|
|
|
|
|
|
|
|
if ( testDir > 0 && testDir <= 16 )
|
|
|
|
{
|
|
|
|
// clockwise
|
|
|
|
|
|
|
|
m_turnDir = NPC_SEA_SNAKE_CIRCLE_CLOCKWISE;
|
|
|
|
m_circleHeading = 0;
|
|
|
|
m_origHeading = m_heading;
|
|
|
|
m_npcPath.incPath();
|
|
|
|
}
|
|
|
|
else if ( testDir < 0 && testDir >= -16 )
|
|
|
|
{
|
|
|
|
// anticlockwise
|
|
|
|
|
|
|
|
m_turnDir = NPC_SEA_SNAKE_CIRCLE_ANTICLOCKWISE;
|
|
|
|
m_circleHeading = 0;
|
|
|
|
m_origHeading = m_heading;
|
|
|
|
m_npcPath.incPath();
|
|
|
|
}
|
2001-06-21 22:10:19 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2001-06-25 22:44:53 +02:00
|
|
|
break;
|
|
|
|
}
|
2001-06-21 22:10:19 +02:00
|
|
|
}
|
|
|
|
}
|
2001-05-24 16:06:10 +02:00
|
|
|
|
|
|
|
Pos.vx += moveX;
|
|
|
|
Pos.vy += moveY;
|
|
|
|
|
2001-07-17 17:40:12 +02:00
|
|
|
// check for hitting ground
|
|
|
|
|
|
|
|
if ( CGameScene::getCollision()->Get( Pos.vx >> 4, Pos.vy >> 4 ) )
|
|
|
|
{
|
|
|
|
switch ( CGameScene::getCollision()->getCollisionBlock( Pos.vx, Pos.vy ) & COLLISION_TYPE_MASK )
|
|
|
|
{
|
|
|
|
case COLLISION_TYPE_FLAG_SOLID:
|
|
|
|
{
|
|
|
|
Pos = oldPos;
|
2001-07-20 17:52:22 +02:00
|
|
|
m_heading += 2048;
|
2001-07-17 17:40:12 +02:00
|
|
|
m_heading &= 4095;
|
|
|
|
|
2001-07-20 17:52:22 +02:00
|
|
|
m_npcPath.decPath();
|
|
|
|
|
|
|
|
DVECTOR waypointPos;
|
|
|
|
m_npcPath.getCurrentWaypointPos( &waypointPos );
|
|
|
|
waypointPos.vy -= 8;
|
|
|
|
|
|
|
|
if ( CGameScene::getCollision()->getHeightFromGround( waypointPos.vx, waypointPos.vy ) < 0 )
|
|
|
|
{
|
|
|
|
// one of the special 'teleport' waypoints
|
|
|
|
|
|
|
|
m_npcPath.incPath();
|
|
|
|
}
|
|
|
|
|
2001-07-17 17:40:12 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2001-07-05 00:02:56 +02:00
|
|
|
updateTail( oldPos, _frames );
|
|
|
|
}
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
void CNpcSeaSnakeEnemy::updateTail( DVECTOR &oldPos, int _frames )
|
|
|
|
{
|
|
|
|
u8 skipCounter;
|
|
|
|
|
2001-05-24 16:06:10 +02:00
|
|
|
m_extension += 256;
|
|
|
|
m_extension &= 4095;
|
|
|
|
|
|
|
|
m_positionHistory = m_positionHistory->prev;
|
|
|
|
m_positionHistory->pos = oldPos;
|
|
|
|
|
|
|
|
CNpcPositionHistory *newPos;
|
|
|
|
newPos = m_positionHistory;
|
|
|
|
for ( skipCounter = 1 ; skipCounter < NPC_SEA_SNAKE_SPACING ; skipCounter++ )
|
|
|
|
{
|
|
|
|
newPos = newPos->next;
|
|
|
|
}
|
|
|
|
|
|
|
|
oldPos = Pos;
|
|
|
|
|
|
|
|
s32 extension = m_extension;
|
|
|
|
u8 downShift = 2;
|
|
|
|
u8 timeShift;
|
|
|
|
|
|
|
|
if ( m_movementTimer > 0 )
|
|
|
|
{
|
|
|
|
m_movementTimer -= _frames;
|
|
|
|
|
|
|
|
if ( m_movementTimer < 0 )
|
|
|
|
{
|
|
|
|
m_movementTimer = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
timeShift = m_movementTimer / GameState::getOneSecondInFrames();
|
|
|
|
|
2001-06-14 18:07:48 +02:00
|
|
|
int segmentCount;
|
|
|
|
|
|
|
|
for ( segmentCount = 0 ; segmentCount < m_segmentCount ; segmentCount++ )
|
2001-05-24 16:06:10 +02:00
|
|
|
{
|
|
|
|
s32 xDist = oldPos.vx - newPos->pos.vx;
|
|
|
|
s32 yDist = oldPos.vy - newPos->pos.vy;
|
|
|
|
|
|
|
|
s16 headingToTarget = ratan2( yDist, xDist );
|
|
|
|
|
|
|
|
DVECTOR sinPos;
|
|
|
|
|
|
|
|
sinPos = newPos->pos;
|
|
|
|
s32 diff = ( ( ( 5 >> downShift ) * rsin( extension ) ) >> 12 ) >> timeShift;
|
|
|
|
sinPos.vx += ( diff * rcos( headingToTarget + 1024 ) ) >> 12;
|
|
|
|
sinPos.vy += ( diff * rsin( headingToTarget + 1024 ) ) >> 12;
|
|
|
|
|
2001-06-14 18:07:48 +02:00
|
|
|
m_segmentArray[segmentCount].setPos( sinPos );
|
2001-06-20 16:21:09 +02:00
|
|
|
|
|
|
|
if ( segmentCount > 3 )
|
|
|
|
{
|
|
|
|
if ( segmentCount == m_segmentCount - 1 )
|
|
|
|
{
|
|
|
|
m_segmentArray[segmentCount].setAnim( ANIM_SEASNAKE_TAILSTATIC );
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if ( segmentCount % 2 )
|
|
|
|
{
|
|
|
|
m_segmentArray[segmentCount].setAnim( ANIM_SEASNAKE_BODY2STATIC );
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
m_segmentArray[segmentCount].setAnim( ANIM_SEASNAKE_BODY3STATIC );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
m_segmentArray[segmentCount].setAnim( ANIM_SEASNAKE_BODY1STATIC );
|
|
|
|
}
|
2001-05-24 16:06:10 +02:00
|
|
|
oldPos = sinPos;
|
|
|
|
|
2001-06-14 18:07:48 +02:00
|
|
|
for ( skipCounter = 0 ; skipCounter < NPC_SEA_SNAKE_SPACING ; skipCounter++ )
|
2001-05-24 16:06:10 +02:00
|
|
|
{
|
2001-06-14 18:07:48 +02:00
|
|
|
newPos = newPos->next;
|
2001-05-24 16:06:10 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
extension += 1024;
|
|
|
|
extension &= 4095;
|
|
|
|
|
|
|
|
if ( downShift > 0 )
|
|
|
|
{
|
|
|
|
downShift--;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
oldPos = Pos;
|
|
|
|
|
2001-06-14 18:07:48 +02:00
|
|
|
for ( segmentCount = 0 ; segmentCount < m_segmentCount ; segmentCount++ )
|
2001-05-24 16:06:10 +02:00
|
|
|
{
|
2001-07-23 21:26:37 +02:00
|
|
|
DVECTOR const ¤tPos = m_segmentArray[segmentCount].getPos();
|
2001-05-24 16:06:10 +02:00
|
|
|
|
|
|
|
s32 xDist = oldPos.vx - currentPos.vx;
|
|
|
|
s32 yDist = oldPos.vy - currentPos.vy;
|
|
|
|
|
|
|
|
s16 headingToPrev = ratan2( yDist, xDist );
|
|
|
|
s16 headingFromNext;
|
|
|
|
s16 heading = headingToPrev;
|
|
|
|
|
|
|
|
oldPos = currentPos;
|
|
|
|
|
2001-06-14 18:07:48 +02:00
|
|
|
if ( segmentCount < m_segmentCount - 1 )
|
2001-05-24 16:06:10 +02:00
|
|
|
{
|
2001-07-23 21:26:37 +02:00
|
|
|
DVECTOR const &nextPos = m_segmentArray[segmentCount + 1].getPos();
|
2001-05-24 16:06:10 +02:00
|
|
|
xDist = currentPos.vx - nextPos.vx;
|
|
|
|
yDist = currentPos.vy - nextPos.vy;
|
|
|
|
headingFromNext = ratan2( yDist, xDist );
|
|
|
|
|
|
|
|
s16 decDir, incDir, moveDist;
|
|
|
|
|
|
|
|
decDir = headingFromNext - headingToPrev;
|
|
|
|
|
|
|
|
if ( decDir < 0 )
|
|
|
|
{
|
|
|
|
decDir += ONE;
|
|
|
|
}
|
|
|
|
|
|
|
|
incDir = headingToPrev - headingFromNext;
|
|
|
|
|
|
|
|
if ( incDir < 0 )
|
|
|
|
{
|
|
|
|
incDir += ONE;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( decDir < incDir )
|
|
|
|
{
|
|
|
|
moveDist = -decDir;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
moveDist = incDir;
|
|
|
|
}
|
|
|
|
|
|
|
|
heading -= moveDist >> 1;
|
|
|
|
}
|
|
|
|
|
2001-06-14 18:07:48 +02:00
|
|
|
m_segmentArray[segmentCount].setHeading( heading );
|
|
|
|
m_segmentArray[segmentCount].updateCollisionArea();
|
2001-05-24 16:06:10 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if ( m_collTimer > 0 )
|
|
|
|
{
|
|
|
|
m_collTimer -= _frames;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
/*void CNpcSeaSnakeEnemy::resetSeaSnakeHeadToTail()
|
|
|
|
{
|
|
|
|
DVECTOR startPos;
|
|
|
|
DVECTOR endPos;
|
|
|
|
int posCounter;
|
|
|
|
CNpcPositionHistory *currentPos;
|
|
|
|
|
|
|
|
startPos = Pos;
|
|
|
|
|
|
|
|
currentPos = m_positionHistory;
|
|
|
|
|
|
|
|
for ( posCounter = 0 ; posCounter < ( NPC_SEA_SNAKE_LENGTH * NPC_SEA_SNAKE_SPACING ) - 1 ; posCounter++ )
|
|
|
|
{
|
|
|
|
currentPos = currentPos->next;
|
|
|
|
}
|
|
|
|
|
|
|
|
endPos = currentPos->pos;
|
|
|
|
|
|
|
|
currentPos = m_positionHistory;
|
|
|
|
|
|
|
|
for ( posCounter = 0 ; posCounter < NPC_SEA_SNAKE_LENGTH * NPC_SEA_SNAKE_SPACING ; posCounter++ )
|
|
|
|
{
|
|
|
|
currentPos->pos.vx = startPos.vx + ( posCounter * ( endPos.vx - startPos.vx ) ) / ( ( NPC_SEA_SNAKE_LENGTH * NPC_SEA_SNAKE_SPACING ) - 1 );
|
|
|
|
currentPos->pos.vy = startPos.vy + ( posCounter * ( endPos.vy - startPos.vy ) ) / ( ( NPC_SEA_SNAKE_LENGTH * NPC_SEA_SNAKE_SPACING ) - 1 );
|
|
|
|
|
|
|
|
currentPos = currentPos->next;
|
|
|
|
}
|
|
|
|
|
|
|
|
CNpcPositionHistory *newPos;
|
|
|
|
newPos = m_positionHistory;
|
|
|
|
|
|
|
|
u8 skipCounter;
|
|
|
|
for ( skipCounter = 1 ; skipCounter < NPC_SEA_SNAKE_SPACING ; skipCounter++ )
|
|
|
|
{
|
|
|
|
newPos = newPos->next;
|
|
|
|
}
|
|
|
|
|
|
|
|
CThing *List=Next;
|
|
|
|
|
|
|
|
DVECTOR oldPos = Pos;
|
|
|
|
|
|
|
|
s32 extension = m_extension;
|
|
|
|
|
|
|
|
while( List )
|
|
|
|
{
|
|
|
|
CNpcEnemy *segment = (CNpcEnemy *) List;
|
|
|
|
|
|
|
|
s32 xDist = oldPos.vx - newPos->pos.vx;
|
|
|
|
s32 yDist = oldPos.vy - newPos->pos.vy;
|
|
|
|
|
|
|
|
s16 headingToTarget = ratan2( yDist, xDist );
|
|
|
|
|
|
|
|
segment->setHeading( headingToTarget );
|
|
|
|
|
|
|
|
List->setPos( newPos->pos );
|
|
|
|
oldPos = newPos->pos;
|
|
|
|
|
|
|
|
List = List->getNext();
|
|
|
|
|
|
|
|
if ( List )
|
|
|
|
{
|
|
|
|
for ( skipCounter = 0 ; skipCounter < NPC_SEA_SNAKE_SPACING ; skipCounter++ )
|
|
|
|
{
|
|
|
|
newPos = newPos->next;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
extension += 1024;
|
|
|
|
extension &= 4095;
|
|
|
|
}
|
|
|
|
}*/
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
void CNpcSeaSnakeEnemy::processClose( int _frames )
|
|
|
|
{
|
2001-07-05 00:02:56 +02:00
|
|
|
DVECTOR oldPos = Pos;
|
|
|
|
|
|
|
|
switch( m_state )
|
|
|
|
{
|
|
|
|
case SEA_SNAKE_VERTICAL_LINEUP:
|
|
|
|
{
|
|
|
|
if ( playerYDistSqr > 100 )
|
|
|
|
{
|
|
|
|
processGenericGotoTarget( _frames, 0, playerYDist, m_speed );
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
m_state = SEA_SNAKE_CHARGE;
|
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case SEA_SNAKE_CHARGE:
|
|
|
|
{
|
|
|
|
if ( playerXDistSqr > 4000 )
|
|
|
|
{
|
|
|
|
processGenericGotoTarget( _frames, playerXDist, 0, m_speed );
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
s16 heading = m_heading;
|
|
|
|
|
|
|
|
CProjectile *projectile;
|
|
|
|
projectile = CProjectile::Create();
|
|
|
|
DVECTOR newPos = Pos;
|
|
|
|
newPos.vx += 50 * ( rcos( m_heading ) >> 12 );
|
|
|
|
newPos.vy += 50 * ( rsin( m_heading ) >> 12 );
|
|
|
|
|
|
|
|
int perpHeading = ( heading - 1024 ) & 4095;
|
|
|
|
|
|
|
|
newPos.vx += 20 * ( rcos( perpHeading ) >> 12 );
|
|
|
|
newPos.vy += 20 * ( rsin( perpHeading ) >> 12 );
|
|
|
|
|
|
|
|
projectile->init( newPos, heading );
|
|
|
|
projectile->setGraphic( FRM__SNAKEBILE );
|
|
|
|
|
|
|
|
m_movementTimer = GameState::getOneSecondInFrames();
|
|
|
|
|
|
|
|
m_controlFunc = NPC_CONTROL_MOVEMENT;
|
|
|
|
m_timerFunc = NPC_TIMER_ATTACK_DONE;
|
|
|
|
m_timerTimer = 5 * GameState::getOneSecondInFrames();
|
|
|
|
m_sensorFunc = NPC_SENSOR_NONE;
|
|
|
|
|
|
|
|
m_snapTimer = m_movementTimer;
|
2001-07-14 00:11:49 +02:00
|
|
|
|
|
|
|
CSoundMediator::playSfx( CSoundMediator::SFX_WORM___HISS );
|
2001-07-05 00:02:56 +02:00
|
|
|
}
|
|
|
|
|
2001-07-12 23:22:26 +02:00
|
|
|
if ( m_soundId == NOT_PLAYING )
|
|
|
|
{
|
|
|
|
m_soundId = (int) CSoundMediator::playSfx( CSoundMediator::SFX_SEASNAKE_ATTACK, true );
|
|
|
|
}
|
|
|
|
|
2001-07-05 00:02:56 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2001-07-17 17:40:12 +02:00
|
|
|
// check for hitting ground
|
|
|
|
|
|
|
|
if ( CGameScene::getCollision()->Get( Pos.vx >> 4, Pos.vy >> 4 ) )
|
|
|
|
{
|
|
|
|
switch ( CGameScene::getCollision()->getCollisionBlock( Pos.vx, Pos.vy ) & COLLISION_TYPE_MASK )
|
|
|
|
{
|
|
|
|
case COLLISION_TYPE_FLAG_SOLID:
|
|
|
|
{
|
|
|
|
m_movementTimer = GameState::getOneSecondInFrames();
|
|
|
|
|
|
|
|
m_controlFunc = NPC_CONTROL_MOVEMENT;
|
|
|
|
m_timerFunc = NPC_TIMER_ATTACK_DONE;
|
|
|
|
m_timerTimer = 5 * GameState::getOneSecondInFrames();
|
|
|
|
m_sensorFunc = NPC_SENSOR_NONE;
|
|
|
|
|
|
|
|
Pos = oldPos;
|
2001-07-20 17:52:22 +02:00
|
|
|
m_heading += 2048;
|
2001-07-17 17:40:12 +02:00
|
|
|
m_heading &= 4095;
|
|
|
|
|
2001-07-20 17:52:22 +02:00
|
|
|
m_npcPath.decPath();
|
|
|
|
|
|
|
|
DVECTOR waypointPos;
|
|
|
|
m_npcPath.getCurrentWaypointPos( &waypointPos );
|
|
|
|
waypointPos.vy -= 8;
|
|
|
|
|
|
|
|
if ( CGameScene::getCollision()->getHeightFromGround( waypointPos.vx, waypointPos.vy ) < 0 )
|
|
|
|
{
|
|
|
|
// one of the special 'teleport' waypoints
|
|
|
|
|
|
|
|
m_npcPath.incPath();
|
|
|
|
}
|
|
|
|
|
2001-07-17 17:40:12 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2001-07-05 00:02:56 +02:00
|
|
|
updateTail( oldPos, _frames );
|
|
|
|
|
|
|
|
/*
|
2001-06-16 20:25:07 +02:00
|
|
|
// fire at player
|
|
|
|
|
2001-06-26 22:57:58 +02:00
|
|
|
//s16 heading = ratan2( playerYDist, playerXDist ) & 4095;
|
|
|
|
s16 heading = m_heading;
|
2001-06-16 20:25:07 +02:00
|
|
|
|
|
|
|
CProjectile *projectile;
|
|
|
|
projectile = CProjectile::Create();
|
|
|
|
DVECTOR newPos = Pos;
|
2001-07-02 21:21:09 +02:00
|
|
|
newPos.vx += 50 * ( rcos( m_heading ) >> 12 );
|
|
|
|
newPos.vy += 50 * ( rsin( m_heading ) >> 12 );
|
|
|
|
|
|
|
|
int perpHeading = ( heading - 1024 ) & 4095;
|
|
|
|
|
|
|
|
newPos.vx += 20 * ( rcos( perpHeading ) >> 12 );
|
|
|
|
newPos.vy += 20 * ( rsin( perpHeading ) >> 12 );
|
|
|
|
|
2001-06-16 20:25:07 +02:00
|
|
|
projectile->init( newPos, heading );
|
2001-06-20 15:24:04 +02:00
|
|
|
projectile->setGraphic( FRM__SNAKEBILE );
|
2001-06-16 20:25:07 +02:00
|
|
|
|
2001-05-24 16:06:10 +02:00
|
|
|
//resetSeaSnakeHeadToTail();
|
|
|
|
|
2001-07-02 21:21:09 +02:00
|
|
|
m_movementTimer = GameState::getOneSecondInFrames();
|
2001-05-24 16:06:10 +02:00
|
|
|
|
|
|
|
m_controlFunc = NPC_CONTROL_MOVEMENT;
|
|
|
|
m_timerFunc = NPC_TIMER_ATTACK_DONE;
|
|
|
|
m_timerTimer = GameState::getOneSecondInFrames();
|
|
|
|
m_sensorFunc = NPC_SENSOR_NONE;
|
2001-06-12 21:30:01 +02:00
|
|
|
|
|
|
|
m_snapTimer = m_movementTimer;
|
2001-06-19 23:43:18 +02:00
|
|
|
//m_openTimer = GameState::getOneSecondInFrames() >> 2;
|
2001-07-05 00:02:56 +02:00
|
|
|
*/
|
2001-05-24 16:06:10 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
void CNpcSeaSnakeEnemy::processEnemyCollision( CThing *thisThing )
|
|
|
|
{
|
|
|
|
// do nothing
|
|
|
|
}
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
void CNpcSeaSnakeSegment::processEnemyCollision( CThing *thisThing )
|
|
|
|
{
|
|
|
|
// do nothing
|
|
|
|
}
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
void CNpcSeaSnakeEnemy::render()
|
|
|
|
{
|
|
|
|
SprFrame = NULL;
|
|
|
|
|
|
|
|
if ( m_isActive )
|
|
|
|
{
|
|
|
|
CEnemyThing::render();
|
|
|
|
|
|
|
|
if (canRender())
|
|
|
|
{
|
|
|
|
DVECTOR &renderPos=getRenderPos();
|
|
|
|
|
2001-05-24 23:25:45 +02:00
|
|
|
SprFrame = m_actorGfx->Render(renderPos,m_animNo,( m_frame >> 8 ),0);
|
2001-05-24 16:06:10 +02:00
|
|
|
m_actorGfx->RotateScale( SprFrame, renderPos, m_heading, 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 );
|
|
|
|
}
|
|
|
|
|
2001-06-14 18:07:48 +02:00
|
|
|
for ( int segmentCount = 0 ; segmentCount < m_segmentCount ; segmentCount++ )
|
2001-05-24 16:06:10 +02:00
|
|
|
{
|
2001-06-14 18:07:48 +02:00
|
|
|
m_segmentArray[segmentCount].render();
|
2001-05-24 16:06:10 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
int CNpcSeaSnakeEnemy::checkCollisionAgainst( CThing *_thisThing, int _frames )
|
|
|
|
{
|
2001-07-23 21:26:37 +02:00
|
|
|
// DVECTOR pos,thisThingPos;
|
2001-05-24 16:06:10 +02:00
|
|
|
int radius;
|
|
|
|
int collided;
|
|
|
|
|
2001-07-23 21:26:37 +02:00
|
|
|
DVECTOR const &pos=getCollisionCentre();
|
|
|
|
DVECTOR const &thisThingPos=_thisThing->getCollisionCentre();
|
2001-05-24 16:06:10 +02:00
|
|
|
|
|
|
|
radius=getCollisionRadius()+_thisThing->getCollisionRadius();
|
|
|
|
collided=false;
|
|
|
|
if(abs(pos.vx-thisThingPos.vx)<radius&&
|
|
|
|
abs(pos.vy-thisThingPos.vy)<radius)
|
|
|
|
{
|
2001-07-23 21:26:37 +02:00
|
|
|
// CRECT thisRect,thatRect;
|
2001-05-24 16:06:10 +02:00
|
|
|
|
2001-07-23 21:26:37 +02:00
|
|
|
CRECT const &thisRect=getCollisionArea();
|
|
|
|
CRECT const &thatRect=_thisThing->getCollisionArea();
|
2001-05-24 16:06:10 +02:00
|
|
|
|
|
|
|
if(((thisRect.x1>=thatRect.x1&&thisRect.x1<=thatRect.x2)||(thisRect.x2>=thatRect.x1&&thisRect.x2<=thatRect.x2)||(thisRect.x1<=thatRect.x1&&thisRect.x2>=thatRect.x2))&&
|
|
|
|
((thisRect.y1>=thatRect.y1&&thisRect.y1<=thatRect.y2)||(thisRect.y2>=thatRect.y1&&thisRect.y2<=thatRect.y2)||(thisRect.y1<=thatRect.y1&&thisRect.y2>=thatRect.y2)))
|
|
|
|
{
|
|
|
|
collided=true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// go through segments
|
|
|
|
|
2001-06-14 18:07:48 +02:00
|
|
|
for ( int segmentCount = 0 ; segmentCount < m_segmentCount ; segmentCount++ )
|
2001-05-24 16:06:10 +02:00
|
|
|
{
|
2001-06-14 18:07:48 +02:00
|
|
|
if ( m_segmentArray[segmentCount].checkCollisionAgainst( _thisThing, _frames ) )
|
2001-05-24 16:06:10 +02:00
|
|
|
{
|
|
|
|
collided = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return collided;
|
|
|
|
}
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
void CNpcSeaSnakeSegment::render()
|
|
|
|
{
|
|
|
|
POLY_FT4 *SprFrame = NULL;
|
|
|
|
|
|
|
|
sBBox &ScrBBox=CThingManager::getRenderBBox();
|
|
|
|
DVECTOR const &CamPos=CLevel::getCameraPos();
|
|
|
|
|
|
|
|
DVECTOR renderPos;
|
|
|
|
renderPos.vx = Pos.vx - CamPos.vx;
|
|
|
|
renderPos.vy = Pos.vy - CamPos.vy;
|
|
|
|
|
|
|
|
u8 renderFlag = true;
|
|
|
|
if ( m_collisionArea.x2 < ScrBBox.XMin || m_collisionArea.x1 > ScrBBox.XMax ) renderFlag=false;
|
|
|
|
if ( m_collisionArea.y2 < ScrBBox.YMin || m_collisionArea.y1 > ScrBBox.YMax ) renderFlag=false;
|
|
|
|
|
|
|
|
if ( renderFlag )
|
|
|
|
{
|
2001-06-20 16:21:09 +02:00
|
|
|
SprFrame = m_actorGfx->Render(renderPos,m_anim,0,0);
|
2001-05-24 16:06:10 +02:00
|
|
|
m_actorGfx->RotateScale( SprFrame, renderPos, m_heading, 4096, m_scale );
|
|
|
|
|
|
|
|
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 );
|
|
|
|
|
|
|
|
#if defined (__USER_charles__)
|
|
|
|
DVECTOR const &ofs=CLevel::getCameraPos();
|
|
|
|
CRECT area;
|
|
|
|
|
|
|
|
area=getCollisionArea();
|
|
|
|
area.x1-=ofs.vx;
|
|
|
|
area.y1-=ofs.vy;
|
|
|
|
area.x2-=ofs.vx;
|
|
|
|
area.y2-=ofs.vy;
|
|
|
|
|
|
|
|
if(area.x1<=511&&area.x2>=0 && area.y1<=255&&area.y2>=0)
|
|
|
|
{
|
|
|
|
DrawLine(area.x1,area.y1,area.x2,area.y1,255,255,255,0);
|
|
|
|
DrawLine(area.x2,area.y1,area.x2,area.y2,255,255,255,0);
|
|
|
|
DrawLine(area.x2,area.y2,area.x1,area.y2,255,255,255,0);
|
|
|
|
DrawLine(area.x1,area.y2,area.x1,area.y1,255,255,255,0);
|
|
|
|
|
|
|
|
area.x1=Pos.vx-10-ofs.vx;
|
|
|
|
area.y1=Pos.vy-10-ofs.vy;
|
|
|
|
area.x2=Pos.vx+10-ofs.vx;
|
|
|
|
area.y2=Pos.vy+10-ofs.vy;
|
|
|
|
DrawLine(area.x1,area.y1,area.x2,area.y2,255,0,0,0);
|
|
|
|
DrawLine(area.x2,area.y1,area.x1,area.y2,255,0,0,0);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
int CNpcSeaSnakeSegment::checkCollisionAgainst( CThing *_thisThing, int _frames )
|
|
|
|
{
|
2001-07-23 21:26:37 +02:00
|
|
|
// DVECTOR pos,thisThingPos;
|
2001-05-24 16:06:10 +02:00
|
|
|
int radius;
|
|
|
|
int collided;
|
|
|
|
|
2001-07-23 21:26:37 +02:00
|
|
|
DVECTOR const &pos = getCollisionCentre();
|
|
|
|
DVECTOR const &thisThingPos = _thisThing->getCollisionCentre();
|
2001-05-24 16:06:10 +02:00
|
|
|
|
|
|
|
radius = getCollisionRadius() + _thisThing->getCollisionRadius();
|
|
|
|
collided = false;
|
|
|
|
if(abs(pos.vx-thisThingPos.vx)<radius&&
|
|
|
|
abs(pos.vy-thisThingPos.vy)<radius)
|
|
|
|
{
|
2001-07-23 21:26:37 +02:00
|
|
|
// CRECT thisRect,thatRect;
|
2001-05-24 16:06:10 +02:00
|
|
|
|
2001-07-23 21:26:37 +02:00
|
|
|
CRECT const &thisRect=getCollisionArea();
|
|
|
|
CRECT const &thatRect=_thisThing->getCollisionArea();
|
2001-05-24 16:06:10 +02:00
|
|
|
|
|
|
|
if(((thisRect.x1>=thatRect.x1&&thisRect.x1<=thatRect.x2)||(thisRect.x2>=thatRect.x1&&thisRect.x2<=thatRect.x2)||(thisRect.x1<=thatRect.x1&&thisRect.x2>=thatRect.x2))&&
|
|
|
|
((thisRect.y1>=thatRect.y1&&thisRect.y1<=thatRect.y2)||(thisRect.y2>=thatRect.y1&&thisRect.y2<=thatRect.y2)||(thisRect.y1<=thatRect.y1&&thisRect.y2>=thatRect.y2)))
|
|
|
|
{
|
|
|
|
collided=true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return collided;
|
|
|
|
}
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
void CNpcSeaSnakeEnemy::processShot( int _frames )
|
|
|
|
{
|
2001-06-14 18:07:48 +02:00
|
|
|
if ( !m_segmentCount )
|
2001-05-24 16:06:10 +02:00
|
|
|
{
|
2001-07-16 16:19:10 +02:00
|
|
|
if ( m_collTimer <= 0 )
|
|
|
|
{
|
|
|
|
m_drawRotation += 64 * _frames;
|
|
|
|
m_drawRotation &= 4095;
|
2001-05-24 16:06:10 +02:00
|
|
|
|
2001-07-16 16:19:10 +02:00
|
|
|
Pos.vy += m_speed * _frames;
|
2001-05-24 16:06:10 +02:00
|
|
|
|
2001-07-16 16:19:10 +02:00
|
|
|
if ( m_speed < 5 )
|
|
|
|
{
|
|
|
|
m_speed++;
|
|
|
|
}
|
2001-05-24 16:06:10 +02:00
|
|
|
|
2001-07-16 16:19:10 +02:00
|
|
|
m_state = NPC_GENERIC_HIT_DEATH_END;
|
2001-06-19 00:10:57 +02:00
|
|
|
|
2001-07-23 21:26:37 +02:00
|
|
|
DVECTOR const &offset = CLevel::getCameraPos();
|
2001-05-24 16:06:10 +02:00
|
|
|
|
2001-07-16 16:19:10 +02:00
|
|
|
if ( Pos.vy - offset.vy > VidGetScrH() )
|
|
|
|
{
|
|
|
|
setToShutdown();
|
|
|
|
CGameScene::setBossHasBeenKilled();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
2001-05-24 16:06:10 +02:00
|
|
|
{
|
2001-07-16 16:19:10 +02:00
|
|
|
m_controlFunc = NPC_CONTROL_MOVEMENT;
|
2001-05-24 16:06:10 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if ( m_collTimer <= 0 )
|
|
|
|
{
|
|
|
|
// knock segment off end of list
|
|
|
|
|
2001-06-14 18:07:48 +02:00
|
|
|
m_segmentCount--;
|
2001-06-16 20:25:07 +02:00
|
|
|
m_health--;
|
2001-07-20 16:57:08 +02:00
|
|
|
m_speed = m_data[m_type].speed + ( ( 3 * ( NPC_SEA_SNAKE_LENGTH - ( m_health - 1 ) ) ) / NPC_SEA_SNAKE_LENGTH );
|
2001-05-24 16:06:10 +02:00
|
|
|
|
|
|
|
m_collTimer = GameState::getOneSecondInFrames();
|
|
|
|
}
|
|
|
|
|
|
|
|
m_controlFunc = NPC_CONTROL_MOVEMENT;
|
|
|
|
}
|
|
|
|
}
|
2001-05-25 18:27:06 +02:00
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
2001-06-12 21:30:01 +02:00
|
|
|
void CNpcSeaSnakeEnemy::processUserCollision( CThing *thisThing )
|
|
|
|
{
|
2001-06-19 23:43:18 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
s32 CNpcSeaSnakeEnemy::getFrameShift( int _frames )
|
|
|
|
{
|
|
|
|
return( ( _frames << 8 ) >> 3 );
|
|
|
|
}
|
2001-06-21 22:10:19 +02:00
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
u8 CNpcSeaSnakeEnemy::isSnakeStopped()
|
|
|
|
{
|
|
|
|
if ( !m_segmentCount )
|
|
|
|
{
|
|
|
|
return( true );
|
|
|
|
}
|
|
|
|
|
2001-07-23 21:26:37 +02:00
|
|
|
DVECTOR const &tailPos = m_segmentArray[m_segmentCount - 1].getPos();
|
2001-06-21 22:10:19 +02:00
|
|
|
|
|
|
|
if ( tailPos.vx == Pos.vx && tailPos.vy == Pos.vy )
|
|
|
|
{
|
|
|
|
return( true );
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
return( false );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
2001-07-23 21:26:37 +02:00
|
|
|
void CNpcSeaSnakeEnemy::moveEntireSnake( DVECTOR const &newPos )
|
2001-06-21 22:10:19 +02:00
|
|
|
{
|
|
|
|
Pos.vx = newPos.vx;
|
|
|
|
Pos.vy = newPos.vy;
|
|
|
|
|
|
|
|
int segmentCount;
|
|
|
|
|
|
|
|
for ( segmentCount = 0 ; segmentCount < m_segmentCount ; segmentCount++ )
|
|
|
|
{
|
|
|
|
m_segmentArray[segmentCount].setPos( Pos );
|
|
|
|
}
|
|
|
|
|
|
|
|
s16 maxArraySize = NPC_SEA_SNAKE_LENGTH * NPC_SEA_SNAKE_SPACING;
|
|
|
|
|
|
|
|
for ( int histLength = 0 ; histLength < maxArraySize ; histLength++ )
|
|
|
|
{
|
|
|
|
m_positionHistoryArray[histLength].pos = Pos;
|
|
|
|
}
|
|
|
|
}
|
2001-06-30 17:39:43 +02:00
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
void CNpcSeaSnakeEnemy::addHealthMeter()
|
|
|
|
{
|
|
|
|
if (!m_meterOn)
|
|
|
|
{
|
|
|
|
m_energyBar=(CFXNRGBar*)CFX::Create(CFX::FX_TYPE_NRG_BAR,this);
|
|
|
|
m_energyBar->SetMax( NPC_SEA_SNAKE_LENGTH + 1 );
|
|
|
|
m_meterOn=true;
|
|
|
|
}
|
|
|
|
}
|