Compare commits
13 commits
7b3e3aaec8
...
84043e31e7
Author | SHA1 | Date | |
---|---|---|---|
|
84043e31e7 | ||
|
10d44d6a88 | ||
|
a27ad9a14f | ||
|
fe824d7340 | ||
|
b26cf53e91 | ||
|
5dab7f9f18 | ||
|
311b7f27df | ||
|
d555c5be6f | ||
|
4d2eb2e8de | ||
|
04d4c9e31e | ||
|
771d844bc4 | ||
|
ad902217ea | ||
|
cce7209221 |
|
@ -10,3 +10,4 @@ IndentCaseBlocks: true
|
||||||
IncludeBlocks: Regroup
|
IncludeBlocks: Regroup
|
||||||
AllowShortBlocksOnASingleLine: Empty
|
AllowShortBlocksOnASingleLine: Empty
|
||||||
ColumnLimit: 80
|
ColumnLimit: 80
|
||||||
|
AllowShortEnumsOnASingleLine: false
|
||||||
|
|
|
@ -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
After Width: | Height: | Size: 223 B |
BIN
assets-cg/inventory.ase
Normal file
BIN
assets-cg/inventory.png
Normal file
After Width: | Height: | Size: 2.8 KiB |
BIN
assets-cg/items.png
Normal file
After Width: | Height: | Size: 684 B |
BIN
assets-cg/player_female_inv.png
Normal file
After Width: | Height: | Size: 536 B |
BIN
assets-cg/player_male_inv.png
Normal file
After Width: | Height: | Size: 421 B |
BIN
assets-cg/selected.png
Normal file
After Width: | Height: | Size: 206 B |
BIN
assets-cg/selection.png
Normal file
After Width: | Height: | Size: 211 B |
|
@ -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
After Width: | Height: | Size: 608 B |
BIN
assets-fx/1b/items.png
Normal file
After Width: | Height: | Size: 181 B |
|
@ -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
After Width: | Height: | Size: 608 B |
BIN
assets-fx/2b/items.png
Normal file
After Width: | Height: | Size: 181 B |
|
@ -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
After Width: | Height: | Size: 93 B |
BIN
assets-fx/selection.png
Normal file
After Width: | Height: | Size: 90 B |
|
@ -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
|
||||||
|
|
19
src/config.h
|
@ -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
|
||||||
|
@ -24,7 +27,13 @@
|
||||||
#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
|
||||||
|
@ -37,7 +46,13 @@
|
||||||
#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
|
||||||
|
|
|
@ -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;
|
||||||
|
|
|
@ -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,
|
||||||
|
|
80
src/game.c
|
@ -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);
|
||||||
}
|
}
|
||||||
|
@ -123,6 +129,8 @@ void game_render_indicator(Game *game) {
|
||||||
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;
|
||||||
|
@ -242,6 +313,7 @@ void game_get_inputs(Game *game) {
|
||||||
|
|
||||||
#endif // USB_FEATURE
|
#endif // USB_FEATURE
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
void game_update_animations(Game *game, unsigned char ms) {
|
void game_update_animations(Game *game, unsigned char ms) {
|
||||||
animation_update(&game->npc_animation, ms);
|
animation_update(&game->npc_animation, ms);
|
||||||
|
|
53
src/game.h
|
@ -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
|
@ -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
|
@ -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
|
19
src/main.c
|
@ -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;
|
||||||
}
|
}
|
||||||
|
|
61
src/npc.c
|
@ -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]);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -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 */
|
||||||
|
|