Collab_RPG/src/player.c

238 lines
7.2 KiB
C
Raw Normal View History

#include "player.h"
2024-07-29 11:36:11 +02:00
#include "config.h"
#include "dialogs.h"
#include "game.h"
#include "map.h"
#include "npc.h"
#include <gint/display.h>
2024-07-27 17:38:58 +02:00
extern bopti_image_t player_male_img;
extern bopti_image_t player_female_img;
2024-07-26 17:01:51 +02:00
extern bopti_image_t npc_male;
extern bopti_image_t npc_female;
extern bopti_image_t npc_milkman;
extern bopti_image_t npc_police;
extern bopti_image_t SGN_Icon_img;
extern bopti_image_t INFO_Icon_img;
2024-07-31 20:10:45 +02:00
const bopti_image_t *faces[FACES] = {&npc_male, &npc_female, &npc_milkman,
&npc_police};
2024-07-26 17:01:51 +02:00
const char one_px_mov[8] = {
2024-07-29 11:36:11 +02:00
0, -1, /* Up */
0, 1, /* Down */
-1, 0, /* Left */
1, 0 /* Right */
};
2023-07-07 19:26:14 +02:00
/* TODO: Search for all hard tiles in the tileset. hard_tiles is a list of their
* IDs */
/* The speed of the player on the diffrent tiles in the walkable layer. */
#define WALKABLE_TILE_MAX 4
2024-07-29 11:36:11 +02:00
const short int walkable_speed[WALKABLE_TILE_MAX] = {SPEED, 0, PXSIZE, PXSIZE};
/* How much damage the player takes on the diffrent tiles in the walkable
* layer. */
2024-07-29 11:36:11 +02:00
const char damage_taken_walkable[WALKABLE_TILE_MAX] = {0, 0, 5, 0};
extern bopti_image_t demo_player_img;
extern NPC *npcRPG;
extern uint32_t nbNPC;
void player_draw(Game *game) {
Player *player = &game->player;
2024-07-29 11:36:11 +02:00
dimage(player->px - P_WIDTH / 2, player->py - P_HEIGHT / 2,
2024-07-27 17:38:58 +02:00
player->is_male ? &player_male_img : &player_female_img);
}
2024-07-29 11:36:11 +02:00
void player_move(Game *game, Direction direction) {
Player *player = &game->player;
/* How this player movement will modify the player x and y. */
char dx, dy;
2024-07-29 11:36:11 +02:00
/* If the player will collide with a hard tile or if the will go outside of
* the map. */
2024-07-29 11:36:11 +02:00
if(player_collision(game, direction, P_CENTER)) {
/* If the will collide with the center of the player. */
2024-07-29 11:36:11 +02:00
dx = one_px_mov[direction * 2] * player->speed;
dy = one_px_mov[direction * 2 + 1] * player->speed;
player_fix_position(game, dx, dy);
2024-07-29 11:36:11 +02:00
} else {
2024-07-29 21:08:55 +02:00
if(player_collision(game, direction, P_RIGHTDOWN) ||
player_collision(game, direction, P_LEFTUP)) {
2024-07-29 11:36:11 +02:00
/* If the will collide with the edges of the player. */
/* I fix his position so he won't be partially in the tile. */
/* I invert dx and dy to fix the axis where he is not moving on. */
/* Do not replace dx==0 with !dx or dy==0 with !dy, it won't work!
*/
2024-07-29 11:36:11 +02:00
dx = one_px_mov[direction * 2] * player->speed;
dy = one_px_mov[direction * 2 + 1] * player->speed;
player_fix_position(game, dx == 0, dy == 0);
}
2024-07-29 11:36:11 +02:00
/* If he won't collide with the center, so I just move him normally */
2024-07-29 11:36:11 +02:00
dx = one_px_mov[direction * 2] * player->speed;
dy = one_px_mov[direction * 2 + 1] * player->speed;
player->x += dx;
player->y += dy;
2023-07-07 14:50:30 +02:00
}
2024-07-31 20:10:45 +02:00
/* Check if we should change map */
Map *target = map_get_for_tile(game,
game->map_level->x + player->x / T_WIDTH +
one_px_mov[direction * 2],
game->map_level->y + player->y / T_HEIGHT +
one_px_mov[direction * 2 + 1]);
if(target != game->map_level) {
if(target->x > game->map_level->x) {
player->x = 0;
}
if(target->x < game->map_level->x) {
player->x = target->w * T_WIDTH;
}
if(target->y > game->map_level->y) {
player->y = 0;
}
if(target->y < game->map_level->y) {
player->y = target->h * T_HEIGHT;
}
npc_reload(game);
2024-07-31 20:10:45 +02:00
game->map_level = target;
}
}
void player_action(Game *game) {
2024-07-31 20:10:45 +02:00
/* already doing something, or can't do anything*/
if(game->player.isDoingAction || !game->player.canDoSomething)
2024-07-29 11:36:11 +02:00
return;
2024-07-31 20:10:45 +02:00
if(!game->player.isInteractingWithNPC) {
2024-07-25 13:07:19 +02:00
/* we can do something */
/* we indicate that the player is occupied */
game->player.isDoingAction = true;
2024-07-31 20:10:45 +02:00
Sign *sign = &game->map_level->signs[game->player.whichAction];
bopti_image_t *face;
2024-07-31 20:10:45 +02:00
/* we use the correct image as per the type of the item */
2024-07-31 20:10:45 +02:00
if(sign->icon)
face = &INFO_Icon_img;
2024-07-31 20:10:45 +02:00
else
face = &SGN_Icon_img;
2024-07-31 20:10:45 +02:00
uint32_t dialogStart = sign->dialogID;
2024-07-25 13:07:19 +02:00
dialogs_initiate_sequence(game, face, dialogStart);
/* when done we release the occupied status of the player */
game->player.isDoingAction = false;
2024-07-31 20:10:45 +02:00
} else {
2024-07-25 13:07:19 +02:00
/* we can do something (action IS with an NPC) */
/* we indicate that the player is occupied */
game->player.isDoingAction = true;
2024-07-31 20:10:45 +02:00
NPC *currentNPC = &game->map_level->npcs[game->player.whichAction];
2024-07-29 11:36:11 +02:00
/* we use the correct image as per the class of the item */
2024-07-26 17:01:51 +02:00
bopti_image_t *face = &npc_male;
2024-07-31 20:10:45 +02:00
2024-07-26 17:01:51 +02:00
/* It's a NPC */
2024-07-31 20:10:45 +02:00
face = (bopti_image_t *)faces[currentNPC->face];
uint32_t dialogStart = currentNPC->dialogID;
2024-07-25 13:07:19 +02:00
/* we set this NPC to paused to avoid changing its position while
* talking (the rest of the NPCs pursue their action) */
currentNPC->paused = true;
2024-07-25 13:07:19 +02:00
dialogs_initiate_sequence(game, face, dialogStart);
/* when done we release the occupied status of the player */
game->player.isDoingAction = false;
currentNPC->paused = false;
}
}
bool player_collision(Game *game, Direction direction,
2024-07-29 11:36:11 +02:00
Checkpos nomov_axis_check) {
Player *player = &game->player;
2023-07-08 16:57:04 +02:00
/* Where is the tile where he will go to from his position. */
2024-07-29 11:36:11 +02:00
char dx = one_px_mov[direction * 2];
char dy = one_px_mov[direction * 2 + 1];
2024-07-29 11:36:11 +02:00
if(!dx) {
dx += nomov_axis_check;
2024-07-29 11:36:11 +02:00
} else if(!dy) {
dy += nomov_axis_check;
}
2024-07-29 11:36:11 +02:00
dx = dx * (P_WIDTH / 2 + 1);
dy = dy * (P_HEIGHT / 2 + 1);
2023-07-08 16:57:04 +02:00
/* The tile he will go to. */
2024-07-29 11:36:11 +02:00
int player_tile_x = player->x + dx;
int player_tile_y = player->y + dy;
/* Handle a negative position differently than a positive one. */
2024-07-29 11:36:11 +02:00
if(player_tile_x < 0)
player_tile_x = player_tile_x / T_WIDTH - 1;
else
player_tile_x = player_tile_x / T_WIDTH;
if(player_tile_y < 0)
player_tile_y = player_tile_y / T_HEIGHT - 1;
else
player_tile_y = player_tile_y / T_HEIGHT;
2024-07-25 13:07:19 +02:00
int on_walkable = map_get_walkable(game, player_tile_x, player_tile_y);
2024-07-29 11:36:11 +02:00
int speed = (on_walkable >= 0 && on_walkable < WALKABLE_TILE_MAX)
? walkable_speed[on_walkable]
: 0;
2024-07-31 20:10:45 +02:00
// speed = SPEED;
2024-07-29 11:36:11 +02:00
/* if he's on a hard tile */
2024-07-29 11:36:11 +02:00
if(!speed) {
return true; /* He will collide with it. */
2023-07-07 14:50:30 +02:00
}
2024-07-29 11:36:11 +02:00
player->speed = speed;
2024-07-29 11:36:11 +02:00
2023-07-08 16:57:04 +02:00
return false; /* He won't collide with a hard tile. */
}
void player_fix_position(Game *game, bool fix_x, bool fix_y) {
Player *player = &game->player;
2024-07-29 11:36:11 +02:00
2023-07-08 16:57:04 +02:00
/* I fix his poition on x or/and on y if y need to, so that he won't be over
* the hard tile that he collided with. */
2024-07-29 11:36:11 +02:00
if(fix_x)
player->x = player->x / T_WIDTH * T_WIDTH + P_WIDTH / 2;
if(fix_y)
player->y = player->y / T_HEIGHT * T_HEIGHT + P_HEIGHT / 2;
}
void player_damage(Game *game, int amount) {
Player *player = &game->player;
2024-07-29 11:36:11 +02:00
player->life -= amount;
/* TODO: Let the player dye if life < 1. */
};