Compare commits

..

13 commits

Author SHA1 Message Date
attilavs2
84043e31e7 npc : Debug pathfinding 2024-08-02 23:15:48 +02:00
attilavs2
10d44d6a88 Merge branch 'dev' into npc_upgrade 2024-08-02 19:02:23 +02:00
mibi88
a27ad9a14f Ran clang-format 2024-08-02 18:29:56 +02:00
mibi88
fe824d7340 Inventory management. 2024-08-02 18:29:27 +02:00
mibi88
b26cf53e91 Inventory item selection. 2024-08-02 16:14:04 +02:00
mibi88
5dab7f9f18 Cleaner inventory rendering 2024-08-02 12:59:10 +02:00
mibi88
311b7f27df Ran clang-format 2024-08-01 23:09:43 +02:00
mibi88
d555c5be6f Started inventory rendering. 2024-08-01 23:09:04 +02:00
mibi88
4d2eb2e8de Graphics for the inventory. 2024-08-01 20:05:28 +02:00
mibi88
04d4c9e31e Cleaner enums with clang-format. 2024-08-01 19:14:29 +02:00
mibi88
771d844bc4 Ran clang-format. 2024-08-01 19:11:11 +02:00
mibi88
ad902217ea Merge branch 'dev' of git.planet-casio.com:Slyvtt/Collab_RPG into dev 2024-08-01 19:08:51 +02:00
mibi88
cce7209221 Starting to add the inventory. 2024-08-01 19:08:47 +02:00
30 changed files with 508 additions and 137 deletions

View file

@ -10,3 +10,4 @@ IndentCaseBlocks: true
IncludeBlocks: Regroup IncludeBlocks: Regroup
AllowShortBlocksOnASingleLine: Empty AllowShortBlocksOnASingleLine: Empty
ColumnLimit: 80 ColumnLimit: 80
AllowShortEnumsOnASingleLine: false

View file

