/*========================================================================= player.cpp Author: PKG Created: Project: Spongebob Purpose: Copyright (c) 2001 Climax Development Ltd ===========================================================================*/ /*---------------------------------------------------------------------- Includes -------- */ #include "player\player.h" #ifndef __ANIM_HEADER__ #include "gfx\anim.h" #endif #ifndef __PAD_PADS_H__ #include "pad\pads.h" #endif #ifndef __GAME_GAMESLOT_H__ #include "game\gameslot.h" #endif #ifndef __GAME_GAME_H__ #include "game\game.h" #endif #ifndef __LAYER_COLLISION_H__ #include "level\collision.h" #endif // to be removed #include "gfx\tpage.h" /* Std Lib ------- */ /* Data ---- */ #ifndef __ANIM_PLAYER_ANIM_HEADER__ #include #endif /*---------------------------------------------------------------------- Tyepdefs && Defines ------------------- */ //#define _RECORD_DEMO_MODE_ /*---------------------------------------------------------------------- Structure defintions -------------------- */ /*---------------------------------------------------------------------- Function Prototypes ------------------- */ /*---------------------------------------------------------------------- Vars ---- */ #ifdef _RECORD_DEMO_MODE_ #include "player\demoplay.h" #define MAX_DEMO_SIZE 512 // So max size of a demo is 1k #define MAX_DEMO_TIME_IN_FRAMES 30*60 // Recorded demo will last 60 seconds static CDemoPlayer::demoPlayerControl s_demoControls[MAX_DEMO_SIZE]={{PI_NONE,0}}; static int s_demoSize=0; static int s_demoFrameCount=0; static void writeDemoControls() { char filename[32]; int fh; int fc=MAX_DEMO_TIME_IN_FRAMES; sprintf(filename,"demo____.dmo"); fh=PCcreat((char *)filename,0); ASSERT(fh!=-1); PCwrite(fh,(char*)&fc,sizeof(fc)); // frame count PCwrite(fh,(char*)&s_demoSize,sizeof(s_demoSize)); // demo size for(int i=0;i=3)_frames=2; for(i=0;i<_frames;i++) { // Think updatePadInput(); m_currentStateClass->think(this); // Horizontal movement if(m_layerCollision->Get((Pos.vx+(m_moveVel.vx>>VELOCITY_SHIFT))>>4,(Pos.vy-1)>>4)) { // Will hit a wall this frame - Do collision // Move flush with the edge of the obstruction int dir,vx,cx,y,i; if(m_moveVel.vx<0) { dir=-1; vx=-m_moveVel.vx; } else { dir=+1; vx=m_moveVel.vx; } cx=Pos.vx; y=(Pos.vy-1)>>4; for(i=0;iGet(cx>>4,y)) { break; } cx+=dir; } Pos.vx=cx-dir; // If running then go to idle, otherwise leave in same state if(m_currentState==STATE_RUN) { setState(STATE_IDLE); } m_moveVel.vx=0; } else { // No obstruction this frame - Do the movement Pos.vx+=m_moveVel.vx>>VELOCITY_SHIFT; } // Vertical movement Pos.vy+=m_moveVel.vy>>VELOCITY_SHIFT; if(isOnSolidGround()) { //stick to ground (PKG) //Pos.vy=23*16+1;//16*15; int colHeight=16; Pos.vy=((Pos.vy-16)&0xfffffff0)+colHeight; if(m_moveVel.vy) { // Was falling.. so we've just hit the ground if(m_currentState==STATE_BUTTFALL) { // landed from a btt bounce setState(STATE_BUTTLAND); } else if(m_currentState==STATE_FALLFAR) { // Landed from a painful long fall setState(STATE_IDLE); takeDamage(DAMAGE__FALL); m_moveVel.vx=0; CSoundMediator::playSfx(CSoundMediator::SFX_SPONGEBOB_LAND_AFTER_FALL); } else if(m_moveVel.vx) { // Landed from a jump with x movement setState(STATE_RUN); } else { // Landed from a standing jump setState(STATE_IDLE); setAnimNo(ANIM_PLAYER_ANIM_JUMPEND); } m_moveVel.vy=0; m_fallFrames=0; } } else { if(m_currentState!=STATE_FALL&&m_currentState!=STATE_FALLFAR&& m_currentState!=STATE_BUTTFALL&&m_currentState!=STATE_BUTTBOUNCE&& m_currentState!=STATE_JUMP) { setState(STATE_FALL); } } // Flashing.. if(m_invincibleFrameCount) { m_invincibleFrameCount--; } // Teeter if on an edge if(m_currentState==STATE_IDLE&&isOnEdge()) { setState(STATE_IDLETEETER); } // Look around int pad=getPadInputHeld(); if(getPadInputDown()&PAD_CIRCLE) { m_skel.blink(); } // Camera scroll.. if(m_cameraScrollDir==-1) { if(m_cameraScrollPos>-CAMERA_SCROLLLIMIT<<8) { m_cameraScrollPos-=CAMERA_SCROLLSPEED; if(m_cameraScrollPos<-CAMERA_SCROLLLIMIT<<8) { m_cameraScrollPos=-CAMERA_SCROLLLIMIT<<8; m_cameraScrollDir=0; } } } else if(m_cameraScrollDir==+1) { if(m_cameraScrollPos<(CAMERA_SCROLLLIMIT<<8)) { m_cameraScrollPos+=CAMERA_SCROLLSPEED; if(m_cameraScrollPos>CAMERA_SCROLLLIMIT<<8) { m_cameraScrollPos=CAMERA_SCROLLLIMIT<<8; m_cameraScrollDir=0; } } } /* if(pad&CPadConfig::getButton(CPadConfig::PAD_CFG_UP)) { if(m_cameraLookTimer<=-LOOKAROUND_DELAY) { m_cameraLookYOffset-=LOOKAROUND_SCROLLSPEED; if(m_cameraLookYOffset<-LOOKAROUND_MAXSCROLL) { m_cameraLookYOffset=-LOOKAROUND_MAXSCROLL; } } else { m_cameraLookTimer--; } } else if(pad&CPadConfig::getButton(CPadConfig::PAD_CFG_DOWN)) { if(m_cameraLookTimer>=LOOKAROUND_DELAY) { m_cameraLookYOffset+=LOOKAROUND_SCROLLSPEED; if(m_cameraLookYOffset>LOOKAROUND_MAXSCROLL) { m_cameraLookYOffset=LOOKAROUND_MAXSCROLL; } } else { m_cameraLookTimer++; } } else { m_cameraLookTimer=0; if(m_cameraLookYOffset<0) { m_cameraLookYOffset+=LOOKAROUND_RESETSPEED; if(m_cameraLookYOffset>0) { m_cameraLookYOffset=0; } } else if(m_cameraLookYOffset>0) { m_cameraLookYOffset-=LOOKAROUND_RESETSPEED; if(m_cameraLookYOffset<0) { m_cameraLookYOffset=0; } } } */ } #endif // Move the camera offset ppos.vx=MAP3D_CENTRE_X+((MAP3D_BLOCKSTEPSIZE*m_cameraScrollPos)>>8); ppos.vy=MAP3D_CENTRE_Y; ofs.vx=MAP2D_CENTRE_X+((MAP2D_BLOCKSTEPSIZE*(-m_cameraScrollPos))>>8); ofs.vy=MAP2D_CENTRE_Y; m_cameraOffset=ofs; if(Pos.vx<0)Pos.vx=0; if(Pos.vy<0)Pos.vy=0; } /*---------------------------------------------------------------------- Function: Purpose: Params: Returns: ---------------------------------------------------------------------- */ int panim=-1; #ifdef __USER_paul__ int mouth=-1,eyes=-1; #endif void CPlayer::render() { CThing::render(); // Render if(m_invincibleFrameCount==0||m_invincibleFrameCount&2) { #ifdef __USER_paul__ if(mouth!=-1) { m_skel.setMouthTex(mouth); mouth=-1; } if(eyes!=-1) { m_skel.setEyeTex(eyes); eyes=-1; } #endif //int xval=255-(MAP2D_BLOCKSTEPSIZE*(-m_cameraScrollPos>>8)); //DrawLine(xval-7,0,xval-7,255,0,128,255,0); //DrawLine(xval+7,0,xval+7,255,0,128,255,0); m_skel.setPos(ppos); if(panim!=-1) m_skel.setAnimNo(panim); else m_skel.setAnimNo(m_animNo); m_skel.setFrame(m_animFrame); m_skel.Animate(this); m_skel.Render(this); } } /*---------------------------------------------------------------------- Function: Purpose: Params: Returns: ---------------------------------------------------------------------- */ DVECTOR CPlayer::getCameraPos() { DVECTOR cameraPos; cameraPos.vx=Pos.vx+m_cameraOffset.vx; cameraPos.vy=Pos.vy+m_cameraOffset.vy; return cameraPos; } /*---------------------------------------------------------------------- Function: Purpose: Params: Returns: ---------------------------------------------------------------------- */ const PlayerMetrics *CPlayer::getPlayerMetrics() { return &s_modes[m_currentMode].m_metrics; } /*---------------------------------------------------------------------- Function: Purpose: Params: Returns: ---------------------------------------------------------------------- */ int CPlayer::setState(PLAYER_STATE _state) { CPlayerState *nextState; int ret=false; nextState=s_modes[m_currentMode].m_states[_state]; if(nextState) { m_currentStateClass=nextState; m_currentStateClass->enter(this); m_currentState=_state; ret=true; } return ret; } /*---------------------------------------------------------------------- Function: Purpose: Params: Returns: ---------------------------------------------------------------------- */ void CPlayer::setMode(PLAYER_MODE _mode) { m_currentMode=_mode; // Need to do something about this setState() for when the new mode doesn't have that state (pkg) setState(m_currentState); } /*---------------------------------------------------------------------- Function: Purpose: Params: Returns: ---------------------------------------------------------------------- */ int CPlayer::getFacing() { return m_facing; } void CPlayer::setFacing(int _facing) { if(m_facing!=_facing) { m_facing=_facing; m_skel.setDir(_facing); } } /*---------------------------------------------------------------------- Function: Purpose: Params: Returns: ---------------------------------------------------------------------- */ int CPlayer::getAnimFrame() { return m_animFrame; } void CPlayer::setAnimFrame(int _animFrame) { const AnimSfx *sfx; m_animFrame=_animFrame; // Are there any sfx for this frame? sfx=&s_animSfx[m_animNo]; if(sfx->m_numAnimFrameSfx) { const AnimFrameSfx *frameSfx; int i; frameSfx=sfx->m_animFrameSfx; for(i=0;im_numAnimFrameSfx;i++) { if(m_animFrame==frameSfx->m_frame) { CSoundMediator::SFXID sfxId=frameSfx->m_sfxId; if(m_currentMode==PLAYER_MODE_SQUEAKYBOOTS) { // Ugh.. horrible way to change the sfx when wearing squeaky boots (pkg) if(sfxId==CSoundMediator::SFX_SPONGEBOB_WALK_1)sfxId=CSoundMediator::SFX_SPONGEBOB_SQUEAKY_SHOES_1; else if(sfxId==CSoundMediator::SFX_SPONGEBOB_WALK_2)sfxId=CSoundMediator::SFX_SPONGEBOB_SQUEAKY_SHOES_2; } CSoundMediator::playSfx(sfxId); } if(m_animFramem_frame) { break; } frameSfx++; } } } int CPlayer::getAnimFrameCount() { return m_skel.getFrameCount(); } int CPlayer::getAnimNo() { return m_animNo; } void CPlayer::setAnimNo(int _animNo) { m_animNo=_animNo; setAnimFrame(0); } /*---------------------------------------------------------------------- Function: Purpose: Params: Returns: ---------------------------------------------------------------------- */ DVECTOR CPlayer::getMoveVelocity() { return m_moveVel; } void CPlayer::setMoveVelocity(DVECTOR *_moveVel) { m_moveVel=*_moveVel; } DVECTOR CPlayer::getPlayerPos() { return Pos; } /*---------------------------------------------------------------------- Function: Purpose: Params: Returns: ---------------------------------------------------------------------- */ PLAYERINPUT CPlayer::getPadInputHeld() { return m_padInput; } PLAYERINPUT CPlayer::getPadInputDown() { return m_padInputDown; } /*---------------------------------------------------------------------- Function: Purpose: Params: Returns: ---------------------------------------------------------------------- */ int CPlayer::isOnSolidGround() { ASSERT(m_layerCollision); return m_layerCollision->Get(Pos.vx>>4,Pos.vy>>4); } /*---------------------------------------------------------------------- Function: Purpose: Params: Returns: FACING_LEFT if left half of player is hanging, FACING_RIGHT if right half of player is hanging or 0 if no part of the player is hanging ---------------------------------------------------------------------- */ int csize=20; int CPlayer::isOnEdge() { int ret=0; ASSERT(m_layerCollision); if(!m_layerCollision->Get((Pos.vx-csize)>>4,Pos.vy>>4)) { ret=FACING_LEFT; } else if(!m_layerCollision->Get((Pos.vx+csize)>>4,Pos.vy>>4)) { ret=FACING_RIGHT; } return ret; } /*---------------------------------------------------------------------- Function: Purpose: Params: Returns: ---------------------------------------------------------------------- */ int CPlayer::canMoveLeft() { ASSERT(m_layerCollision); return m_layerCollision->Get((Pos.vx-1)>>4,(Pos.vy-1)>>4)==0; } int CPlayer::canMoveRight() { ASSERT(m_layerCollision); return m_layerCollision->Get((Pos.vx+1)>>4,(Pos.vy-1)>>4)==0; } /*---------------------------------------------------------------------- Function: Purpose: Params: Returns: ---------------------------------------------------------------------- */ void CPlayer::moveLeft() { const PlayerMetrics *metrics; metrics=getPlayerMetrics(); setFacing(FACING_LEFT); if(m_moveVel.vx<=0) { m_moveVel.vx-=metrics->m_metric[PM__RUN_SPEEDUP]; if(m_moveVel.vx<-metrics->m_metric[PM__MAX_RUN_VELOCITY]<m_metric[PM__MAX_RUN_VELOCITY]<m_metric[PM__RUN_REVERSESLOWDOWN]; } if(m_moveVel.vx<-CAMERA_STARTMOVETHRESHOLD||m_cameraScrollPos<-CAMERA_SCROLLTHRESHOLD<<8) { m_cameraScrollDir=+1; } else if(m_moveVel.vx>-CAMERA_STOPMOVETHRESHOLD) { m_cameraScrollDir=0; } } void CPlayer::moveRight() { const PlayerMetrics *metrics; metrics=getPlayerMetrics(); setFacing(FACING_RIGHT); if(m_moveVel.vx>=0) { m_moveVel.vx+=metrics->m_metric[PM__RUN_SPEEDUP]; if(m_moveVel.vx>metrics->m_metric[PM__MAX_RUN_VELOCITY]<m_metric[PM__MAX_RUN_VELOCITY]<m_metric[PM__RUN_REVERSESLOWDOWN]; } if(m_moveVel.vx>CAMERA_STARTMOVETHRESHOLD||m_cameraScrollPos>CAMERA_SCROLLTHRESHOLD<<8) { m_cameraScrollDir=-1; } else if(m_moveVel.vxm_metric[PM__RUN_SLOWDOWN]; if(m_moveVel.vx>0)m_moveVel.vx=0; } else if(m_moveVel.vx>0) { m_moveVel.vx-=metrics->m_metric[PM__RUN_SLOWDOWN]; if(m_moveVel.vx<0)m_moveVel.vx=0; } } void CPlayer::jump() { } void CPlayer::fall() { const PlayerMetrics *metrics; metrics=getPlayerMetrics(); m_moveVel.vy+=PLAYER_GRAVITY; if(m_moveVel.vy>=PLAYER_TERMINAL_VELOCITY<metrics->m_metric[PM__MAX_SAFE_FALL_FRAMES]) { setState(STATE_FALLFAR); } } } } /*---------------------------------------------------------------------- Function: Purpose: Params: Returns: ---------------------------------------------------------------------- */ #ifdef __VERSION_DEBUG__ int invincibleSponge=false; // NB: This is for debugging purposes only #endif void CPlayer::takeDamage(DAMAGE_TYPE _damage) { // Don't take damage if still recovering from the last hit if(!m_invincibleFrameCount) { int ouchThatHurt=true; // Check if we are currently immune to this damage type switch(_damage) { case DAMAGE__FALL: case DAMAGE__LAVA: break; case DAMAGE__ELECTROCUTION: // if squeaky boots then ouchThatHurt=false; break; } if(ouchThatHurt) { #ifdef __VERSION_DEBUG__ if(invincibleSponge){m_invincibleFrameCount=INVINCIBLE_FRAMES__HIT;return;} #endif if(s_health) { m_invincibleFrameCount=INVINCIBLE_FRAMES__HIT; s_health--; } else { CSoundMediator::playSfx(CSoundMediator::SFX_SPONGEBOB_DEFEATED_JINGLE); setState(STATE_DEAD); } } } } /*---------------------------------------------------------------------- Function: Purpose: Params: Returns: ---------------------------------------------------------------------- */ void CPlayer::updatePadInput() { m_lastPadInput=m_padInput; m_padInput=readPadInput(); m_padInputDown=(PLAYERINPUT)(m_padInput&(m_lastPadInput^-1)); } /*---------------------------------------------------------------------- Function: Purpose: Params: Returns: ---------------------------------------------------------------------- */ PLAYERINPUT CPlayer::readPadInput() { PLAYERINPUT input; int pad; input=PI_NONE; pad=PadGetHeld(0); if(pad&CPadConfig::getButton(CPadConfig::PAD_CFG_UP)) { input=(PLAYERINPUT)(input|PI_UP); } if(pad&CPadConfig::getButton(CPadConfig::PAD_CFG_DOWN)) { input=(PLAYERINPUT)(input|PI_DOWN); } if(pad&CPadConfig::getButton(CPadConfig::PAD_CFG_LEFT)) { input=(PLAYERINPUT)(input|PI_LEFT); } if(pad&CPadConfig::getButton(CPadConfig::PAD_CFG_RIGHT)) { input=(PLAYERINPUT)(input|PI_RIGHT); } if(pad&CPadConfig::getButton(CPadConfig::PAD_CFG_JUMP)) { input=(PLAYERINPUT)(input|PI_JUMP); } if(pad&CPadConfig::getButton(CPadConfig::PAD_CFG_ACTION)) { input=(PLAYERINPUT)(input|PI_ACTION); } #ifdef _RECORD_DEMO_MODE_ CDemoPlayer::demoPlayerControl *crnt; PLAYERINPUT lastInput; crnt=&s_demoControls[s_demoSize]; if(s_demoFrameCount==0) { crnt->m_inputValue=input; } lastInput=(PLAYERINPUT)crnt->m_inputValue; if(crnt->m_length==255) { lastInput=(PLAYERINPUT)(input-1); } if(lastInput==input) { crnt->m_length++; } else { s_demoSize++; ASSERT(s_demoSizem_inputValue=input; crnt->m_length=1; } s_demoFrameCount++; if(s_demoFrameCount==30*60) { writeDemoControls(); ASSERT(!"DEMO ENDED"); } #endif return input; } /*=========================================================================== end */