This commit is contained in:
parent
a3e34a14f6
commit
ea618523f8
461
source/enemy/nprojjf.cpp
Normal file
461
source/enemy/nprojjf.cpp
Normal file
@ -0,0 +1,461 @@
|
|||||||
|
/*=========================================================================
|
||||||
|
|
||||||
|
nprojjf.cpp
|
||||||
|
|
||||||
|
Author: CRB
|
||||||
|
Created:
|
||||||
|
Project: Spongebob
|
||||||
|
Purpose:
|
||||||
|
|
||||||
|
Copyright (c) 2000 Climax Development Ltd
|
||||||
|
|
||||||
|
===========================================================================*/
|
||||||
|
|
||||||
|
#ifndef __ENEMY_NPC_H__
|
||||||
|
#include "enemy\npc.h"
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifndef __ENEMY_NPROJJF_H__
|
||||||
|
#include "enemy\nprojjf.h"
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifndef __UTILS_HEADER__
|
||||||
|
#include "utils\utils.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
|
||||||
|
|
||||||
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
|
void CNpcSmallJellyfishProjectileEnemy::postInit()
|
||||||
|
{
|
||||||
|
CNpcEnemy::postInit();
|
||||||
|
|
||||||
|
m_lifetime = 8 * GameState::getOneSecondInFrames();
|
||||||
|
}
|
||||||
|
|
||||||
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
|
void CNpcSmallJellyfishProjectileEnemy::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_isCaught )
|
||||||
|
{
|
||||||
|
processCoralBlower( moveFrames );
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
if ( m_isActive )
|
||||||
|
{
|
||||||
|
if ( m_animPlaying )
|
||||||
|
{
|
||||||
|
s32 frameCount;
|
||||||
|
|
||||||
|
frameCount = getFrameCount();
|
||||||
|
|
||||||
|
s32 frameShift = ( moveFrames << 8 ) >> 1;
|
||||||
|
|
||||||
|
if ( ( frameCount << 8 ) - m_frame > frameShift )
|
||||||
|
{
|
||||||
|
m_frame += frameShift;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
m_frame = ( frameCount - 1 ) << 8;
|
||||||
|
m_animPlaying = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
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();
|
||||||
|
|
||||||
|
break;
|
||||||
|
|
||||||
|
case NPC_CONTROL_CLOSE:
|
||||||
|
processClose( moveFrames );
|
||||||
|
|
||||||
|
break;
|
||||||
|
|
||||||
|
case NPC_CONTROL_COLLISION:
|
||||||
|
processCollision();
|
||||||
|
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
processGraphicFlipping();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if ( !m_isCaught )
|
||||||
|
{
|
||||||
|
processTimer( moveFrames );
|
||||||
|
}
|
||||||
|
|
||||||
|
m_lifetime -= _frames;
|
||||||
|
|
||||||
|
if ( m_lifetime < 0 )
|
||||||
|
{
|
||||||
|
setToShutdown();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
|
void CNpcSmallJellyfishProjectileEnemy::render()
|
||||||
|
{
|
||||||
|
SprFrame = NULL;
|
||||||
|
|
||||||
|
if ( m_isActive )
|
||||||
|
{
|
||||||
|
CEnemyThing::render();
|
||||||
|
|
||||||
|
// Render
|
||||||
|
DVECTOR renderPos;
|
||||||
|
DVECTOR origRenderPos;
|
||||||
|
DVECTOR offset = CLevel::getCameraPos();
|
||||||
|
int spriteWidth = m_spriteBank->getFrameWidth( m_frame >> 8 );
|
||||||
|
int spriteHeight = m_spriteBank->getFrameHeight( m_frame >> 8 );
|
||||||
|
|
||||||
|
renderPos.vx = Pos.vx - offset.vx;
|
||||||
|
origRenderPos.vx = renderPos.vx;
|
||||||
|
renderPos.vx += m_drawOffset.vx;
|
||||||
|
|
||||||
|
renderPos.vy = Pos.vy - offset.vy;
|
||||||
|
origRenderPos.vy = renderPos.vy;
|
||||||
|
renderPos.vy += m_drawOffset.vy - ( spriteHeight >> 1 );
|
||||||
|
|
||||||
|
if ( renderPos.vx >= 0 && renderPos.vx <= VidGetScrW() )
|
||||||
|
{
|
||||||
|
if ( renderPos.vy >= 0 && renderPos.vy <= VidGetScrH() )
|
||||||
|
{
|
||||||
|
SprFrame = m_spriteBank->printRotatedScaledSprite( FRM_JELLYFISH1_SWIM1 + ( m_frame >> 8 ),renderPos.vx,renderPos.vy,4096,4096,m_drawRotation,10);
|
||||||
|
setRGB0( SprFrame, 255, 128, 255 ); // Let me know if these change! ALso ket me know when the different coloured ones go in pls! (pkg)
|
||||||
|
|
||||||
|
// get xmax, xmin, ymax, ymin
|
||||||
|
|
||||||
|
s32 XMax;
|
||||||
|
s32 XMin;
|
||||||
|
|
||||||
|
s32 YMax;
|
||||||
|
s32 YMin;
|
||||||
|
|
||||||
|
XMin=SprFrame->x0;
|
||||||
|
if (XMin>SprFrame->x1) XMin=SprFrame->x1;
|
||||||
|
if (XMin>SprFrame->x2) XMin=SprFrame->x2;
|
||||||
|
if (XMin>SprFrame->x3) XMin=SprFrame->x3;
|
||||||
|
XMax=SprFrame->x0;
|
||||||
|
if (XMax<SprFrame->x1) XMax=SprFrame->x1;
|
||||||
|
if (XMax<SprFrame->x2) XMax=SprFrame->x2;
|
||||||
|
if (XMax<SprFrame->x3) XMax=SprFrame->x3;
|
||||||
|
YMin=SprFrame->y0;
|
||||||
|
if (YMin>SprFrame->y1) YMin=SprFrame->y1;
|
||||||
|
if (YMin>SprFrame->y2) YMin=SprFrame->y2;
|
||||||
|
if (YMin>SprFrame->y3) YMin=SprFrame->y3;
|
||||||
|
YMax=SprFrame->y0;
|
||||||
|
if (YMax<SprFrame->y1) YMax=SprFrame->y1;
|
||||||
|
if (YMax<SprFrame->y2) YMax=SprFrame->y2;
|
||||||
|
if (YMax<SprFrame->y3) YMax=SprFrame->y3;
|
||||||
|
|
||||||
|
XMax -= origRenderPos.vx;
|
||||||
|
XMin -= origRenderPos.vx;
|
||||||
|
YMax -= origRenderPos.vy;
|
||||||
|
YMin -= origRenderPos.vy;
|
||||||
|
|
||||||
|
setCollisionSize( ( XMax - XMin ), ( YMax - YMin ) );
|
||||||
|
setCollisionCentreOffset( ( XMax + XMin ) >> 1, ( YMax + YMin ) >> 1 );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
|
void CNpcSmallJellyfishProjectileEnemy::processClose( int _frames )
|
||||||
|
{
|
||||||
|
s32 moveX = 0, moveY = 0;
|
||||||
|
|
||||||
|
s16 moveDist = 0;
|
||||||
|
|
||||||
|
s32 moveVel = 0;
|
||||||
|
|
||||||
|
if ( playerXDistSqr + playerYDistSqr > 10000 )
|
||||||
|
{
|
||||||
|
this->m_controlFunc = NPC_CONTROL_MOVEMENT;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
bool pathComplete;
|
||||||
|
bool waypointChange;
|
||||||
|
s32 xDist, yDist;
|
||||||
|
|
||||||
|
s16 headingToTarget = m_npcPath.think( Pos, &pathComplete, &waypointChange, &xDist, &yDist );
|
||||||
|
|
||||||
|
if ( pathComplete )
|
||||||
|
{
|
||||||
|
this->m_controlFunc = NPC_CONTROL_MOVEMENT;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
if ( !m_animPlaying )
|
||||||
|
{
|
||||||
|
m_animPlaying = true;
|
||||||
|
m_animNo = m_data[m_type].moveAnim;
|
||||||
|
m_frame = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
s16 headingToPlayer = ratan2( playerYDist, playerXDist );
|
||||||
|
|
||||||
|
if ( m_evadeClockwise )
|
||||||
|
{
|
||||||
|
headingToPlayer += 1024;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
headingToPlayer -= 1024;
|
||||||
|
}
|
||||||
|
|
||||||
|
headingToPlayer &= 4095;
|
||||||
|
|
||||||
|
s16 decDir, incDir;
|
||||||
|
|
||||||
|
decDir = headingToPlayer - headingToTarget;
|
||||||
|
|
||||||
|
if ( decDir < 0 )
|
||||||
|
{
|
||||||
|
decDir += ONE;
|
||||||
|
}
|
||||||
|
|
||||||
|
incDir = headingToTarget - headingToPlayer;
|
||||||
|
|
||||||
|
if ( incDir < 0 )
|
||||||
|
{
|
||||||
|
incDir += ONE;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ( decDir < incDir )
|
||||||
|
{
|
||||||
|
moveDist = decDir;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
moveDist = incDir;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ( moveDist < 128 )
|
||||||
|
{
|
||||||
|
// evasion angle is close to angle required to go to waypoint, hence resume waypoint movement
|
||||||
|
|
||||||
|
this->m_controlFunc = NPC_CONTROL_MOVEMENT;
|
||||||
|
this->m_timerFunc = NPC_TIMER_EVADE_DONE;
|
||||||
|
this->m_timerTimer = GameState::getOneSecondInFrames();
|
||||||
|
this->m_sensorFunc = NPC_SENSOR_NONE;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
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 < -128 )
|
||||||
|
{
|
||||||
|
moveDist = -128;
|
||||||
|
}
|
||||||
|
else if ( moveDist > 128 )
|
||||||
|
{
|
||||||
|
moveDist = 128;
|
||||||
|
}
|
||||||
|
|
||||||
|
m_heading += moveDist;
|
||||||
|
|
||||||
|
m_heading &= 4095;
|
||||||
|
|
||||||
|
moveX = ( _frames * 3 * rcos( m_heading ) ) >> 12;
|
||||||
|
moveY = ( _frames * 3 * rsin( m_heading ) ) >> 12;
|
||||||
|
moveVel = ( _frames * 3 ) << 8;
|
||||||
|
|
||||||
|
if ( processGroundCollisionReverse( &moveX, &moveY ) )
|
||||||
|
{
|
||||||
|
m_evadeClockwise = !m_evadeClockwise;
|
||||||
|
}
|
||||||
|
|
||||||
|
processMovementModifier(_frames, moveX, moveY, moveVel, moveDist);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
|
bool CNpcSmallJellyfishProjectileEnemy::processSensor()
|
||||||
|
{
|
||||||
|
switch( m_sensorFunc )
|
||||||
|
{
|
||||||
|
case NPC_SENSOR_NONE:
|
||||||
|
return( false );
|
||||||
|
|
||||||
|
default:
|
||||||
|
{
|
||||||
|
if ( playerXDistSqr + playerYDistSqr < 5625 )
|
||||||
|
{
|
||||||
|
m_controlFunc = NPC_CONTROL_CLOSE;
|
||||||
|
m_evadeClockwise = getRnd() % 2;
|
||||||
|
|
||||||
|
return( true );
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
return( false );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
|
void CNpcSmallJellyfishProjectileEnemy::processMovementModifier( int _frames, s32 distX, s32 distY, s32 dist, s16 headingChange )
|
||||||
|
{
|
||||||
|
s32 newX, newY;
|
||||||
|
s32 preShiftX, preShiftY;
|
||||||
|
s16 headingVal;
|
||||||
|
|
||||||
|
//u16 jellyfishData[5] = { 96, 192, 256, 192, 128, };
|
||||||
|
u16 jellyfishData[6] = { 256, 256, 96, 48, 96, 256, };
|
||||||
|
|
||||||
|
u32 dataPoint;
|
||||||
|
|
||||||
|
m_movementTimer += _frames;
|
||||||
|
|
||||||
|
if ( m_movementTimer > ( GameState::getOneSecondInFrames() >> 1 ) )
|
||||||
|
{
|
||||||
|
m_movementTimer = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
dataPoint = 5 * m_movementTimer;
|
||||||
|
|
||||||
|
if ( dataPoint != 0 )
|
||||||
|
{
|
||||||
|
dataPoint /= ( GameState::getOneSecondInFrames() >> 1 );
|
||||||
|
}
|
||||||
|
|
||||||
|
m_frame = ( ( m_movementTimer * ( getFrameCount() - 1 ) << 8 ) ) / GameState::getOneSecondInFrames();
|
||||||
|
|
||||||
|
s32 resistance;
|
||||||
|
s32 absVelocity = abs( m_velocity );
|
||||||
|
s32 reqVelocity = dist;
|
||||||
|
|
||||||
|
resistance = _frames * NPC_JELLYFISH_RESISTANCE;
|
||||||
|
|
||||||
|
if ( absVelocity < resistance )
|
||||||
|
{
|
||||||
|
resistance = absVelocity;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ( absVelocity != 0 )
|
||||||
|
{
|
||||||
|
resistance = -( resistance * m_velocity ) / absVelocity;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
resistance = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
m_velocity += resistance;
|
||||||
|
|
||||||
|
headingVal = abs( headingChange );
|
||||||
|
if ( headingVal > 128 )
|
||||||
|
{
|
||||||
|
headingVal = 128;
|
||||||
|
}
|
||||||
|
|
||||||
|
reqVelocity = dist * jellyfishData[dataPoint];
|
||||||
|
reqVelocity >>= 8;
|
||||||
|
reqVelocity *= 128 + ( 128 - headingVal );
|
||||||
|
reqVelocity >>= 8;
|
||||||
|
|
||||||
|
s32 absReqVelocity = abs( reqVelocity );
|
||||||
|
|
||||||
|
if ( absReqVelocity > absVelocity )
|
||||||
|
{
|
||||||
|
m_velocity += reqVelocity >> 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
preShiftX = ( m_velocity >> 8 ) * rcos( m_heading );
|
||||||
|
preShiftY = ( m_velocity >> 8 ) * rsin( m_heading );
|
||||||
|
|
||||||
|
newX = preShiftX >> 12;
|
||||||
|
if ( !newX && preShiftX )
|
||||||
|
{
|
||||||
|
newX = preShiftX / abs( preShiftX );
|
||||||
|
}
|
||||||
|
|
||||||
|
newY = preShiftY >> 12;
|
||||||
|
if ( !newY && preShiftY )
|
||||||
|
{
|
||||||
|
newY = preShiftY / abs( preShiftY );
|
||||||
|
}
|
||||||
|
|
||||||
|
Pos.vx += newX;
|
||||||
|
Pos.vy += newY;
|
||||||
|
|
||||||
|
m_drawRotation = m_heading;
|
||||||
|
}
|
46
source/enemy/nprojjf.h
Normal file
46
source/enemy/nprojjf.h
Normal file
@ -0,0 +1,46 @@
|
|||||||
|
/*=========================================================================
|
||||||
|
|
||||||
|
nprojjf.h
|
||||||
|
|
||||||
|
Author: CRB
|
||||||
|
Created:
|
||||||
|
Project: Spongebob
|
||||||
|
Purpose:
|
||||||
|
|
||||||
|
Copyright (c) 2000 Climax Development Ltd
|
||||||
|
|
||||||
|
===========================================================================*/
|
||||||
|
|
||||||
|
#ifndef __ENEMY_NPROJJF_H__
|
||||||
|
#define __ENEMY_NPROJJF_H__
|
||||||
|
|
||||||
|
#ifndef __ENEMY_NPC_H__
|
||||||
|
#include "enemy\npc.h"
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifndef __SPR_SPRITES_H__
|
||||||
|
#include <sprites.h>
|
||||||
|
#endif
|
||||||
|
|
||||||
|
class CNpcSmallJellyfishProjectileEnemy : public CNpcEnemy
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
void render();
|
||||||
|
virtual int getFrameCount() {return( FRM_JELLYFISH1_SWIM6 - FRM_JELLYFISH1_SWIM1 + 1 );}
|
||||||
|
virtual u8 canCollideWithEnemy() {return( false );}
|
||||||
|
virtual void postInit();
|
||||||
|
virtual void think(int _frames);
|
||||||
|
protected:
|
||||||
|
virtual bool processSensor();
|
||||||
|
virtual void processClose( int _frames );
|
||||||
|
virtual void processMovementModifier( int _frames, s32 distX, s32 distY, s32 dist, s16 headingChange );
|
||||||
|
|
||||||
|
enum
|
||||||
|
{
|
||||||
|
NPC_JELLYFISH_RESISTANCE = 64,
|
||||||
|
};
|
||||||
|
|
||||||
|
s32 m_lifetime;
|
||||||
|
};
|
||||||
|
|
||||||
|
#endif
|
Loading…
Reference in New Issue
Block a user