PC Games
• Orb Android Games Tutorials
• 2D shoot 'em up Latest Updates
SDL2 Versus game tutorial
Download keys for SDL2 tutorials on itch.io
The Legend of Edgar 1.37
SDL2 Santa game tutorial 🎅
SDL2 Shooter 3 tutorial
Tags • android (3) • battle-for-the-solar-system (10) • blob-wars (10) • brexit (1) • code (6) • edgar (9) • games (43) • lasagne-monsters (1) • making-of (5) • match3 (1) • numberblocksonline (1) • orb (2) • site (1) • tanx (4) • three-guys (3) • three-guys-apocalypse (3) • tutorials (17) • water-closet (4) Books The Honour of the Knights (Second Edition) (Battle for the Solar System, #1) When starfighter pilot Simon Dodds is enrolled in a top secret military project, he and his wingmates begin to suspect that there is a lot more to the theft of a legendary battleship and the Mitikas Empire's civil war than the Helios Confederation is willing to let on. Somewhere out there the Pandoran army is gathering, preparing to bring ruin to all the galaxy... |
— Making a 2D split screen game — Note: this tutorial assumes knowledge of C, as well as prior tutorials.
Introduction Okay, let's cut straight to the chase - we're going to be adding in a HUD, to provide some essential on-screen information. Extract the archive, run cmake CMakeLists.txt, followed by make, and then use ./versus07 to run the code. You will see a window open like the one above, with each player on either side of our zone. Use the default controls (or, at your option, copy the config.json file from a previous tutorial, to use that - remember to exit the game before copying the replacement file). Play the game as normal. Take note of the health bars for the two players at the bottom of the screen, and how damage received it reflected in the form of red bars, that quickly drain. Once you're finished, close the window to exit. Inspecting the code Our HUD is pretty bland right now, containing just each of the players' health bars. What it does provide though is scope to grow in future. Let's take a look at how we're handling it. Starting with structs.h, we've updated Zone:
We've added in a array of Entity pointers here called `players`, that will contain the two players. These can be referenced from anywhere in our game, simply by extracting the data from Zone. With that in mind, let's head over to player.c now, where we've updated initPlayer:
At the end of the function, we're setting the appropriate index of Zone's `players` entity pointer to `e` (being the player Entity we setup). That's all that's needed to be able to reference our player. With that done, we can move across to hud.c, that will contain all the code we need to handle the HUD itself. Starting with initHUD:
As expected, we're doing some setup here. For each of our players, we're setting the value of healthTimer (a static double array in hud.c) to 0, and also settting the value of prevHealth (another static double array) to the current value of the player's `health`. These two variables are used to control the display of the on-screen health of the current player, and control the duration of the red bar that appears whenever damage is taken. We'll see this in action in a bit. Next, we have a function named doHUD:
Right now, this does nothing more than calls doBottomBar:
This function is responsible for handling the bottom portion of the HUD, where the health bars are. Using a for-loop, we'll handle the bars for each player. The healthTimer value is used both as a timer and also as a type of state control. We'll describe how this works, from top to bottom. First of all, if our healthTimer is 0, we'll do one of two things - if the value of prevHealth is greater than the player's current `health`, we'll set the value of healthTimer to 3/4 of a second. Otherwise, the value of prevHealth will be set to the value of the player's `health`. What this means is that we're essentially monitoring the player's `health`, to see if it's fallen compared to our last recorded value. If it has, we'll prepare to animate the difference. Next, if the value of healthTimer is 1, we'll start to reduce the value of prevHealth, limiting it to value of the player's current health. If the values become equal, we'll set the value of healthTimer to 0. Again, what this means is that we'll keep decreasing the value of prevHealth until it hits the new health value. Finally, if healthTimer not 0 or 1, we'll just decrease the value of healthTimer, limiting it to 1. This limit value of 1 is important for our control state to work. Let's recap - we'll monitor the value of player's `health`, and store it in a value of prevHealth. If we notice that the player's health has fallen below this value, we'll start to reduce the value of prevHealth, but only after a set time (3/4 of a second). This generates a small window in which the reduction of the player's `health` can be made known, before we take steps to make the two equal. You may have seen this effect being used in modern games, such as one-on-one fighting games, where it provides the player with visual feedback on the amount of damage they just received. Let's move onto the rendering phase now, starting with drawHUD:
For now, we're delegating to drawBottomBars:
We're drawing a dark rectangle across the width of the screen, at the bottom, and then calling drawHealth, passing over each of our Players, via a for-loop:
To begin with, we setup an SDL_Rect (`r`), to define the position and size of our health bar (including the overall width). We first draw a dark turquoise rectangle at this position. Next, we set a variable called `w` the width of our SDL_Rect, adjusted by the percentage of prevHealth for the player, divided by MAX_PLAYER_HEALTH (to give us a value between 0 and 1). This we'll use to render a red bar, that will show our "old" health (controlled by the timer in doHUD). Finally, we assign `w` a value based on the real player health (which may be smaller than the previous health). We render a light blue bar with this value. This layering of rectangles results in a red bar that appears when we take damage, and then "drains" after a short period, as the value of prevHealth lowers to the new value of the player's health. If that's not entirely clear, take another look at doBottomBar again, to consider how the value of prevHealth will change. That's all for hud.c. We now just have to update zone.c, and we're all finished with this part. These updates will be extremely easy. Starting with initZone:
We're just calling initHUD. We next update `logic`:
Here, we're calling doHUD. Finally, we have `draw`:
We're calling drawHUD, so long as our options aren't being displayed. We want to hide the HUD if we're showing our options, else it could get messy to read with text overlaying each other. Another part done. A short part, that on the surface might not look like it's done much, but has infact added in a place to display vital information pertinent to our game. We'll be adding to the HUD as we continue the next several parts, and some of the details displayed within will be essential. So, where do we go from here? I think it's about time we completed the combat portion of our game. Now that we can display the player details, and also respawn them, we can move over to the phase where it is actually possible for the players to destroy one another, using their weapons. So, in our next part, prepare to die! Purchase The source code for all parts of this tutorial (including assets) is available for purchase, as part of the SDL2 tutorials bundle: From itch.io |