by Ramon Santamaria (@raysan5)
In this challenge we will implement a 3D Maze game similar to classic game Doom (id Software, 1993). Along this process we will learn how to manage vertex data and contruct levels with it.
This game is developed using OpenGL. A low level graphics library used to access and control GPU processes like data transmission (vertex, textures, shaders) and drawing.
Before starting with this challenge, it's recommended to complete the previous challenges:
- Challenge 01: Blocks game - A blocks game where player has to break a wall of blocks controlling a ball with a paddle.
- Challenge 02: 2D Dungeon game - A tile-based dungeon 2D game where player moves around a 2D dungeon finding keys and defeating monsters.
It's assumed that all concepts explained in those challenges 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:
- OpenGL 3.3 graphic pipeline functionality
- Window creation, configuration and management (GLFW3)
- Context creation (OpenGL 3.3) and extensions loading (GLAD)
- Inputs management (keyboard, mouse) (GLFW3)
- Image loading (RAM), texture creation (VRAM) and drawing
- Models loading, transform and drawing (3d meshes)
- Level map data loading and vertex buffers generation (VBO)
- Camera system creation and management (1st person)
- Collision detection and resolution (AABB collisions)
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 | opengl functionality, window and context creation, extensions loading |
01_maze_game_intro.c | InitWindow(), CloseWindow(), InitGraphicsDevice() |
02 | inputs management (keyboard, mouse) | 02_maze_game_inputs.c | IsKeyDown(), IsKeyPressed(), IsMouseButtonDown(), IsMouseButtonPressed(), GetMousePosition() |
03 | image loading, texture creation and drawing |
03_maze_game_textures.c | LoadImage(), LoadTexture() |
04 | models loading and drawing | 04_maze_game_models.c | LoadOBJ(), LoadModel(), UnloadModel(), DrawModel() |
05 | level map loading, vertex buffers creation |
05_maze_game_cubicmap.c | GenMeshCubicmap() |
06 | camera system (1st person) | 06_maze_game_camera.c | UpdateCamera() |
07 | collision detection and resolution | 07_maze_game_collisions.c | CheckCollisionCircleRec() |
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_maze_game_intro.c
For a great introduction to modern OpenGL, check this documentation.
For windows creation and management using GLFW3, check Lesson 02 from Challenge 02.
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 graphics device context
Lesson code file to review: 02_maze_game_inputs.c
We will read user inputs from keyboard and mouse, to do that we will 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.
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 (key held down)
bool IsMouseButtonPressed(int button); // Detect if a mouse button has been pressed once
bool IsMouseButtonDown(int button); // Detect if a mouse button is being pressed
Vector2 GetMousePosition(void); // Returns mouse position XY
Lesson code file to review: 03_maze_game_textures.c
We will load image data from a file, decompressing and/or decodifying information if required, and convert that image data to a texture (GPU uploading) and learn to draw that texture on the canvas.
In our implementation we will support multiple image fileformats (.bmp, .jpg, .tga, .png...) using the header-only library: stb_image.
Once image data is loaded, we will convert it to a texture and use it for drawing on the screen.
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 DrawTexture(Texture2D texture, Vector2 position, Color tint); // Draw texture in screen position coordinates
Lesson code file to review: 04_maze_game_models.c
In this lesson we will learn to load simple 3D models from OBJ fileformat, one of the most simple mesh formats. Once mesh is loaded, we can place it anywhere in the scene and draw it using a texture.
Functions to be implemented:
Mesh LoadOBJ(const char *fileName); // Load mesh from OBJ file
Model LoadModel(const char *fileName); // Load 3d model from file
void UnloadModel(Model model); // Unload model data from memory (RAM and VRAM)
void DrawModel(Model model, Vector3 position, float scale, Color tint); // Draw model on screen
Lesson code file to review: 05_maze_game_cubicmap.c
In this lesson we are loading the level map from image data and we will generate a 3D cubes-based mesh for the level, defining all required vertex data. Basically we will understand every white pixel as a cube with walls and every black pixel by an empty space, using this convention we will create every required cube vertex by vertex.
Functions to be implemented:
Mesh GenMeshCubicmap(Image cubicmap, float cubeSize); // Generate cubicmap mesh from image data
Lesson code file to review: 06_maze_game_camera.c
We will learn how to manage a camera system in 3d and we will implement a first person camera.
Actually, a camera system, as we understand it, doesn't exist, it only consist of a series of transformation we apply to all elements of our 3D world to simulate a 3D perspective and positioning.
Functions to be implemented:
void UpdateCamera(Camera *camera); // Update camera system with transformation relative to user inputs
Lesson code file to review: 07_maze_game_collisions.c
Collision detection for our 3D cubes-based map could be simplyfied to a 2D grid problem, just comparing player position and a defined player radius against cell positions in the grid that represent walls. Actually, a pretty similar implementation than the one used in Lesson 07 from Challenge 02.
Function to be implemented:
bool CheckCollisionCircleRec(Vector2 center, float radius, Rectangle rec); // Check collision between circle and rectangle
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)