pokeplatinum/docs/maps/loading_maps.md
Kuruyia eab4f9a4ad Start Markdown documentation of the maps system
This starts a human-readable documentation of the various subsystems
used in the game's map system. The following files are included in this
commit:

- `maps.md`: a general overview of maps
- `bdhc.md`: an overview of the BDHC subsystem, which is used to provide
  height information on maps
- `loading_maps.md`: an overview of how maps are loaded by the game
- `file_format_specifications.md`: specs for the maps data structures

The `file_format_specifications.md` is meant to be a purely technical
document that contains a precise description of the data structures used
by the game's map system.

The other files are meant to be semi-technical documentation, where some
of the nitty-gritty details have been either simplified or left out
entirely.

Some of the maps subsystems were purposefully left out of this commit,
such as map objects and map scripts. They will need to be addressed in
future updates to those documents.

Signed-off-by: Kuruyia <github@kuruyia.net>
2025-03-16 21:40:56 +01:00

254 lines
9.8 KiB
Markdown

# Map loading
This document aims to provide an overview of how the maps are loaded during the
life cycle of the game.
## Introduction and initial loading
There are two concepts that are fundamental to understand how maps are loaded:
- the "_loaded maps quadrants_"
- the "_current map quadrants_"
The loaded maps quadrants are a 2x2 grid of maps that are currently loaded in
memory. Those 4 maps are the maximum number of maps that can be loaded at any
given time. If more maps are needed (from a map design perspective), then the
game either uses warps (common in buildings to switch between floors) or lazy
loading (the overworld and underground, for instance) if seamless transitions
are needed.
The current map quadrants are simply a division of the map the player is currently
in, into 4 quadrants. This is used to determine which maps to lazy load when the
player moves around.
One interesting property of both quadrants is that they are related: the current
map quadrant the player is in is always the opposite of the loaded maps quadrant
the player is in.
For instance, if the player is standing in the top-left quadrant of the current
map, this means that they are standing in the bottom-right quadrant of the loaded
maps quadrants.
In other words, this means that the maps loaded will be the map above the player,
on the left of the player, and the map on the top-left of the player.
Here's an illustration of this:
![Illustration of map quadrants](https://github.com/user-attachments/assets/8d3528d0-c609-4a15-8e36-a4e946988290)
Here:
- Dawn represents the player
- The black squares represent a 3x3 map matrix
- The orange dotted line represent the loaded maps quadrants when the player is
in the top-left quadrant of the current map
- The blue dotted line represent the loaded maps quadrants when the player is
in the top-right quadrant of the current map
- The red dotted line represent the loaded maps quadrants when the player is
in the bottom-left quadrant of the current map
- The green dotted line represent the loaded maps quadrants when the player is
in the bottom-right quadrant of the current map
- The central map, where Dawn is standing, is the current map
- The current map quadrants can be imagined on the current map using the dotted
lines
The map matrix is the data structure that contains the grid used to determine
which map (more precisely, which map header and map data) to load at a certain
location.
The game engine makes a distinction between two types of map loading: the first
one is what is called initial loading, which is when the game loads all 4 maps
and the entirety of their related data in other maps subsystems. This kind of
load happens when the player loads a save file, or when the player takes a warp
for instance.
Here's a flow chart that provides an overview of initial loading:
```mermaid
flowchart TD
start(Initial loading starts)
init_map_prop_animation_managers[Initialize map prop animations managers]
init_area_data_manager[Initialize area data structures]
load_map_tex_set[Load map textures]
load_map_areabm_texset[Load map props textures]
load_map_prop_models[Load map prop models]
load_map_prop_animations[Load map prop animations]
load_map_prop_matshp[Load map prop materials & shapes]
init_loaded_maps[Initialize loaded maps quadrants]
done_loading_map@{ shape: diamond, label: "Finished loading \n all 4 maps?" }
init_bdhc[Initialize BDHC data structures]
load_terrain_attributes[Load terrain attributes]
load_map_props[Load map props]
load_map_model[Load map base 3D model]
load_bdhc[Load BDHC data]
call_map_loaded_callback[Call the map loaded callback]
start --> init_map_prop_animation_managers
init_map_prop_animation_managers --> init_area_data_manager
init_area_data_manager --> load_area_data
subgraph load_area_data[Load area data]
direction LR
load_map_tex_set --> load_map_areabm_texset
load_map_areabm_texset --> load_map_prop_models
load_map_prop_models --> load_map_prop_animations
load_map_prop_animations --> load_map_prop_matshp
end
load_area_data --> load_land_data
subgraph load_land_data[Load map data]
direction LR
init_loaded_maps --> load_land_data_single
subgraph load_land_data_single[Load data of a single map]
direction LR
init_bdhc --> load_terrain_attributes
load_terrain_attributes --> load_map_props
load_map_props --> load_map_model
load_map_model --> load_bdhc
load_bdhc --> call_map_loaded_callback
end
load_land_data_single --> done_loading_map
done_loading_map -->|No| load_land_data_single
end
```
The other one is lazy loading, which is addressed in the next section.
## Lazy loading
Lazy loading is a lighter version of loading a map. For instance, it does not
load any new area data, which is why sometimes, the player must go through a gate
to force a warp and trigger an initial loading.
The game engine is flexible enough to allow tracking any map object and lazy load
maps according to its position, but most of the time the map object that is tracked
is the player. Therefore, this document assumes that the player is the map object
being tracked.
Due to how this system is designed, maps are always lazy loaded in pairs. This is
better explained with the example animations present at the end of this document.
Behind the scenes, the game has 2 separate queues for lazy loading maps, and a
space to store one extra lazy load request. This extra lazy load request overrides
one of the two queues (when the other one has finished working), and is likely
there to help in situations where the player is moving quickly between maps.
In practice, this system seems to misbehave under pressure, as the Tweaking
glitch demonstrates.
Those queues are processed sequentially and asynchronously, which gives the player
the illusion that the map is actually one single big playground.
Here's a flow chart that provides an overview of lazy loading:
```mermaid
flowchart TD
start(Lazy loading starts)
load_terrain_attributes[Load terrain attributes]
load_map_props[Load map props]
start_lazy_load_map_model[Start lazy loading the map base 3D model]
start_lazy_load_bdhc[Start lazy loading the BDHC data]
cancel_lazy_loading_map_model_bdhc_requested@{ shape: diamond, label: "Has cancelling lazy loading been requested?" }
cancel_lazy_loading_map_model_bdhc[Cancel lazy loaders]
lazy_loading_map_model_bdhc_finished@{ shape: diamond, label: "Has map base 3D model and BDHC data finished lazy loading?" }
wait_lazy_loading_map_model_bdhc_finished[Wait for both lazy loaders]
call_map_loaded_callback[Call the map loaded callback]
start --> land_data_lazy_loader_task
subgraph land_data_lazy_loader_task[Map data lazy loader task]
subgraph land_data_lazy_loader_load_map_subtask[Load maps sub-task]
direction LR
load_terrain_attributes --> load_map_props
load_map_props --> start_lazy_load_map_model
start_lazy_load_map_model --> start_lazy_load_bdhc
end
land_data_lazy_loader_load_map_subtask --> land_data_lazy_loader_finish_map_load_subtask
subgraph land_data_lazy_loader_finish_map_load_subtask[Finish loading maps sub-task]
direction LR
cancel_lazy_loading_map_model_bdhc_requested -->|No| lazy_loading_map_model_bdhc_finished
cancel_lazy_loading_map_model_bdhc_requested -->|Yes| cancel_lazy_loading_map_model_bdhc
cancel_lazy_loading_map_model_bdhc --> lazy_loading_map_model_bdhc_finished
lazy_loading_map_model_bdhc_finished -->|No| wait_lazy_loading_map_model_bdhc_finished
wait_lazy_loading_map_model_bdhc_finished --> lazy_loading_map_model_bdhc_finished
lazy_loading_map_model_bdhc_finished -->|Yes| call_map_loaded_callback
end
end
```
Below are four animations showing the player changing maps. The player is
represented by Dawn, and the maps are represented by the black squares.
The loaded maps are in green. The current map is split in four quadrants.
<details>
<summary>Player moving to the map on the top (click me)</summary>
https://github.com/user-attachments/assets/a27da2a2-f017-4133-bbf3-260cbd425baf
</details>
<details>
<summary>Player moving to the map on the right (click me)</summary>
https://github.com/user-attachments/assets/8ca91358-a0f0-46fc-8df5-aaea70b8cef9
</details>
<details>
<summary>Player moving to the map on the left (click me)</summary>
https://github.com/user-attachments/assets/795aa6b5-fff1-4717-b980-0c5b4fb9cf3f
</details>
<details>
<summary>Player moving to the map on the bottom (click me)</summary>
https://github.com/user-attachments/assets/6fe01327-d6f9-4f48-9f0b-c338655e759c
</details>
Below are four animations showing the player changing current map quadrants.
The player is represented by Dawn, and the maps are represented by the black
squares. The loaded maps are in green. The current map is split in four quadrants.
The maps being unloaded are in red, and the maps being loaded are in orange.
<details>
<summary>Player moving to the current map quadrant on the top (click me)</summary>
https://github.com/user-attachments/assets/9420b723-1b16-4ca7-954d-9eba4da3430e
</details>
<details>
<summary>Player moving to the current map quadrant on the right (click me)</summary>
https://github.com/user-attachments/assets/2205445f-e0bc-4e16-b507-5047c5548810
</details>
<details>
<summary>Player moving to the current map quadrant on the left (click me)</summary>
https://github.com/user-attachments/assets/15530353-b2be-43e1-9a16-7b2436990460
</details>
<details>
<summary>Player moving to the current map quadrant on the bottom (click me)</summary>
https://github.com/user-attachments/assets/13773394-ab72-4c0a-ae16-3e1f62c9a912
</details>