Closed a327ex closed 6 years ago
Second line, typo: "programmers" instead of "peogrammers". Other than that, great post!
Great post, keep em coming!!!
There's a problem with indentation in the first code sample of Game Loop
@Kinrany Thanks for pointing it out, it was rendering fine on Firefox but not on Chrome. I fixed it now!
@adonaac That's odd, I'm using Firefox too :) You might want to use spaces instead of tabs there, just like you do in other code snippets. Github's tab size is 8 for some reason :/
Thank you for this, it's extremely helpful!
About the first exercise... It doesn't look to me that changing the value of dt
would be enough to change the loop itself. I'm not sure if I'm missing something, but I don't see anywhere on the code that "holds" the code from executing until dt
seconds pass. For reference, the original code (the commented section) does the opposite, it generates the dt
time it took for the engine to do its thing.
@rafaelferreiraql VSync is enabled by default in LÖVE, https://love2d.org/wiki/Config_Files#window.vsync. This keeps the framerate of the game consistent with the refresh rate of the monitor (usually 60Hz) and this is what does the "holding" that you mentioned. If you call love.window.setMode(800, 600, {vsync = false})
in love.load
the game will run in an unbounded manner and will be much faster than real time, as you would expect from looking at the code alone. With VSync off, to get the fixed time you can probably take the time it took for the last frame, subtract it from the fixed dt of 1/60 and then call love.timer.sleep
with that amount. But I haven't tested this yet so I don't know if it would work.
In any case, this is probably something important to mention in the article and in the answer to the first exercise that I 100% forgot about. I'll edit it in the future and add this in. Thanks!
@adonaac so basically, VSync stops the framerate from exceeding the monitor rate, and with it off, the only thing that can control the framerate is love.timer.sleep
?
That's some great insight, thanks ;)
Thanks for making this series!
Really cool stuff here :D
Good stuff! Look forward to continuing on.
For any that are interested, here's a solution to the fixed delta time exercise that will limit the framerate to a maximum of fixed_dt
function run_fixed_delta()
if love.math then
love.math.setRandomSeed(os.time())
end
if love.load then love.load(arg) end
-- We don't want the first frame's dt to include time taken by love.load.
if love.timer then love.timer.step() end
local dt = 0
local fixed_dt = 1/24
-- Main loop time.
while true do
-- Process events.
if love.event then
love.event.pump()
for name, a,b,c,d,e,f in love.event.poll() do
if name == "quit" then
if not love.quit or not love.quit() then
return a
end
end
love.handlers[name](a,b,c,d,e,f)
end
end
if love.timer then
love.timer.step()
dt = love.timer.getDelta()
if fixed_dt > dt then
love.timer.sleep(fixed_dt - dt)
love.timer.step()
dt = dt + love.timer.getDelta()
end
end
-- Call update and draw
if love.update then love.update(dt) end -- will pass 0 if love.timer is disabled
if love.graphics and love.graphics.isActive() then
love.graphics.clear(love.graphics.getBackgroundColor())
love.graphics.origin()
if love.draw then love.draw() end
love.graphics.present()
end
end
end
Very interesting article, but I don't know how to do the exercises, since I don't know where Fix Your Timestep article is. Could add some link, right? Why should I it guess when, as a beginner, I don't really know what's going on?
@SSYGEA
Introduction
This tutorial series will cover the creation of a complete game with Lua and LÖVE. It's aimed at programmers who have some experience but are just starting out with game development, or game developers who already have some experience with other languages or frameworks but want to figure out Lua or LÖVE better.
The game that will be created is called 「BYTEPATH」 and it's a mix of Bit Blaster XL and Path of Exile's Passive Skill Tree. It's simple enough that it can be covered in a number of articles without extending for too long, but with enough content that a beginner would feel uncomfortable with the code and end up giving up before finishing.
It's also at a level of complexity that most game development tutorials don't cover. Most of the problems beginners have when starting out with game development has to do with scope. The usual advice is to start small and work your way up, and while that might be a good idea, if the types of projects you're interested in cannot be made any smaller then there are very few resources out there that attempt to guide you through the problems that come up.
In my case, I've always been interested in making games with lots and lots of items/passives/skills and so when I was starting out it was really hard to figure out a good way to structure my code so that I wouldn't get lost. Hopefully these tutorials can help someone with that.
Requirements
Before you start there are some programming knowledge requirements:
The basics of programming, like variables, loops, conditionals, basic data structures and so on;
The basics of OOP, like knowing what classes, instances, attributes and methods are;
And the very basics of Lua, this quick tutorial should be good enough.
Essentially this is not for people who are just getting started with programming in general. Also, this tutorial series will have exercises. If you've ever been in the situation where you finish a tutorial and you don't know what to do next it's probably because it had no exercises, so if you don't want that to happen here then I recommend at least trying to do them.
START
To start off you need to install LÖVE on your system and then figure out how to run LÖVE projects. You can follow the steps from here for that.
Once that's done you should create a
main.lua
file in your project folder with the following contents:If you run this you should see a window popup and it should show a black screen. In the code above, once your LÖVE project is run the
love.load
function is run once at the start of the program andlove.update
andlove.draw
are run every frame. So, for instance, if you wanted to load an image and draw it, you'd do something like this:love.graphics.newImage
loads the image texture to theimage
variable and then every frame it's drawn at position 0, 0. To see thatlove.draw
actually draws the image on every frame, try this:The default size of the window is 800x600, so what this should do is randomly draw the image around the screen really fast:
Note that between every frame the screen is cleared, otherwise the image you're drawing randomly would slowly fill the entire screen as it is drawn in random positions. This happens because LÖVE provides a default game loop for its projects that clears the screen at the end of every frame. I'll go over this game loop and how you can change it now.
Game Loop
The default game loop LÖVE uses can be found in the
love.run
page, and it looks like this:When the program starts
love.run
is run and then from there everything happens. The function is fairly well commented and you can find out what each function does on the LÖVE wiki. But I'll go over the basics:In the first line we're checking to see if
love.math
is not nil. In Lua all values are true, except for false and nil, so theif love.math
condition will be true iflove.math
is defined as anything at all. In the case of LÖVE these variables are set to be enabled or not in theconf.lua
file. You don't need to worry about this file for now, but I'm just mentioning it because it's in that file that you can enable or disable individual systems likelove.math
, and so that's why there's a check to see if it's enabled or not before anything is done with one of its functions.In general, if a variable is not defined in Lua and you refer to it in any way, it will return a nil value. So if you ask
if random_variable
then this will be false unless you defined it before, likerandom_variable = 1
.In any case, if the
love.math
module is enabled (which it is by default) then its seed is set based on the current time. Seelove.math.setRandomSeed
andos.time
. After doing this, thelove.load
function is called:arg
are the command line arguments passed to the LÖVE executable when it runs the project. And as you can see, the reason whylove.load
only runs once is because it's only called once, while the update and draw functions are called multiple times inside a loop (and each iteration of that loop corresponds to a frame).After calling
love.load
and after that function does all its work, we verify thatlove.timer
is defined and calllove.timer.step
, which measures the time taken between the two last frames. As the comment explains,love.load
might take a long time to process (because it might load all sorts of things like images and sounds) and that time shouldn't be the first thing returned bylove.timer.getDelta
on the first frame of the game.dt
is also initialized to 0 here. Variables in Lua are global by default, so by sayinglocal dt
it's being defined only to the local scope of the current block, which in this case is thelove.run
function. See more on blocks here.This is where the main loop starts. The first thing that is done on each frame is the processing of events.
love.event.pump
pushes events to the event queue and according to its description those events are generated by the user in some way, so think key presses, mouse clicks, window resizes, window focus lost/gained and stuff like that. The loop usinglove.event.poll
goes over the event queue and handles each event.love.handlers
is a table of functions that calls the relevant callbacks. So, for instance,love.handlers.quit
will call thelove.quit
function if it exists.One of the things about LÖVE is that you can define callbacks in the
main.lua
file that will get called when an event happens. A full list of all callbacks is available here. I'll go over callbacks in more detail later, but this is how all that happens. Thea, b, c, d, e, f
arguments you can see passed tolove.handlers[name]
are all the possible arguments that can be used by the relevant functions. For instance,love.keypressed
receives as arguments the key pressed, its scancode and if the key press event is a repeat. So in the case oflove.keypressed
thea, b, c
values would be defined as something whiled, e, f
would be nil.love.timer.step
measures the time between the two last frames and changes the value returned bylove.timer.getDelta
. So in this casedt
will contain the time taken for the last frame to run. This is useful because then this value is passed to thelove.update
function, and from there it can be used in the game to define things with constant speeds, despite frame rate changes.After calling
love.update
,love.draw
is called. But before that we verify that thelove.graphics
module exists and that we can draw to the screen vialove.graphics.isActive
. The screen is cleared to the defined background color (initially black) vialove.graphics.clear
, transformations are reset vialove.graphics.origin
,love.draw
is finally called and thenlove.graphics.present
is used to push everything drawn inlove.draw
to the screen. And then finally:I never understood why
love.timer.sleep
needs to be here at the end of the frame, but the explanation given by a LÖVE developer here seems reasonable enough.And with that the
love.run
function ends. Everything that happens inside thewhile true
loop is referred to as a frame, which means thatlove.update
andlove.draw
are called once per frame. The entire game is basically repeating the contents of that loop really fast (like at 60 frames per second), so get used to that idea. I remember when I was starting it took me a while to get an instinctive handle on how this worked for some reason.There's a helpful discussion on this function on the LÖVE forums if you want to read more about it.
Anyway, if you don't want to you don't need to understand all of this at the start, but it's helpful to be somewhat comfortable with editing how your game loop works and to figure out how you want it to work exactly. There's an excellent article that goes over different game loop techniques and does a good job of explaining each. You can find it here.
Game Loop Exercises
1. Implement the
Fixed Delta Time
loop from the Fix Your Timestep article by changinglove.run
.2. Implement the
Variable Delta Time
loop from the Fix Your Timestep article by changinglove.run
.3. Implement the
Semi-Fixed Timestep
loop from the Fix Your Timestep article by changinglove.run
.4. Implement the
Free the Physics
loop from the Fix Your Timestep article by changinglove.run
.Game Loop Exercises HELP!
I'm going to go over the solution for the first two exercises. In general I'm not going to provide answers to exercises because it would take me a lot of time to do so and because the process you take to get to the answers is the most important part of it all, and often that will involve some googling. But for these first parts I'll go over a few of the solutions to help a little.
For the first question, we want to implement the
Fixed Delta Time
loop. As the article states, this one uses a fixed delta of 1/60s. In the defaultlove.run
function the delta changes based on the value returned bylove.timer.getDelta
, and that changes based on the time taken between the two last frames. So the default implementation is definitely not a fixed delta of 1/60s. To achieve that, we need to first remove the sections that change the delta in any way. So this piece of code will be commented out:And then we need to define our
dt
variable to the value we want it to have, which is 1/60:And then the final
love.run
function would look like this:So we commented out sections that updated
dt
or that calledlove.timer.step
. (--
is used to comment out a line and--[[]]--
is used to comment out blocks of code in Lua) We defineddt
as 1/60 and the rest of the function remains exactly the same. The problems with this approach are explained in the article, but that's how you'd do it if you wanted to for some reason.The second question asks for the implementation of the
Variable Delta Time
loop. The way this is described is as follows:If we go back to the description of the
love.timer.step
andlove.timer.getDelta
functions, that's exactly what they do. So, as it turns out, there seems to be no differences between the defaultlove.run
implementation andVariable Delta Time
.END
And with that this part of the tutorial is over. By now you should be able to run basic LÖVE projects and have some understanding of how its game loop works. For the next part I'll introduce some useful libraries that will be used throughout the entire project.