PC Games
• Orb 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 A North-South Divide For over a hundred years, messenger Duncan has wandered the world, searching for the missing pieces of an amulet that will rid him of his curse; a curse that has burdened him with an extreme intolerance of the cold, an unnaturally long life, and the despair of watching all he knew and loved become lost to the ravages of time. But now, Duncan is close to the end of his long quest. |
— Making a 2D split screen game — Note: this tutorial assumes knowledge of C, as well as prior tutorials.
Introduction Our game is mostly complete, but is missing a handful of things. The first of these is our title screen, the thing that the player will first see when they start up the game. There won't be anything super special about it, but the logo will be constructed from Triangles. Extract the archive, run cmake CMakeLists.txt, followed by make, and then use ./versus17 to run the code. You will see a screen appear as above, with a load of scrambled triangles coming together to form our logo. From here, you may play the game as normal. Once you're finished, close the window to exit, or Choose Quit from the main title options. Inspecting the code This part is focused mostly on title.c, and the loading and display of our logo. We've also added in a small starfield to the background, to make it look a little more attractive. All of these things simple to implement. A small addition to structs.h to begin with:
This Star struct holds the data about a star in our background starfield. `position` is the position of the star, `size` is its size, `pulse` is its effective brightness, while `speed` is how fast the star moves. We'll come to our starfield a bit later on. First, let's look at the changes to title.c. First, we've made a few changes to initTitle:
We're now calling a new function: loadLogo. We only want to do this once, hence we've put it into our wasInit block. Next, we're setting a variable called `scramble` to 128. `scramble` will be used to randomly adjust the the position of the Triangles that form our logo. The greater this value, the more the Triangle points will be distorted about from their original placements. Next, we set a variable called shineX to 0. This is used to control the current horizontal position of our shine effect, that moves across our logo. Finally, we clear a variable called starCamera (an SDL_FPoint), that we'll be using when drawing our stars. Now, turning to `logic`, we can see these things put into action:
We first call doStars to update our stars, and then decrease the value of `scramble`, limiting it to 0. We then test if `scramble` is 0, and update shineX. We'll increase shineX's value, wrapping it back to 0 if it equals or exceeds SCREEN_WIDTH; remember that this is used to move our shine effect across our logo, so it moves across the width of the screen. We also only want our logo shine effect to happen once our logo is settled, hence the check for `scramble` being 0. Lastly, we increase starCamera's `y`, to make our stars move up the screen when they are drawn. Now for the updates to our rendering phase, starting with `draw`:
We're doing a few next things here - we're calling drawStars, to draw our stars; calling drawLogo, to draw the logo itself; and then flushing our geometry, to ensure everything is output in the correct manner and order. We want to ensure our geometry is output before our widgets, etc. Now for drawLogo. This function will look quite familiar:
Here, we're looping through all the Triangles that make up our logo (numLogoTriangles and logoTriangles are both static variables in title.c), and then preparing to draw the Triangle as we normally would, making use of its 3 points, as SDL_Vertexes. However, once we have set the position of the vertex, we then test if `scramble` is greater than 1. If so, we'll randomly adjusted the `x` and `y` values of the vertice's position, according to `scramble`. As you can see, this will result in the position of our vertices jumping about very violently to begin with, before settling as `scramble` approaches 1. Next, we're setting the colour of our vertex. In a nutshell, our vertices will be coloured green. However, as they move further from the centre of the screen (the distance calculated being set into a variable called `d`), they'll take on a blue tint. We're only allowing the blue component of our SDL_Color to have a maximum value of 64 here. This is but a small thing, to give our logo a little more personality, rather than just being shades of green! Next up, we check how far the vertex is from the value of shineX (once again, calculating the distance and setting it into a variable called `d`). If the distance is less than SHINE_WIDTH (35), we'll want to adjust the colour. Rather than made the vertex completely white if it's within the shine width, we'll calculate the distance from the middle of the shine, to create a fall off effect. So, those vertices closer to shineX will be completely white, while those further away will simply be green. Lastly, we'll store the vertex's `x` and `y` in an SDL_Point. `points` is an array of 3 of these, so we'll choose the SDL_Point at index `n`. After we've finished rendering the Triangle itself, we'll call SDL_RenderDrawLines, passing over our 3 SDL_Points. This create an outline of our triangle. Due to the order in which our Triangles are loaded (left to right, top to bottom) this will make it appear as though our logo has a outline on the right. Just a small thing, but a nice touch. That's our logo drawing done. It's not overly complicated, but the shine effect does introduce the need for some additional understanding. Moving on now to loadLogo:
For our logo, we're opening a file that contains the data for our Triangles, and reading in the values. The first line will be the number of Triangles in the file (read into numLogoTriangles), and so we're mallocing an array to hold them all (logoTriangles), before reading in the values, and setting them to the Triangle (`t`). Again, recall the comment about our Triangle linked list support being sort of redundant; we're not using a linked list here, so if we chose to do so, we could refactor the code elsewhere to do away with them. That's the changes to title.c. We can now look at how we're handling our starfield. Over then to a new file called stars.c. This file contains just a few functions, that will be quite simple to understand. First up, we have initStars:
As the name suggests, this function sets up our stars. For each star (NUM_STARS) in our array (stars, a static array in stars.c), we're randomly setting a `position` on screen, a `pulse`, a `size` (1 or 2), and a `speed` (between 0.01 and 0.25). Next up is doStars:
Here, we're just looping through all our stars and increasing the value of its `pulse`. Finally, we have drawStars:
This function takes an SDL_PointF as an argument, as `camera`. This is the adjustment position of the stars, as we've seen with a lot of other things in our game. To begin with, we're converting our Star's `pulse` into the "value" component of a HSV to RGB conversion. We're using the sin of the star's `pulse`, to basically make the star fade and out. Our RGB will now have values between -255 and 255. We add 255, to give it a value between 0 and 512, and then multiply these values by 0.3 / 0.45 to keep them within the expected bounds of 0 - 255. Note that we're favouring a blue tint to our stars, by multiplying by 0.45, giving it a higher maximum value, compared to red and green. Next, we calculate the position of the star relative to the camera (as `x` and `y`), and then wrap these values around to the other side of the screen, should they exceed the bounds. Finally, we draw the stars with a call to drawRect. The very last thing we need to do is call initStars. We do this in initGameSystem, in init.c:
We've added initStars to our initFuncs array. Our title screen finished! It features a pleasing scrambling effect of our logo, as well as a horizontal shine. There's also a nice little starfield is the background (more or less confirming our game is set in space..!). So, we have a title screen, the ability to configure matches, goals for our games, and support for control configuration. All that is missing is sound and music, and some nice little finishing touches (you might be wondering why we're passing a camera to our starfield rendering - we'll get to that!). So, in our final part, we'll add all of these in, and conclude the creation of this game. 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 |