@ -34,6 +34,7 @@ set(SOURCES
src/npc.c src/npc.c
src/events.c src/events.c
src/animation.c src/animation.c
src/inventory.c
# ... # ...
) )
# Shared assets, fx-9860G-only assets and fx-CG-50-only assets # Shared assets, fx-9860G-only assets and fx-CG-50-only assets
@ -56,6 +57,8 @@ set(ASSETS
set(ASSETS_cg set(ASSETS_cg
assets-cg/player_male.png assets-cg/player_male.png
assets-cg/player_female.png assets-cg/player_female.png
assets-cg/player_male_inv.png
assets-cg/player_female_inv.png
assets-cg/npc/char/npc_male.png assets-cg/npc/char/npc_male.png
assets-cg/npc/char/npc_female.png assets-cg/npc/char/npc_female.png
assets-cg/npc/char/npc_milkman.png assets-cg/npc/char/npc_milkman.png
@ -69,6 +72,10 @@ set(ASSETS_cg
assets-cg/INFO_Icon.png assets-cg/INFO_Icon.png
assets-cg/player_face.png assets-cg/player_face.png
assets-cg/font.png assets-cg/font.png
assets-cg/inventory.png
assets-cg/items.png
assets-cg/selection.png
assets-cg/selected.png
) )
set(ASSETS_cg_EGA64 set(ASSETS_cg_EGA64
@ -83,6 +90,8 @@ set(ASSETS_fx
assets-fx/SGN_Icon.png assets-fx/SGN_Icon.png
assets-fx/player_face.png assets-fx/player_face.png
assets-fx/font.png assets-fx/font.png
assets-fx/selection.png
assets-fx/selected.png
# ... # ...
) )
@ -98,6 +107,8 @@ set(ASSETS_fx_1b
assets-fx/1b/npc/face/npc_milkman.png assets-fx/1b/npc/face/npc_milkman.png
assets-fx/1b/npc/face/npc_police.png assets-fx/1b/npc/face/npc_police.png
assets-fx/1b/INFO_Icon.png assets-fx/1b/INFO_Icon.png
assets-fx/1b/inventory.png
assets-fx/1b/items.png
# ... # ...
) )
@ -112,7 +123,9 @@ set(ASSETS_fx_2b
assets-fx/2b/npc/face/npc_female.png assets-fx/2b/npc/face/npc_female.png
assets-fx/2b/npc/face/npc_milkman.png assets-fx/2b/npc/face/npc_milkman.png
assets-fx/2b/npc/face/npc_police.png assets-fx/2b/npc/face/npc_police.png
assets-fx/1b/INFO_Icon.png assets-fx/2b/INFO_Icon.png
assets-fx/2b/inventory.png
assets-fx/2b/items.png
# ... # ...
) )

BIN
assets-cg/base_slot.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 223 B

BIN
assets-cg/inventory.ase Normal file

Binary file not shown.

BIN
assets-cg/inventory.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 2.8 KiB

BIN
assets-cg/items.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 684 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 536 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 421 B

BIN
assets-cg/selected.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 206 B

BIN
assets-cg/selection.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 211 B

View file

@ -1,3 +1,11 @@
INFO_Icon.png: INFO_Icon.png:
type: bopti-image type: bopti-image
name: INFO_Icon_img name: INFO_Icon_img
inventory.png:
type: bopti-image
name: inventory_img
items.png:
type: bopti-image
name: items_img

BIN
assets-fx/1b/inventory.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 608 B

BIN
assets-fx/1b/items.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 181 B

View file

@ -1,3 +1,11 @@
INFO_Icon.png: INFO_Icon.png:
type: bopti-image type: bopti-image
name: INFO_Icon_img name: INFO_Icon_img
inventory.png:
type: bopti-image
name: inventory_img
items.png:
type: bopti-image
name: items_img

BIN
assets-fx/2b/inventory.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 608 B

BIN
assets-fx/2b/items.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 181 B

View file

@ -17,6 +17,14 @@ SGN_Icon.png:
type: bopti-image type: bopti-image
name: SGN_Icon_img name: SGN_Icon_img
selection.png:
type: bopti-image
name: selection_img
selected.png:
type: bopti-image
name: selected_img
font.png: font.png:
name: fontRPG name: fontRPG
type: font type: font

BIN
assets-fx/selected.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 93 B

BIN
assets-fx/selection.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 90 B

View file

@ -142,7 +142,7 @@ def convert_map(input: str, output: str, params: dict, target):
indoor = int(input_map.get_property("indoor")) indoor = int(input_map.get_property("indoor"))
except Exception as e: except Exception as e:
# Show a warning # Show a warning
print(f"WARNING: Indoor property not found.\n") print(f"WARNING: Indoor property not found.")
if indoor: if indoor:
# Get the indoor tileset # Get the indoor tileset

View file

@ -12,6 +12,9 @@
#define GRAYMODEOK 1 #define GRAYMODEOK 1
#endif #endif
#define SLOT_NUM 9
#define SLOT_COLUMNS 3
#if GINT_RENDER_RGB #if GINT_RENDER_RGB
/* The tile size */ /* The tile size */
#define T_HEIGHT 16 #define T_HEIGHT 16
@ -22,9 +25,15 @@
/* The size of the player */ /* The size of the player */
#define P_WIDTH 16 #define P_WIDTH 16
#define P_HEIGHT 16 #define P_HEIGHT 16
/*Max number of dynamic NPCs.*/ /* Max number of dynamic NPCs. */
#define NPC_STACK_SIZE 256 #define NPC_STACK_SIZE 256
/* The position of the slots in the inventory */
#define SLOT_Y 87
#define SLOT_X_EQUIPPED 222
#define SLOT_X 272
#define SLOT_W 28
#define SLOT_H 27
#define SLOT_SPACING 32
#else #else
/* The tile size */ /* The tile size */
#define T_HEIGHT 8 #define T_HEIGHT 8
@ -35,9 +44,15 @@
/* The player size */ /* The player size */
#define P_WIDTH 8 #define P_WIDTH 8
#define P_HEIGHT 8 #define P_HEIGHT 8
/*Max number of "dynamic" NPCs. We are starved for static ram on fx !*/ /* Max number of "dynamic" NPCs. We are starved for static ram on fx ! */
#define NPC_STACK_SIZE 32 #define NPC_STACK_SIZE 32
/* The position of the slots in the inventory */
#define SLOT_Y 24
#define SLOT_X_EQUIPPED 72
#define SLOT_X 88
#define SLOT_W 8
#define SLOT_H 8
#define SLOT_SPACING 12
#endif #endif
/* SPEED should NOT be 8 or bigger: it may cause bugs when handling /* SPEED should NOT be 8 or bigger: it may cause bugs when handling

View file

@ -48,7 +48,7 @@ int (*_operations[OP_AMOUNT])(int, int) = {_op_null, _op_set, _op_add, _op_sub,
#define MIN(a, b) a < b ? a : b #define MIN(a, b) a < b ? a : b
char _message_buffer[MESSAGE_BUFFER_SZ]; char *_message_buffer;
char *events_parse_string(EventHandler *handler, char *message) { char *events_parse_string(EventHandler *handler, char *message) {
size_t message_pos = 0; size_t message_pos = 0;
char in_token = 0; char in_token = 0;

View file

@ -15,7 +15,11 @@ typedef struct {
unsigned int vars; unsigned int vars;
} EventHandler; } EventHandler;
typedef enum { T_NULL, T_VAR_EDIT, T_AMOUNT } Token; typedef enum {
T_NULL,
T_VAR_EDIT,
T_AMOUNT
} Token;
typedef enum { typedef enum {
OP_NULL, OP_NULL,

View file

@ -1,6 +1,7 @@
#include "game.h" #include "game.h"
#include "config.h" #include "config.h"
#include "inventory.h"
#include "map.h" #include "map.h"
#include "mapdata.h" #include "mapdata.h"
#include "npc.h" #include "npc.h"
@ -35,6 +36,11 @@ void game_init(Game *game) {
events_init_handler(&game->handler); events_init_handler(&game->handler);
events_bind_variable(&game->handler, (int *)&game->player.life, "life"); events_bind_variable(&game->handler, (int *)&game->player.life, "life");
events_bind_variable(&game->handler, &game->mana, "mana"); events_bind_variable(&game->handler, &game->mana, "mana");
inventory_init(&game->inventory);
/* For debugging */
game->inventory.slots[5].i = I_GLOVE;
game->inventory.slots[1].i = I_ARMOR;
game->inventory.slots[8].i = I_TALISMAN;
// reload_npc(&game); // reload_npc(&game);
} }
@ -121,8 +127,10 @@ void game_render_indicator(Game *game) {
/* Draw everything. */ /* Draw everything. */
void game_draw(Game *game) { void game_draw(Game *game) {
/*Only clear if we are inside, the screen is guaranteed to be filled /* Only clear if we are inside, the screen is guaranteed to be filled
* otherwise */ * otherwise */
/* (Mibi88) if we do so, we should only use opaque tiles in the background
layer (it's currently not the case), to artefacts when rendering dialogs. */
if(game->map_level->indoor) if(game->map_level->indoor)
dclear(C_WHITE); dclear(C_WHITE);
map_render_by_layer(game, BACKGROUND); map_render_by_layer(game, BACKGROUND);
@ -135,6 +143,7 @@ void game_draw(Game *game) {
dprint(8, 8, C_BLACK, "npc_count: %d", npc_count); dprint(8, 8, C_BLACK, "npc_count: %d", npc_count);
dprint(8, 16, C_BLACK, "Mana: %d", game->mana); dprint(8, 16, C_BLACK, "Mana: %d", game->mana);
dprint(8, 24, C_BLACK, "X: %d Y: %d", game->player.x, game->player.y); dprint(8, 24, C_BLACK, "X: %d Y: %d", game->player.x, game->player.y);
inventory_draw(&game->inventory, &game->player);
} }
/* Key management */ /* Key management */
@ -149,7 +158,67 @@ void game_get_inputs(Game *game) {
if(keydown(KEY_EXIT)) if(keydown(KEY_EXIT))
game->exittoOS = true; game->exittoOS = true;
/* Player actions - Prototypes in player.h and implementation in player.c */ /* Inventory */
if(keydown(KEY_ALPHA)) {
game->inventory.open = !game->inventory.open;
/* TODO: Make something cleaner */
while(keydown(KEY_ALPHA)) {
clearevents();
sleep();
}
} else if(game->inventory.open) {
if(keydown(KEY_LEFT))
inventory_move_selection(&game->inventory, D_LEFT);
/* TODO: Make something cleaner */
while(keydown(KEY_LEFT)) {
clearevents();
sleep();
}
if(keydown(KEY_RIGHT))
inventory_move_selection(&game->inventory, D_RIGHT);
/* TODO: Make something cleaner */
while(keydown(KEY_RIGHT)) {
clearevents();
sleep();
}
if(keydown(KEY_UP))
inventory_move_selection(&game->inventory, D_UP);
/* TODO: Make something cleaner */
while(keydown(KEY_UP)) {
clearevents();
sleep();
}
if(keydown(KEY_DOWN))
inventory_move_selection(&game->inventory, D_DOWN);
/* TODO: Make something cleaner */
while(keydown(KEY_DOWN)) {
clearevents();
sleep();
}
if(keydown(KEY_OPTN))
inventory_move_from_selected(&game->inventory);
/* TODO: Make something cleaner */
while(keydown(KEY_OPTN)) {
clearevents();
sleep();
}
if(keydown(KEY_SQUARE)) {
inventory_use(&game->inventory, &game->player);
}
if(keydown(KEY_F1)) {
inventory_unequip(&game->inventory, IT_TALISMAN);
}
if(keydown(KEY_F2)) {
inventory_unequip(&game->inventory, IT_ARMOR);
}
if(keydown(KEY_F3)) {
inventory_unequip(&game->inventory, IT_WEAPON);
}
if(keydown(KEY_SHIFT))
game->inventory.selected = game->inventory.selection;
} else {
/* Player actions - Prototypes in player.h and implementation in
* player.c */
if(keydown(KEY_LEFT)) if(keydown(KEY_LEFT))
player_move(game, D_LEFT); player_move(game, D_LEFT);
if(keydown(KEY_RIGHT)) if(keydown(KEY_RIGHT))
@ -180,8 +249,10 @@ void game_get_inputs(Game *game) {
Portal *dest_portal = (Portal *)portal->portal; Portal *dest_portal = (Portal *)portal->portal;
Collider dest_collider = dest_portal->collider; Collider dest_collider = dest_portal->collider;
Map *dest_map = (Map *)portal->map; Map *dest_map = (Map *)portal->map;
player->x = (dest_collider.x1 + dest_collider.x2) / 2 * PXSIZE; player->x =
player->y = (dest_collider.y1 + dest_collider.y2) / 2 * PXSIZE; (dest_collider.x1 + dest_collider.x2) / 2 * PXSIZE;
player->y =
(dest_collider.y1 + dest_collider.y2) / 2 * PXSIZE;
game->map_level = dest_map; game->map_level = dest_map;
/* TODO: Make something cleaner */ /* TODO: Make something cleaner */
while(keydown(KEY_SHIFT)) { while(keydown(KEY_SHIFT)) {
@ -202,7 +273,7 @@ void game_get_inputs(Game *game) {
mynpc->y = player->x; mynpc->y = player->x;
mynpc->hasPath = 0; mynpc->hasPath = 0;
mynpc->owns_path = false; mynpc->owns_path = false;
mynpc->face = 1; mynpc->face = 0;
mynpc->paused = 0; mynpc->paused = 0;
mynpc->has_dialog = 0; mynpc->has_dialog = 0;
mynpc->xpath = NULL; mynpc->xpath = NULL;
@ -241,6 +312,7 @@ void game_get_inputs(Game *game) {
game->record = !game->record; game->record = !game->record;
#endif // USB_FEATURE #endif // USB_FEATURE
}
} }
void game_update_animations(Game *game, unsigned char ms) { void game_update_animations(Game *game, unsigned char ms) {

View file

@ -8,14 +8,59 @@
#include <stdint.h> #include <stdint.h>
/* The direction where the player is going to. */ /* The direction where the player is going to. */
typedef enum { D_UP, D_DOWN, D_LEFT, D_RIGHT } Direction; typedef enum {
D_UP,
D_DOWN,
D_LEFT,
D_RIGHT
} Direction;
typedef enum { P_LEFTUP = -1, P_CENTER = 0, P_RIGHTDOWN = 1 } Checkpos; typedef enum {
P_LEFTUP = -1,
P_CENTER = 0,
P_RIGHTDOWN = 1
} Checkpos;
typedef enum {
I_NONE,
I_ARMOR,
I_GLOVE,
I_SWORD,
I_BEER,
I_MILK,
I_TALISMAN,
I_AMOUNT
} Item;
typedef enum {
IT_NONE,
IT_TALISMAN,
IT_ARMOR,
IT_WEAPON,
IT_FOOD,
IT_AMOUNT
} ItemType;
typedef struct {
Item i : 4;
unsigned char durability;
} Slot;
typedef struct {
/* Backpack slots. */
Slot slots[SLOT_NUM];
/* Equipped items: first slot: talisman, second slot: armor, last slot:
weapon. */
Slot equipped[3];
/* 1 if the inventory is open. */
char open : 1;
char selected;
char selection;
} Inventory;
typedef struct { typedef struct {
uint32_t x1, y1; uint32_t x1, y1;
uint32_t x2, y2; uint32_t x2, y2;
} Collider; } Collider;
/* Struct that define player parameters */ /* Struct that define player parameters */
@ -109,6 +154,7 @@ typedef struct {
uint8_t current_group; uint8_t current_group;
uint8_t hostile_to_group; uint8_t hostile_to_group;
/* uint16_t to keep the struct aligned */
uint16_t __padding; uint16_t __padding;
} NPC; } NPC;
@ -178,6 +224,7 @@ typedef struct {
bool debug_extra; bool debug_extra;
Animation npc_animation; Animation npc_animation;
Inventory inventory;
int mana; /* Only for testing events TODO: Remove this! */ int mana; /* Only for testing events TODO: Remove this! */
} Game; } Game;

151
src/inventory.c Normal file
View file

@ -0,0 +1,151 @@
#include "inventory.h"
#include <gint/display.h>
#include <string.h>
extern bopti_image_t inventory_img;
extern bopti_image_t items_img;
extern bopti_image_t selection_img;
extern bopti_image_t selected_img;
extern bopti_image_t player_male_inv_img;
extern bopti_image_t player_female_inv_img;
char item_types[I_AMOUNT] = {IT_NONE, IT_ARMOR, IT_WEAPON, IT_WEAPON,
IT_FOOD, IT_FOOD, IT_TALISMAN};
void inventory_init(Inventory *inventory) {
inventory->open = 0;
memset(inventory->slots, 0, sizeof(Slot) * SLOT_NUM);
memset(inventory->equipped, 0, sizeof(Slot) * 3);
inventory->selected = -1;
inventory->selection = 0;
}
void inventory_draw(Inventory *inventory, Player *player) {
size_t i;
if(inventory->open) {
dimage(0, 0, &inventory_img);
for(i = 0; i < SLOT_NUM; i++) {
dsubimage(SLOT_X + (i % SLOT_COLUMNS) * SLOT_SPACING,
SLOT_Y + (i / SLOT_COLUMNS) * SLOT_SPACING, &items_img,
inventory->slots[i].i * SLOT_W, 0, SLOT_W, SLOT_H,
DIMAGE_NONE);
if(i == inventory->selection) {
dimage(SLOT_X + (i % SLOT_COLUMNS) * SLOT_SPACING,
SLOT_Y + (i / SLOT_COLUMNS) * SLOT_SPACING,
&selection_img);
}
if(i == inventory->selected) {
dimage(SLOT_X + (i % SLOT_COLUMNS) * SLOT_SPACING,
SLOT_Y + (i / SLOT_COLUMNS) * SLOT_SPACING,
&selected_img);
}
}
for(i = 0; i < 3; i++) {
dsubimage(SLOT_X_EQUIPPED, SLOT_Y + i * SLOT_SPACING, &items_img,
inventory->equipped[i].i * SLOT_W, 0, SLOT_W, SLOT_H,
DIMAGE_NONE);
}
#if GINT_RENDER_RGB
/* Render the player between the two swords if we are on cg. */
dimage(183, 20,
player->is_male ? &player_male_inv_img : &player_female_inv_img);
#endif
}
/* TODO: Stats. */
}
char inventory_add(Inventory *inventory, Item item) {
size_t i;
for(i = 0; i < SLOT_NUM; i++) {
if(!inventory->slots[i].i) {
inventory->slots[i].i = item;
inventory->slots[i].durability = 255;
return 0;
}
}
return 1;
}
void inventory_move_from_selected(Inventory *inventory) {
if(inventory->selected < 0)
return;
Slot current = inventory->slots[inventory->selection];
Slot new = inventory->slots[inventory->selected];
inventory->slots[inventory->selection] = new;
inventory->slots[inventory->selected] = current;
}
void inventory_use(Inventory *inventory, Player *player) {
Item item = inventory->slots[inventory->selection].i;
switch(item_types[item]) {
case IT_TALISMAN:
inventory->equipped[0] = inventory->slots[inventory->selection];
break;
case IT_ARMOR:
inventory->equipped[1] = inventory->slots[inventory->selection];
break;
case IT_WEAPON:
inventory->equipped[2] = inventory->slots[inventory->selection];
break;
case IT_FOOD:
/* TODO */
break;
default:
break;
}
inventory->slots[inventory->selection].i = I_NONE;
inventory->slots[inventory->selection].durability = 255;
}
void inventory_unequip(Inventory *inventory, ItemType type) {
if(inventory->slots[inventory->selection].i)
return;
switch(type) {
case IT_TALISMAN:
inventory->slots[inventory->selection] = inventory->equipped[0];
inventory->equipped[0].i = I_NONE;
inventory->equipped[0].durability = 255;
break;
case IT_ARMOR:
inventory->slots[inventory->selection] = inventory->equipped[1];
inventory->equipped[1].i = I_NONE;
inventory->equipped[1].durability = 255;
break;
case IT_WEAPON:
inventory->slots[inventory->selection] = inventory->equipped[2];
inventory->equipped[2].i = I_NONE;
inventory->equipped[2].durability = 255;
break;
default:
break;
}
}
void inventory_move_selection(Inventory *inventory, Direction direction) {
switch(direction) {
case D_UP:
inventory->selection -= SLOT_COLUMNS;
if(inventory->selection < 0)
inventory->selection = 0;
break;
case D_DOWN:
inventory->selection += SLOT_COLUMNS;
if(inventory->selection >= SLOT_NUM) {
inventory->selection = SLOT_NUM - 1;
}
break;
case D_LEFT:
if(inventory->selection > 0) {
inventory->selection--;
}
break;
case D_RIGHT:
if(inventory->selection < SLOT_NUM - 1) {
inventory->selection++;
}
break;
default:
break;
}
}

15
src/inventory.h Normal file
View file

@ -0,0 +1,15 @@
#ifndef INVENTORY_H
#define INVENTORY_H
/* The structs related to the inventory are defined in game.h */
#include "game.h"
void inventory_init(Inventory *inventory);
void inventory_draw(Inventory *inventory, Player *player);
char inventory_add(Inventory *inventory, Item item);
void inventory_move_from_selected(Inventory *inventory);
void inventory_use(Inventory *inventory, Player *player);
void inventory_unequip(Inventory *inventory, ItemType type);
void inventory_move_selection(Inventory *inventory, Direction direction);
#endif

View file

@ -32,7 +32,8 @@
extern Map *worldRPG[]; extern Map *worldRPG[];
/* Game data (defined in "game.h")*/ /* Game data (defined in "game.h")*/
Game game = {NULL, Game game = {
NULL,
{12 * PXSIZE, {12 * PXSIZE,
36 * PXSIZE, 36 * PXSIZE,
0, 0,
@ -56,7 +57,9 @@ Game game = {NULL,
false, false,
false, false,
{}, {},
100}; {},
100,
};
/* screen capture management code. TODO: Clean this up! */ /* screen capture management code. TODO: Clean this up! */
@ -117,6 +120,17 @@ int main(void) {
} }
timer_start(timer); timer_start(timer);
extern char *_message_buffer;
_message_buffer = NULL;
_message_buffer = malloc(MESSAGE_BUFFER_SZ);
if(!_message_buffer) {
dtext(64, 64, C_BLACK,
"Failed to allocate the message buffer: not "
"enough RAM available. Press any key to quit.");
dupdate();
getkey();
return 0;
}
game_init(&game); game_init(&game);
#if USB_FEATURE #if USB_FEATURE
@ -213,5 +227,6 @@ int main(void) {
#endif #endif
timer_stop(timer); timer_stop(timer);
free(_message_buffer);
return 1; return 1;
} }

View file

@ -91,43 +91,48 @@ void as_clean(uint8_t *visited, uint8_t *gscore, uint8_t *fscore) {
free(fscore); free(fscore);
} }
// TODO : Fix int as_reconstruct_path(int16_t *came_from, int w, int h, int16_t start,
int as_reconstruct_path(int16_t *came_from, int w, int h, int16_t spos, int16_t dest, bool is_alloc, NPC *npc) {
int16_t dest, NPC *npc) {
if(npc_clear_path(npc)) if(npc_clear_path(npc))
goto as_recons_fail; goto as_recons_fail;
int16_t next = came_from[dest]; int16_t prev = came_from[dest];
unsigned int i; uint32_t i;
int x, y;
for(i = 0; i < 64; i++) { for(i = 0; i < PATHFIND_MAX_ITER; i++) {
if(npc_append_path((next % w) * T_WIDTH, (next / h) * T_HEIGHT, npc)) { x = ((prev%w)*T_WIDTH + T_WIDTH/2) << PRECISION;
y = ((prev/w)*T_HEIGHT + T_HEIGHT/2) << PRECISION;
if(npc_append_path(x,y,npc)) {
goto as_recons_fail; goto as_recons_fail;
} }
prev = came_from[prev];
next = came_from[next]; if(prev == start){
if(next == spos) { x = ((prev%w)*T_WIDTH + T_WIDTH/2) << PRECISION;
if(npc_append_path((spos % w) * T_WIDTH, (spos / h) * T_HEIGHT, y = ((prev/w)*T_HEIGHT + T_HEIGHT/2) << PRECISION;
npc)) if(npc_append_path(x,y,npc))
goto as_recons_fail; goto as_recons_fail;
break; break;
} }
if(prev == -1)
goto as_recons_fail;
} }
uint16_t tx, ty; uint16_t tx, ty;
// Flip the path because it started from the end // Flip the path because it started from the end
for(i = 0; i < npc->path_length / 2; i++) { /*for(i = 0; i < npc->path_length / 2; i++) {
tx = npc->xpath[i]; tx = npc->xpath[i];
ty = npc->ypath[i]; ty = npc->ypath[i];
npc->xpath[i] = npc->xpath[npc->path_length - i - 1]; npc->xpath[i] = npc->xpath[npc->path_length - i - 1];
npc->ypath[i] = npc->ypath[npc->path_length - i - 1]; npc->ypath[i] = npc->ypath[npc->path_length - i - 1];
npc->ypath[npc->path_length - i - 1] = tx; npc->ypath[npc->path_length - i - 1] = tx;
npc->ypath[npc->path_length - i - 1] = ty; npc->ypath[npc->path_length - i - 1] = ty;
} }*/
if(is_alloc)
free(came_from); free(came_from);
npc->hasPath = true; npc->hasPath = true;
@ -136,6 +141,7 @@ int as_reconstruct_path(int16_t *came_from, int w, int h, int16_t spos,
as_recons_fail: as_recons_fail:
if(is_alloc)
free(came_from); free(came_from);
return 1; return 1;
@ -148,8 +154,8 @@ uint32_t xyram = 0xe500e000 + 32;
int npc_pathfind(int32_t dest_x, int32_t dest_y, Map *full_map, NPC *npc) { int npc_pathfind(int32_t dest_x, int32_t dest_y, Map *full_map, NPC *npc) {
int32_t i, j; int32_t i, j;
int32_t w = full_map->w; int32_t w = full_map->w/T_WIDTH/PXSIZE;
int32_t h = full_map->h; int32_t h = full_map->h/T_HEIGHT/PXSIZE;
int32_t x = (npc->curx >> PRECISION) / T_WIDTH; int32_t x = (npc->curx >> PRECISION) / T_WIDTH;
int32_t y = (npc->cury >> PRECISION) / T_HEIGHT; int32_t y = (npc->cury >> PRECISION) / T_HEIGHT;
dest_x /= T_WIDTH; dest_x /= T_WIDTH;
@ -158,8 +164,8 @@ int npc_pathfind(int32_t dest_x, int32_t dest_y, Map *full_map, NPC *npc) {
uint8_t *map = full_map->walkable; uint8_t *map = full_map->walkable;
if(dest_x < 0 || dest_x > w || dest_y < 0 || dest_x > h) /*if(dest_x < 0 || dest_x > w || dest_y < 0 || dest_x > h)
return 2; return 2;*/
if(map[spos]) if(map[spos])
return 2; return 2;
if(map[dest_y * w + dest_x]) if(map[dest_y * w + dest_x])
@ -172,7 +178,11 @@ int npc_pathfind(int32_t dest_x, int32_t dest_y, Map *full_map, NPC *npc) {
uint8_t *gscore; uint8_t *gscore;
uint8_t *fscore; uint8_t *fscore;
if(isSH3() || w * h * 5 > 1024 * 15) { bool is_alloc;
if(1 || isSH3() || w * h * 5 > 1024 * 15) {
is_alloc = true;
visited = malloc(w * h); visited = malloc(w * h);
came_from = malloc(w * h * 2); came_from = malloc(w * h * 2);
gscore = malloc(w * h * 2); gscore = malloc(w * h * 2);
@ -182,6 +192,7 @@ int npc_pathfind(int32_t dest_x, int32_t dest_y, Map *full_map, NPC *npc) {
free(came_from); free(came_from);
return 4; return 4;
} }
for(i = 0; i < w * h; i++) for(i = 0; i < w * h; i++)
visited[i] = 1; visited[i] = 1;
for(i = 0; i < w * h; i++) for(i = 0; i < w * h; i++)
@ -191,6 +202,8 @@ int npc_pathfind(int32_t dest_x, int32_t dest_y, Map *full_map, NPC *npc) {
for(i = 0; i < w * h; i++) for(i = 0; i < w * h; i++)
fscore[i] = 255; fscore[i] = 255;
} else { } else {
is_alloc = false;
visited = (void *)xyram; visited = (void *)xyram;
gscore = (void *)(xyram + w * h); gscore = (void *)(xyram + w * h);
fscore = (void *)(xyram + w * h * 2); fscore = (void *)(xyram + w * h * 2);
@ -213,7 +226,7 @@ int npc_pathfind(int32_t dest_x, int32_t dest_y, Map *full_map, NPC *npc) {
int32_t bx = x; int32_t bx = x;
int32_t by = y; int32_t by = y;
for(int iter = 0; iter < 64; iter++) { for(int iter = 0; iter < PATHFIND_MAX_ITER; iter++) {
bscore = 255; bscore = 255;
/* Cheapest known tile */ /* Cheapest known tile */
/* Could be improved with a priority queue*/ /* Could be improved with a priority queue*/
@ -230,9 +243,8 @@ int npc_pathfind(int32_t dest_x, int32_t dest_y, Map *full_map, NPC *npc) {
} }
if(bx == dest_x && by == dest_y) { if(bx == dest_x && by == dest_y) {
as_clean(visited, gscore, fscore); as_clean(visited, gscore, fscore);
return 1 /*as_reconstruct_path(came_from, w, h, spos, return 0; /*as_reconstruct_path(came_from, w, h, spos,
dest_y * w + dest_x, npc)*/ dest_y * w + dest_x, npc)*/
;
} }
visited[by * w + bx] = 1; visited[by * w + bx] = 1;
@ -276,8 +288,9 @@ void update_npcs(Game *game) {
} }
for(i = 0; i < npc_count; i++) { for(i = 0; i < npc_count; i++) {
update_npc(&npc_stack[i]); update_npc(&npc_stack[i]);
/*npc_pathfind(game->player.x, game->player.y, game->map_level, /*Temp debug*/
&npc_stack[i]);*/ game->mana = npc_pathfind(game->player.x, game->player.y, game->map_level,
&npc_stack[i]);
} }
} }

View file

@ -7,15 +7,16 @@
#include <stdbool.h> #include <stdbool.h>
#include <stdint.h> #include <stdint.h>
/*Maximum iterations before the pathfinding considers the target unreacheable*/
#define PATHFIND_MAX_ITER 64
enum { enum {
NPC_NONE = 0, NPC_NONE = 0,
NPC_FRIENDLY = 1, // The player's team NPC_FRIENDLY = 1, // The player's team
NPC_HOSTILE = 2, // to the player NPC_HOSTILE = 2, // to the player
NPC_ALL = 3 NPC_ALL = 3
}; };
/* /!\ Warning /!\ /* /!\ Warning /!\
* Do not keep hard references to non-static NPCs, as they will likely move * Do not keep hard references to non-static NPCs, as they will likely move
* in the stack */ * in the stack */