2021-05-22 21:13:26 +00:00
# include "mario.hpp"
# include "global_vars.hpp"
# include "objectids.hpp"
# include "sprites.hpp"
Mario : : Mario ( const std : : shared_ptr < SDLPP : : Renderer > & renderer ) : SDLPP : : RectangleRender ( 0 , 0 , BLOCK_SIZE , BLOCK_SIZE , renderer , g_mario_texture , MARIO_STANDING_SRC ) {
setAnimationFrames ( MARIO_WALK_ANIM ) ;
setId ( MARIO_ID ) ;
setAlignment ( SDLPP : : OBJ_CENTER , SDLPP : : OBJ_CENTER ) ;
setAnimationSpeed ( 12.5 ) ;
pauseAnimation ( ) ;
setMovement ( 0 , 0 ) ;
2021-05-23 21:32:15 +00:00
setMovementSpeed ( 1 ) ;
2021-05-26 15:48:33 +00:00
auto bottom_detect = SDLPP : : RectColider ( 0.2 , 1 , 0.6 , 0 , MARIO_FLOOR_DETECT ) ;
2021-05-27 14:33:00 +00:00
bottom_detect . setColor ( " #FF0000 " ) ;
bottom_detect . setOutlineColor ( " #FF0000 " ) ;
2021-05-25 22:46:19 +00:00
bottom_detect . setMinHeight ( 1 ) ;
addCollision ( bottom_detect ) ;
2021-05-22 21:13:26 +00:00
addCollision (
2021-05-28 17:51:02 +00:00
SDLPP : : RectColider ( 0 , 0.25 , 0.1 , 0.6 , MARIO_LEFT_SIDE_DETECT ) ) ;
2021-05-22 21:13:26 +00:00
addCollision (
2021-05-28 17:51:02 +00:00
SDLPP : : RectColider ( 0.9 , 0.25 , 0.1 , 0.6 , MARIO_RIGHT_SIDE_DETECT ) ) ;
2021-05-23 21:45:45 +00:00
addCollision (
2021-05-27 14:33:00 +00:00
SDLPP : : RectColider ( 0 , 0 , 0.1 , 0.1 , MARIO_TOP_LEFT_DETECT ) ) ;
2021-05-25 10:26:04 +00:00
addCollision (
2021-05-27 14:33:00 +00:00
SDLPP : : RectColider ( 0.9 , 0 , 0.1 , 0.1 , MARIO_TOP_LEFT_DETECT ) ) ;
2021-05-25 20:05:50 +00:00
top_collision = std : : make_shared < SDLPP : : RectColider > ( 0.5 , 0 , 0.2 , 0.15 , MARIO_TOP_DETECT ) ;
addCollision ( top_collision ) ;
2021-05-27 14:33:00 +00:00
setColiderColor ( " #FF0000 " ) ;
2021-05-22 21:13:26 +00:00
setStatic ( false ) ;
}
void Mario : : walkLeft ( ) {
2021-05-23 21:45:45 +00:00
if ( on_ground )
resumeAnimation ( ) ;
2021-05-22 21:13:26 +00:00
addMovement ( - side_movement , 0 ) ;
if ( getMovement ( ) . getX ( ) < 0 & & faces_right ) {
flipHorizontally ( ) ;
2021-05-25 20:05:50 +00:00
top_collision - > setPos ( 0.3 , 0 ) ;
updateSizeAndPosition ( ) ;
2021-05-22 21:13:26 +00:00
faces_right = false ;
}
}
void Mario : : walkRight ( ) {
2021-05-23 21:45:45 +00:00
if ( on_ground )
resumeAnimation ( ) ;
2021-05-22 21:13:26 +00:00
addMovement ( side_movement , 0 ) ;
if ( getMovement ( ) . getX ( ) > 0 & & ! faces_right ) {
flipHorizontally ( ) ;
2021-05-25 20:05:50 +00:00
top_collision - > setPos ( 0.5 , 0 ) ;
updateSizeAndPosition ( ) ;
2021-05-22 21:13:26 +00:00
faces_right = true ;
}
}
void Mario : : setStanding ( ) {
if ( getMovement ( ) . getX ( ) = = 0 ) {
pauseAnimation ( ) ;
}
}
2021-05-27 14:33:00 +00:00
void Mario : : handleVisitor ( MarioVisitor & visitor ) {
2021-05-25 10:26:04 +00:00
// TODO - https://web.archive.org/web/20130807122227/http://i276.photobucket.com/albums/kk21/jdaster64/smb_playerphysics.png
2021-05-22 21:13:26 +00:00
// handle gravity
2021-05-22 21:54:01 +00:00
on_ground = visitor . isOnGround ( ) ;
2021-05-23 21:32:15 +00:00
if ( ! jumping & & on_ground ) {
2021-05-22 21:13:26 +00:00
resetMovementY ( ) ;
2021-05-23 21:45:45 +00:00
setTextureSourceRect ( MARIO_STANDING_SRC ) ;
if ( getMovement ( ) . getX ( ) ! = 0 )
resumeAnimation ( ) ;
2021-05-23 21:32:15 +00:00
// for some reason falling of the edge causes on_ground to be true, but
// visitor ground_y is 0
if ( visitor . getGroundY ( ) ! = 0 ) {
setPos ( getPos ( ) . getX ( ) , visitor . getGroundY ( ) - BLOCK_SIZE ) ;
}
2021-05-22 21:13:26 +00:00
}
2021-05-27 14:33:00 +00:00
// if we just left ground gravity didn't work in custom_move
if ( ! on_ground & & ! jumping & & getMovement ( ) . getY ( ) = = 0 ) {
addMovement ( 0 , 2 * gravity_add_falling ) ;
}
2021-05-23 21:57:29 +00:00
if ( visitor . topBlock ( ) & & getMovement ( ) . getY ( ) < 0 ) {
resetMovementY ( ) ;
2021-05-25 10:26:04 +00:00
stop_jump = true ;
2021-05-23 21:57:29 +00:00
}
2021-05-22 21:13:26 +00:00
// make sure Mario isn't stuck inside a wall
2021-05-25 10:26:04 +00:00
// TODO more readable function names
2021-05-25 22:46:19 +00:00
if ( visitor . isStopped ( ) ) {
2021-05-27 14:33:00 +00:00
setPos ( visitor . getStopX ( ) , getPos ( ) . getY ( ) ) ;
2021-05-27 14:57:21 +00:00
} else if ( visitor . canGoLeft ( ) ! = visitor . canGoRight ( ) & & ! ( on_ground & & visitor . getMovementBlockage ( ) . getY ( ) > getPos ( ) . getY ( ) + BLOCK_SIZE / 2 ) ) {
// only stop mario on ground if the block obstructs at least half of him (important for bug when mario lands from a high jump and is teleported if visitor is fired at wrong moment)
2021-05-25 22:46:19 +00:00
SDLPP : : Vec2D < double > next_pos = { visitor . getMovementBlockage ( ) . getX ( ) + ( visitor . canGoLeft ( ) * - 1 + visitor . canGoRight ( ) * 1 ) * BLOCK_SIZE , getPos ( ) . getY ( ) } ;
setPos ( next_pos ) ;
2021-05-27 14:33:00 +00:00
} else if ( visitor . moveTop ( ) & & jumping & & ! stop_jump ) {
2021-05-25 10:26:04 +00:00
auto objPos = visitor . getRightLeftPos ( ) ;
if ( objPos . getX ( ) < getPos ( ) . getX ( ) ) {
setPos ( objPos . getX ( ) + BLOCK_SIZE , getPos ( ) . getY ( ) ) ;
} else {
setPos ( objPos . getX ( ) - BLOCK_SIZE , getPos ( ) . getY ( ) ) ;
}
2021-05-22 21:13:26 +00:00
}
}
2021-05-22 21:54:01 +00:00
void Mario : : jump ( ) {
if ( ! on_ground )
return ;
jumping = true ;
stop_jump = false ;
max_jump = getPos ( ) . getY ( ) - 3 * BLOCK_SIZE ;
min_jump = getPos ( ) . getY ( ) - 1 * BLOCK_SIZE ;
slow_jump = getPos ( ) . getY ( ) - 2 * BLOCK_SIZE ;
2021-05-23 21:32:15 +00:00
addMovement ( 0 , - jump_movement ) ;
ticks_till_gravity = base_gravity_ticks ;
2021-05-23 21:45:45 +00:00
setTextureSourceRect ( MARIO_JUMP_SRC ) ;
pauseAnimation ( ) ;
2021-05-22 21:54:01 +00:00
}
void Mario : : stopJump ( ) {
stop_jump = true ;
}
2021-05-23 21:32:15 +00:00
void Mario : : custom_move ( int ticks ) {
if ( ! jumping & & on_ground )
return ;
if ( getMovement ( ) . getY ( ) > = 1.0625 * jump_movement )
return ;
ticks_till_gravity - = ticks ;
if ( ticks_till_gravity < 0 ) {
if ( getMovement ( ) . getY ( ) > 0 ) {
jumping = false ;
addMovement ( 0 , gravity_add_jumping ) ;
} else {
if ( stop_jump ) {
addMovement ( 0 , gravity_add_falling ) ;
} else {
addMovement ( 0 , gravity_add_jumping ) ;
}
}
ticks_till_gravity + = base_gravity_ticks ;
}
}