Deprecation Notice:
I've been updating the lessons now that SDL2 is officially released, please visit my new site on Github that has updated lessons and improved code.In this lesson we'll add onto our small class library of one (the Window class) by creating a simple timer and then use it in a simple program. To do this we'll be making use of the SDL_Timer functions, specifically SDL_GetTicks which returns the milliseconds elapsed since SDL was initialized.
So how would we be able to measure a time if we can only tell how long it's been since SDL was initialized? Well we could mark down the value of SDL_GetTicks when we start the timer as startTicks and then again when we stop it as endTicks. We can then subtract endTicks - startTicks to get the milliseconds elapsed during the measurement. Seems easy enough right?
Let's begin planning out how we'd like our Timer to function before we start putting it together. We'll definitely need functions for starting, stopping and getting the elapsed ticks (measured in milliseconds), and we'll also want to be able to check if the timer has been started. In addition I think it'd be nice if we could pause and unpause the timer, and maybe a function to restart it that would return the elapsed ticks and start the timer over again all in one go. We'll also need variables to track the start and pause points as mentioned above in how to use SDL_GetTicks to determine the elapsed time, along with some values to track the state of the timer.
So let's try something like this:
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
#ifndef TIMER_H | |
#define TIMER_H | |
/** | |
* What we think our timer should look like | |
*/ | |
class Timer { | |
public: | |
Timer(); | |
///Start the timer | |
void Start(); | |
///Stop the timer | |
void Stop(); | |
///Pause the timer | |
void Pause(); | |
///Unpause the timer | |
void Unpause(); | |
/** | |
* Restart the timer and return the elapsed ticks | |
* @return The elapsed ticks | |
*/ | |
int Restart(); | |
/** | |
* Get the elapsed ticks | |
* @return The elapsed ticks | |
*/ | |
int Ticks() const; | |
///Check if Timer is started | |
bool Started() const; | |
///Check if Timer is paused | |
bool Paused() const; | |
private: | |
int mStartTicks, mPausedTicks; | |
bool mStarted, mPaused; | |
}; | |
#endif |
When we construct a new timer we want it to be off, ie. not started or paused. We can do this like so:
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Timer::Timer() | |
: mStartTicks(0), mPausedTicks(0), mStarted(false), mPaused(false) | |
{ | |
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
void Timer::Start(){ | |
mStarted = true; | |
mPaused = false; | |
mStartTicks = SDL_GetTicks(); | |
} | |
void Timer::Stop(){ | |
mStarted = false; | |
mPaused = false; | |
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
void Timer::Pause(){ | |
if (mStarted && !mPaused){ | |
mPaused = true; | |
mPausedTicks = SDL_GetTicks() - mStartTicks; | |
} | |
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
void Timer::Unpause(){ | |
if (mPaused){ | |
mPaused = false; | |
mStartTicks = SDL_GetTicks() - mPausedTicks; | |
mPausedTicks = 0; | |
} | |
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
int Timer::Restart(){ | |
int elapsedTicks = Ticks(); | |
Start(); | |
return elapsedTicks; | |
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
int Timer::Ticks() const { | |
if (mStarted){ | |
if (mPaused) | |
return mPausedTicks; | |
else | |
return SDL_GetTicks() - mStartTicks; | |
} | |
return 0; | |
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
bool Timer::Started() const { | |
return mStarted; | |
} | |
bool Timer::Paused() const { | |
return mPaused; | |
} |
I'll only be posting code relevant to the specifics of using the timer in the lesson, but if you have difficulty with some code that isn't posted you can always find the full source and assets for each lesson on Github.
After opening our Window we'll want to create an instance of the Timer class and some SDL_Textures to hold our messages.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
//Our timer: | |
Timer timer; | |
//Textures to display a message and ticks elapsed | |
SDL_Texture *msg = nullptr, *ticks = nullptr; | |
//Color for the text | |
SDL_Color white = { 255, 255, 255 }; | |
//Rects for the text | |
SDL_Rect msgBox, ticksBox; | |
//Setup msg text | |
msg = Window::RenderText("Ticks Elapsed: ", "../res/Lesson8/SourceSansPro-Regular.ttf", | |
white, 30); | |
//Setup msg dstRect | |
msgBox.x = 0; | |
msgBox.y = Window::Box().h / 2; | |
//Query w & h from texture | |
SDL_QueryTexture(msg, NULL, NULL, &msgBox.w, &msgBox.h); |
We also want to display the value of the timer's Ticks function, the elapsed time, after the end of the "Ticks Elapsed: " message. There's one small issue though, Ticks returns an int, but we need a string to render a message with. This can be resolved by using a stringstream; we write Ticks to the stream and then pass it to the message creation function as a string, like so:
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
//Setup ticks message | |
//We must use a stringstream to convert int to string | |
std::stringstream ssTicks; | |
ssTicks << timer.Ticks(); | |
ticks = Window::RenderText(ssTicks.str(), "../res/Lesson8/SourceSansPro-Regular.ttf", | |
white, 30); | |
//clear the stream | |
ssTicks.str(""); | |
//Setup the ticks dstRect | |
ticksBox.x = msgBox.w + 20; | |
ticksBox.y = Window::Box().h / 2; | |
SDL_QueryTexture(ticks, NULL, NULL, &ticksBox.w, &ticksBox.h); |
Within our event polling loop we'll want to check for some key presses to tell the timer to start/stop/pause/unpause as desired.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
//If user presses any key | |
if (e.type == SDL_KEYDOWN){ | |
switch (e.key.keysym.sym){ | |
//Start/stop the timer | |
case SDLK_s: | |
//If timer was running, stop it | |
if (timer.Started()) | |
timer.Stop(); | |
else | |
timer.Start(); | |
break; | |
case SDLK_p: | |
if (timer.Paused()) | |
timer.Unpause(); | |
else | |
timer.Pause(); | |
break; | |
//For quitting, escape key | |
case SDLK_ESCAPE: | |
quit = true; | |
break; | |
default: | |
break; | |
} | |
} |
Sounds like we'll just need an if statement, and to copy down the code we used to create the texture initially:
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
//If the timer is running, update the ticks message | |
if (timer.Started() && !timer.Paused()){ | |
//Update text | |
ssTicks << timer.Ticks(); | |
SDL_DestroyTexture(ticks); | |
ticks = Window::RenderText(ssTicks.str(), "../res/Lesson8/SourceSansPro-Regular.ttf", | |
white, 30); | |
ssTicks.str(""); | |
//Update w/h | |
SDL_QueryTexture(ticks, NULL, NULL, &ticksBox.w, &ticksBox.h); | |
} |
When you run the program you should see something like this:
Where the timer will start at 0 and begin increasing once you start it. Pausing will cause the timer to stop counting, unpausing will resume it from where it left off. Stopping the timer will stop it, and when restarting it will begin again at 0. Note that the value displayed doesn't reset when the timer is stopped but rather when it's resumed.
Lesson 8 Extra Challenge!
Make an additional message display to state whether the timer is stopped/paused
Hint