No Description

dynamic_overworld.c 3.5KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109
  1. #include <pokeagb/pokeagb.h>
  2. #define MAX_PAL_STORE 16
  3. struct PalStoreEntry {
  4. u8 reference_count;
  5. u16 tag;
  6. };
  7. extern struct PalStoreEntry stored_palettes[16];
  8. extern struct NpcType *npc_get_type(u16 id);
  9. extern void dprintf(const char *str, ...);
  10. s8 npc_dynamic_find_palette(u16 tag) {
  11. for (s8 i = 0; i < MAX_PAL_STORE; ++i) {
  12. if (stored_palettes[i].reference_count > 0 && stored_palettes[i].tag == tag)
  13. return i;
  14. }
  15. return -1;
  16. }
  17. s8 npc_dynamic_allocate_palette(u16 tag) {
  18. for (s8 i = 0; i < MAX_PAL_STORE; ++i) {
  19. if (stored_palettes[i].reference_count == 0) {
  20. stored_palettes[i].tag = tag;
  21. stored_palettes[i].reference_count++;
  22. return i;
  23. }
  24. }
  25. return -1;
  26. }
  27. u8 npc_dynamic_load_palette(u16 tag) {
  28. s8 store_entry = npc_dynamic_find_palette(tag);
  29. if (store_entry != -1) {
  30. stored_palettes[store_entry].reference_count++;
  31. return store_entry;
  32. }
  33. store_entry = npc_dynamic_allocate_palette(tag);
  34. if (store_entry == -1) {
  35. /* we do not have allocation space left */
  36. dprintf("npc_dynamic: ATTENTION - TRIED TO ALLOCATE DYNOVER PALETTE WITHOUT SPACE LEFT, INCREASING ZERO REFERENCE\n");
  37. stored_palettes[0].reference_count++;
  38. return 0;
  39. }
  40. pal_patch_for_npc(tag, (u8)store_entry);
  41. return (u8)store_entry;
  42. }
  43. void npc_dynamic_reset() {
  44. for (u8 i = 0; i < MAX_PAL_STORE; ++i) {
  45. stored_palettes[i].reference_count = 0;
  46. stored_palettes[i].tag = 0;
  47. }
  48. }
  49. void npc_dynamic_remove_entry(u8 id) {
  50. if (stored_palettes[id].reference_count > 0) {
  51. stored_palettes[id].reference_count--;
  52. dprintf("npc_dynamic: removed entry #%d\n", id);
  53. if (stored_palettes[id].reference_count == 0)
  54. stored_palettes[id].tag = 0;
  55. }
  56. }
  57. u8 npc_spawn_with_provided_template(struct RomNpc *npc, struct Template *template, u8 map, u8 bank, s16 x, s16 y) {
  58. u8 state = rom_npc_to_npc_state(npc, map, bank);
  59. if (state >= 16)
  60. return 16;
  61. struct NpcState *created_state = &npc_states[state];
  62. struct NpcType *type = npc_get_type(created_state->type_id | (npc->field3 << 8));
  63. s8 pal_slot = npc_dynamic_load_palette(type->pal_num);
  64. if (created_state->running_behavior == 76)
  65. created_state->field1 |= 0x20;
  66. template->pal_tag = 0xFFFF;
  67. u8 obj_id = template_instanciate_forward_search(template, 0, 0, 0);
  68. if (obj_id == 64) {
  69. created_state->bitfield &= 0xFE;
  70. return 16;
  71. }
  72. struct Object *npc_object = &objects[obj_id];
  73. npc_fix_position(x + created_state->to.x, y + created_state->to.y, &npc_object->pos1.x, &npc_object->pos1.y);
  74. npc_object->shift.x = -(type->pos_neg_center.x / 2);
  75. npc_object->shift.y = -(type->pos_neg_center.y / 2);
  76. npc_object->pos1.x += 8;
  77. npc_object->pos1.y += (s8)npc_object->shift.y + 16;
  78. /* Set our allocated index */
  79. npc_object->final_oam.palette_num = pal_slot;
  80. npc_object->bitfield2 |= 2;
  81. npc_object->priv[0] = state;
  82. created_state->oam_id = obj_id;
  83. u8 unknown = (created_state->field1 & 0xEF) | (16 * (type->pal_slot_unk << 25 >> 31));
  84. created_state->field1 = unknown;
  85. if (!(unknown & 0x10)) {
  86. obj_anim_image_start(npc_object, npc_direction_to_obj_anim_image_number(created_state->direction));
  87. }
  88. npc_y_height_related(created_state->height >> 4, npc_object, 1);
  89. npc_obj_offscreen_culling_and_flag_update(created_state, npc_object);
  90. return state;
  91. }