State Machines: Difference between revisions
(Created page with "== Overview == What on earth is a StateMachine and why do I want to build one? These are two great questions and in this tutorial I will show how to build a state machine and explain why it is critical to building anything more than even the simplest game on the Arduboy platform. Think about how most games are structured. You start on a title screen and select to play the game. While playing the game, you may select to view your inventory before battling a monster....") |
|||
(12 intermediate revisions by the same user not shown) | |||
Line 1: | Line 1: | ||
== Tutorial Instructions == | |||
The sample code for this tutorial is available below. To download the files, right click on the filename and select 'Save Link As …' | |||
* [https://github.com/filmote/StateMachine_Tutorial/blob/main/1.%20What%20is%20a%20State%20Machine/ExampleCode/StateMachine.ino StateMachine.ino] | |||
== Overview == | == Overview == | ||
What on earth is a | What on earth is a 'State Machine' and why do I want to build one? | ||
These are two great questions and in this tutorial I will show how to build a state machine and explain why it is critical to building anything more than even the simplest game on the Arduboy platform. | These are two great questions and in this tutorial I will show how to build a state machine and explain why it is critical to building anything more than even the simplest game on the Arduboy platform. | ||
Line 13: | Line 19: | ||
To keep it simple, our sample program will start with two states - a title and a game play screen. Later you will add a third screen, a high score screen, and add the plumbing for the user to move between the states. | To keep it simple, our sample program will start with two states - a title and a game play screen. Later you will add a third screen, a high score screen, and add the plumbing for the user to move between the states. | ||
Reviewing the code, you will see two functions that relate to the title screen, | Reviewing the code, you will see two functions that relate to the title screen, <code>title_Init()</code> and <code>title()</code>, and two more that handle the game play, <code>playGame_Init()</code> and <code>playGame()</code>. The function declarations for the game play screen are shown below for reference. | ||
void playGame() { | <pre> | ||
void playGame_Init() { | |||
} | // Initialise variables and get ready for the state .. | ||
} | |||
void playGame() { | |||
// Handle the users actions and render the screen .. | |||
} | |||
</pre> | |||
As you can see, each state has two basic functions. | As you can see, each state has two basic functions. | ||
The one suffixed with | The one suffixed with <code>_Init</code> is called as the program transitions into that state and can be used to initialise variables before the actual state is executed. This is similar in concept to the Arduino's <code>setup()</code> however it is executed every time we transition into the state. | ||
The second function is executed continually after the initial invocation of the | The second function is executed continually after the initial invocation of the <code>_Init</code> function is called. Again, this is analogous to the <code>loop()</code> function of every Arduino program. | ||
An example of the use of these functions is shown below. The `lives` and `score` variables are initialised in before game play has begun. | An example of the use of these functions is shown below. The `lives` and `score` variables are initialised in before game play has begun. | ||
<pre>void playGame_Init() { | |||
score = 0; | score = 0; | ||
lives = 3; | lives = 3; | ||
Line 41: | Line 47: | ||
if (touchingEnemy()) lives--; | if (touchingEnemy()) lives--; | ||
if (touchingGold()) score++; | if (touchingGold()) score++; | ||
} | }</pre> | ||
Our state functions are really simple, but how do we string them together? | Our state functions are really simple, but how do we string them together? | ||
== Creating the State Engine == | |||
We need to track the current state as the program progresses from screen to screen. This can be achieved using an <code>enum</code> as shown below. | |||
We need to track the current state as the program progresses from screen to screen. This can be achieved using an | <pre>enum GameState { | ||
Title_Init, | Title_Init, | ||
Title, | Title, | ||
Line 56: | Line 60: | ||
}; | }; | ||
GameState gameState = GameState::Title_Init; | GameState gameState = GameState::Title_Init; | ||
</pre> | |||
You will notice that I have kept the state names the same as the function names that we declared earlier. | You will notice that I have kept the state names the same as the function names that we declared earlier. | ||
The variable that is going to track our current state, | The variable that is going to track our current state, <code>gameState</code>, is initially set to <code>GameState::Title_Init</code> as this is the first screen we want the player to see. | ||
We can now use this variable in our | We can now use this variable in our <code>loop()</code> to control which state to show. | ||
<pre>void loop() { | |||
... | ... | ||
Line 76: | Line 81: | ||
title(); | title(); | ||
break; | break; | ||
case GameState::PlayGame_Init: | |||
case GameState::PlayGame_Init: | |||
playGame_Init(); | playGame_Init(); | ||
[[fallthrough]] | [[fallthrough]] | ||
default: break; | case GameState::PlayGame: | ||
playGame(); | |||
break; | |||
default: break; | |||
} | } | ||
... | ... | ||
} | } | ||
</pre> | |||
What is that | What is that <code><nowiki>[[fallthrough]]</nowiki></code> I hear you ask? After executing the <code>Title_Init</code> state we want the <code>Title</code> state to execute immediately. Had we put a `break` in place of the <code><nowiki>[[fallthrough]]</nowiki></code> , the code would wait until the next iteration of the loop to execute the <code>Title</code> state which may cause the screen to not be rendered, resulting in a black screen for a moment, | ||
I could have omitted the | I could have omitted the <code><nowiki>[[fallthrough]]</nowiki></code> line altogether and the code will work the same. You will notice that the compiler warns you that there is a potential issue with the code falling through - even though in this case we want that behaviour. The Arduino compiler uses C++ 11 - quite an old version - however in later versions of C++, 17 or greater, this statement will suppress the warning. I like to leave it in the code as it reminds me that I left the <code>break</code> statement off intentionally! | ||
The progression between the states is performed within the states themselves. | The progression between the states is performed within the states themselves. | ||
As you can see below, in addition to any other work the | As you can see below, in addition to any other work the <code>_Init()</code> function does the <code>gameState</code> must be updated to the next state name otherwise it will repeat the initialisation forever. In the <code>main</code> state function, <code>title()</code>, we react to the player pressing a button to start the game and changing the <code>gameState</code> variable to the initialisation state <code>GameState::PlayGame_Init</code>. | ||
<pre>void title_Init() { | |||
... | ... | ||
gameState = GameState::Title; | gameState = GameState::Title; | ||
Line 104: | Line 115: | ||
gameState = GameState::PlayGame_Init; | gameState = GameState::PlayGame_Init; | ||
} | } | ||
} | } | ||
</pre> | |||
So let's add a High Score state! | So let's add a High Score state! | ||
> Your Turn | <div style="width: 600px; padding: 15px; border: 1px solid; background-color: ligthgrey; box-shadow: 10px 10px 5px grey;"> | ||
<span style="font-size: 160%">Your Turn</span> | |||
Add a new state called High Scrore to the program. To do this, you will need to add two functions <code>highScore_Init()</code> and <code>highScore()</code>. You will then need to add the states to the <code>GameState</code> enum and change the `loop()` to accommodate these extra states. | |||
From the title screen, detect the player pressing the B button and move directly to the high score screen. From the game play screen, pressing A should take you to the new high score screen, not the title. Finally, pressing A from the high score screen should return you to the title screen. | |||
</div> | |||
Hopefully, this simple tutorial has shown you how to construct a simple state machine that will make structuring your next epic game easier! | Hopefully, this simple tutorial has shown you how to construct a simple state machine that will make structuring your next epic game easier! | ||
In the next topic, we will look at a simple way to organise state specific variables. | In the next topic, we will look at a simple way to organise state specific variables. | ||
---- | |||
Next [[State_Machines_-_Organising_Variables|Organising Variables]] |
Latest revision as of 10:16, 5 August 2024
Tutorial Instructions
The sample code for this tutorial is available below. To download the files, right click on the filename and select 'Save Link As …'
Overview
What on earth is a 'State Machine' and why do I want to build one?
These are two great questions and in this tutorial I will show how to build a state machine and explain why it is critical to building anything more than even the simplest game on the Arduboy platform.
Think about how most games are structured. You start on a title screen and select to play the game. While playing the game, you may select to view your inventory before battling a monster. If your battle is unsuccessful, you progress to a game over screen and maybe even are allowed to enter your details on a high score screen. Each of these 'screens' - the title screen, game play screen, inventory screen, game over screen and high score screen - are states in the game. The execution of code progresses from one state to another (and sometimes back again!) in response to the user's actions.
Even if your code is totally unstructured, you have probably already built a number of states into a game you have written without even realising it.
In this tutorial, I will present a structured approach to building a state machine that is scaled (downwards!) to the Arduboy. Splitting code logically into separate states will help you segregate code making it easier to follow and maintain.
Creating a State
To keep it simple, our sample program will start with two states - a title and a game play screen. Later you will add a third screen, a high score screen, and add the plumbing for the user to move between the states.
Reviewing the code, you will see two functions that relate to the title screen, title_Init()
and title()
, and two more that handle the game play, playGame_Init()
and playGame()
. The function declarations for the game play screen are shown below for reference.
void playGame_Init() { // Initialise variables and get ready for the state .. } void playGame() { // Handle the users actions and render the screen .. }
As you can see, each state has two basic functions.
The one suffixed with _Init
is called as the program transitions into that state and can be used to initialise variables before the actual state is executed. This is similar in concept to the Arduino's setup()
however it is executed every time we transition into the state.
The second function is executed continually after the initial invocation of the _Init
function is called. Again, this is analogous to the loop()
function of every Arduino program.
An example of the use of these functions is shown below. The `lives` and `score` variables are initialised in before game play has begun.
void playGame_Init() { score = 0; lives = 3; } void playGame() { if (touchingEnemy()) lives--; if (touchingGold()) score++; }
Our state functions are really simple, but how do we string them together?
Creating the State Engine
We need to track the current state as the program progresses from screen to screen. This can be achieved using an enum
as shown below.
enum GameState { Title_Init, Title, PlayGame_Init, PlayGame, }; GameState gameState = GameState::Title_Init;
You will notice that I have kept the state names the same as the function names that we declared earlier.
The variable that is going to track our current state, gameState
, is initially set to GameState::Title_Init
as this is the first screen we want the player to see.
We can now use this variable in our loop()
to control which state to show.
void loop() { ... switch (gameState) { case GameState::Title_Init: title_Init(); [[fallthrough]] case GameState::Title: title(); break; case GameState::PlayGame_Init: playGame_Init(); [[fallthrough]] case GameState::PlayGame: playGame(); break; default: break; } ... }
What is that [[fallthrough]]
I hear you ask? After executing the Title_Init
state we want the Title
state to execute immediately. Had we put a `break` in place of the [[fallthrough]]
, the code would wait until the next iteration of the loop to execute the Title
state which may cause the screen to not be rendered, resulting in a black screen for a moment,
I could have omitted the [[fallthrough]]
line altogether and the code will work the same. You will notice that the compiler warns you that there is a potential issue with the code falling through - even though in this case we want that behaviour. The Arduino compiler uses C++ 11 - quite an old version - however in later versions of C++, 17 or greater, this statement will suppress the warning. I like to leave it in the code as it reminds me that I left the break
statement off intentionally!
The progression between the states is performed within the states themselves.
As you can see below, in addition to any other work the _Init()
function does the gameState
must be updated to the next state name otherwise it will repeat the initialisation forever. In the main
state function, title()
, we react to the player pressing a button to start the game and changing the gameState
variable to the initialisation state GameState::PlayGame_Init
.
void title_Init() { ... gameState = GameState::Title; } void title() { if (arduboy.justPressed(A_BUTTON)) { gameState = GameState::PlayGame_Init; } }
So let's add a High Score state!
Your Turn
Add a new state called High Scrore to the program. To do this, you will need to add two functions highScore_Init()
and highScore()
. You will then need to add the states to the GameState
enum and change the `loop()` to accommodate these extra states.
From the title screen, detect the player pressing the B button and move directly to the high score screen. From the game play screen, pressing A should take you to the new high score screen, not the title. Finally, pressing A from the high score screen should return you to the title screen.
Hopefully, this simple tutorial has shown you how to construct a simple state machine that will make structuring your next epic game easier!
In the next topic, we will look at a simple way to organise state specific variables.
Next Organising Variables