by Ramon Santamaria (@raysan5)
In this challenge we will implement a 2D Dungeons game similar to the dungeons of The Legend of Zelda classic game (Nintendo, 1986). Along this process we will learn how to manage game tiles and construct levels with them, how to manage the window and player inputs from a lower level perspective (in comparison to previous challenge), how to load image data from files and convert it to textures in GPU, how to draw basic shapes and textures and, finally, some basic 2D graphics animations (spritesheet based).
This game is developed using rlgl, a raylib auxiliar module intended to simplify low level GPU access and teach basic principles of graphics programming like vertex buffers usage, textures binding, shaders usage...
Before starting with this challenge, it's recommended to complete the previous challenge:
- Challenge 01: Blocks game - A blocks game where player has to break a wall of blocks controlling a ball with a paddle.
It's assumed that all concepts explained in that challenge have already been learnt by student.
Previous knowledge required:
- Videogame life cycle (Init -> Update -> Draw -> DeInit)
- Basic screens management with screens transition
- Collision detection and resolution
- Sounds and music loading and playing
Learning Outcomes:
- rlgl functionality and possibilities
- Window creation, configuration and management (GLFW3)
- Context creation (OpenGL 3.3) and extensions loading (GLAD)
- Inputs management (keyboard, mouse) (GLFW3)
- Basic shaped drawing defining vertex data (immediate-mode)
- Image loading (RAM), texture creation (VRAM) and drawing
- Tile map data loading from text file
- Tile map collisions management
NOTE: All code provided is in C language for simplicity and clearness but it's up to the student to use more complex C++ code structures (OOP) if desired.
Lesson | Learning outcome | Source file | Related functions |
---|---|---|---|
01 | window creation and management | 01_dungeon_game_window.c | InitWindow(), CloseWindow() |
02 | context initialization, extensions loading |
02_dungeon_game_graphics.c | InitGraphicsDevice() |
03 | inputs management (keyboard) | 03_dungeon_game_inputs.c | IsKeyDown(), IsKeyPressed() |
04 | basic shapes definition | 04_dungeon_game_shapes.c | DrawLine(), DrawTriangle(), DrawRectangle() |
05 | image data loading, texture creation and drawing |
05_dungeon_game_textures.c | LoadImage(), UnloadImage(), LoadTexture(), UnloadTexture(), LoadBMP() |
06 | tilemap data loading | 06_dungeon_game_tilemap.c | LoadTilemap(), UnloadTileMap() |
07 | tilemap collision detection | 07_dungeon_game_collisions.c | CheckCollisionTilemap() |
NOTE: Most of the documentation for the challenge is directly included in the source code files as code comments. Read carefully those comments to understand every task and how implement the proposed solutions.
Lesson code file to review: 01_dungeon_game_intro.c
In this first lesson we will setup a window and see how to manage it using auxiliar library GLFW3:
Window creation
To place our graphic device (understand it as a drawing canvas), we need a window (understand it as the frame for the canvas); but that window can change from system to system. In Windows OS, that window is managed by the underlying system libraries (usually GDI) while in Linux is managed by the underlying X11 system; additionally, that window frame should match the graphic device (drawing canvas) attached to it. To make sure we create the correct frame with the correct canvas (for the current OS) we will use GLFW3 library that simplyfies that task.
Functions to be implemented:
void InitWindow(int screenWidth, int screenHeight); // Initialize window using GLFW3
void CloseWindow(void); // Close window
Lesson code file to review: 02_dungeon_game_window.c
In this lesson we will learn how to initialize the graphic device context to be able to access and control the GPU. We will use two great auxiliar libraries:
- rlgl - Simple library to manage graphic device and the underlying OpenGL layer.
- glad - Library to manage OpenGL extensions loading.
rlgl is a very thin layer (wrapper) over OpenGL that simplyfies its usage to a immediate-mode programming style, it means, just defining vertex in a very direct mode to draw elements on the screen. OpenGL 1.1 just worked that way and it was very intuitive to the user but since OpenGL 2.1 that working mode became deprecated and replaced by a more complex (and efficient) way of working, using shaders. rlgl allows programming in an immediate mode style over any OpenGL version, it just takes care internally of vertex buffers filling and setting things up simplifying graphics programming to the user without losing the power of newer OpenGL versions.
More details on the utility of rlgl intermediate layer can be found here.
Graphic Device initialization
Once the window is created with the correct configuration for the desired graphic device context (in our case, OpenGL 3.3 Core profile), we need to initialize any required OpenGL extensión and initialize some context configuration parameters.
Functions to be implemented:
void InitGraphicsDevice(int screenWidth, int screenHeight); // Initialize graphic device using rlgl
Lesson code file to review: 03_dungeon_game_inputs.c
We will need to read user inputs from keyboard, to do that we will also use GLFW3 library, to abstract our code from multiple platforms. In GLFW3 inputs come as events polled at a regular basis (usually every frame) and can be read in callback functions. Basically, we can detect a input state (key) at a specific moment and we will use that information to implement a series of useful functions.
Functions to be implemented:
bool IsKeyPressed(int key); // Detect if a key has been pressed once
bool IsKeyDown(int key); // Detect if a key is being pressed
Lesson code file to review: 04_dungeon_game_shapes.c
To draw basic shapes using rlgl, we can just define them as a series of vertices attributes (position, texture coordinates, colors...). As explained in Lesson 01, we will use a immediate mode (original from OpenGL 1.1) to do that.
Functions to be implemented:
void DrawLine(Vector2 startPos, Vector2 endPos, Color color); // Draw a line between two points
void DrawRectangle(int posX, int posY, int width, int height, Color color); // Draw a filled rectangle
Lesson code file to review: 05_dungeon_game_textures.c
To draw textures on our canvas, first we need to understand how to load some image data from an image file (probably decompressing and decodyfing read data) to obtain an array of pixels; after that, image data that is placed in RAM memory should be uploaded to VRAM memory (also referred as GPU memory) and configured with some additional display parameters, this is called a texture. Once image is loaded and converted to texture, it's ready to be drawn.
Some important concepts to remember:
- Image data is loaded from an image file and is stored in RAM memory. That data is usually compressed and/or codyfied in the image file and should be expanded to a simple array of pixels.
- Following the above declaration, note that any image file (.bmp, .jpg, .tga, .png...) will presumably have the same size once loaded into RAM, independently of the disk size of that compressed and/or codyfied data.
- To convert that image data into a texture, we upload pixels data to VRAM... and we set a series of display configuration parameters for that texture.
- Once image data is converted to a texture, we usually don't need that data in RAM memory any more.
Functions to be implemented:
Image LoadImage(const char *fileName); // Load image data from file (RAM)
void UnloadImage(Image image); // Unload image data from RAM
Texture2D LoadTextureFromImage(Image image); // Load texture from image data (VRAM)
void UnloadTexture(Texture2D texture); // Unload texture from VRAM
void void DrawTexture(Texture2D texture, int posX, int posY, Color tint) // Draw texture in screen position coordinates
Lesson code file to review: 06_dungeon_game_tiles.c
In this lesson we will learn how to load tilemap data from a simple text file and use a tileset to draw our level based on that tilemap data. We will complete the lesson adding extra information for every tile (collision information) and multiple tile-based layers to our level.
Functions to be implemented:
Tilemap LoadTilemap(const char* valuesMap, const char* collidersMap); // Load tilemap data from file (RAM)
void UnloadTilemap(Tilemap map); // Unload tilemap data from RAM
void DrawTilemap(Tilemap map, Texture2D tileset); // Draw tilemap using tileset texture
To build our map, we will use tiles. A tile is a small image piece that we use as a brick to build a level. A tilemap defines the type and position of each tile (brick) to create the level. More info about tiles here.
All tiles required for a level could be compiled into a single tileset image:
Each tile in the tileset is asigned with an ID:
Tilemap consist only in an array of IDs defining how level is build using the tileset pieces:
The same way, we can define multiple tilemap layers, for example one layer for base map and another layer for objects of the map:
Lesson code file to review: 07_dungeon_game_collision.c
We will check tilemap collisions, to avoid player moving through blocked tiles.
To check collisions, we can also use collision IDs on tileset to define which tiles are transitable, which ones are not and also, which tiles could be transitable but draw over the player (useful for trees and some walls parts):
This tileset translates into the following tilemap collisions:
Functions to be implemented:
bool CheckCollisionRecs(Rectangle rec1, Rectangle rec2); // Check collision between rectangles
We recommend joining raylib Discord community to discuss challenges with other students and developers. However, we recommend not to look at any source code written by other students or share your source code with others while working on the challenge.
This lecture is licensed under a Creative Commons Attribution-NonCommercial 4.0 International License.
Challenge code is licensed under an unmodified zlib/libpng license.
Check LICENSE for further details.
Copyright (c) 2017 Ramon Santamaria (@raysan5)