I tested it out by using two static ints which were used to count the number of frames and events respectively, and printing out the number frames that had been produced whenever the while loop in the function run() was iterated and the number of events that had been processed whenever the while loop in the function processEvents() was iterated.
And indeed a KeyPressed event was always tens of frames apart from the following KeyReleased event. One other thing that interested me during the testing was that there was always an event whose type I did not know being processed immediately after every KeyPressed event in the same frame as the KeyPressed event. Could you tell me what the event was and why it was processed?
Moreover, in order to deal with case where there were one KeyPressed event and one KeyReleased event in the same queue and thus no update would be made (though it's very unlikely as you said), I shifted the function update() to inside the while loop in the function processEvents(), and run the program again, but this time the motion of the circle became extremely slow and unsteady as compared to before. I tested the number of frames produced per second as well as the value of deltaTime, and they were generally around the same amount before the change and after the change, so I was expecting it run as well as before it's changed, but it didn't. Why is it so?
The code used to print the number frames and loops as well as the changed code are below:
Before change:
#include<SFML/Graphics.hpp>
#include <iostream>
class Game;
class Game
{
public:
Game();
void run();
private:
void processEvents();
void update(sf::Time deltaTime);
void render();
void handlePlayerInput(sf::Keyboard::Key key, bool isPressed);
sf::RenderWindow mWindow;
sf::CircleShape mPlayer;
bool mIsMovingUp = false;
bool mIsMovingDown = false;
bool mIsMovingRight = false;
bool mIsMovingLeft = false;
static int counter, frame;
};
int Game::counter = 1, Game::frame = 1;
int main()
{
Game game;
game.run();
}
Game::Game():mWindow(sf::VideoMode(640,480),"SFML Application"),mPlayer()
{
mPlayer.setRadius(40.f);
mPlayer.setPosition(100.f,100.f);
mPlayer.setFillColor(sf::Color::Cyan);
}
void Game::run()
{
sf::Clock clock;
while (mWindow.isOpen())
{
sf::Time deltaTime = clock.restart();
processEvents();
update(deltaTime);
render();
++frame;
}
}
void Game::processEvents()
{
sf::Event event;
while(mWindow.pollEvent(event))
{
std::cout << "loop NO" << counter << std::endl;
++counter;
std::cout << "frame NO" << frame << std::endl;
switch(event.type)
{
case sf::Event::KeyPressed:
handlePlayerInput(event.key.code, true);
std::cout << "subloopPressed" << std::endl;
break;
case sf::Event::KeyReleased:
handlePlayerInput(event.key.code, false);
std::cout << "subloopReleased" << std::endl;
break;
case sf::Event::Closed:
mWindow.close();
break;
}
}
}
void Game::handlePlayerInput(sf::Keyboard::Key key, bool isPressed)
{
if(key == sf::Keyboard::W)
mIsMovingUp = isPressed;
else if(key == sf::Keyboard::S)
mIsMovingDown = isPressed;
else if(key == sf::Keyboard::A)
mIsMovingLeft = isPressed;
else if(key == sf::Keyboard::D)
mIsMovingRight = isPressed;
}
void Game::update(sf::Time deltaTime)
{
sf::Vector2f movement(0.f, 0.f);
if (mIsMovingUp)
movement.y -= 100.f;
if (mIsMovingDown)
movement.y += 100.f;
if (mIsMovingLeft)
movement.x -= 100.f;
if (mIsMovingRight)
movement.x += 100.f;
mPlayer.move(movement * deltaTime.asSeconds());
}
void Game::render()
{
mWindow.clear();
mWindow.draw(mPlayer);
mWindow.display();
}
After change:
#include<SFML/Graphics.hpp>
#include <iostream>
class Game;
class Game
{
public:
Game();
void run();
private:
void processEvents(sf::Time deltaTime);
void update(sf::Time deltaTime);
void render();
void handlePlayerInput(sf::Keyboard::Key key, bool isPressed);
sf::RenderWindow mWindow;
sf::CircleShape mPlayer;
bool mIsMovingUp = false;
bool mIsMovingDown = false;
bool mIsMovingRight = false;
bool mIsMovingLeft = false;
static int counter, frame;
};
int Game::counter = 1, Game::frame = 1;
int main()
{
Game game;
game.run();
}
Game::Game():mWindow(sf::VideoMode(640,480),"SFML Application"),mPlayer()
{
mPlayer.setRadius(40.f);
mPlayer.setPosition(100.f,100.f);
mPlayer.setFillColor(sf::Color::Cyan);
}
void Game::run()
{
sf::Clock clock;
while (mWindow.isOpen())
{
sf::Time deltaTime = clock.restart();
processEvents(deltaTime);
render();
++frame;
}
}
void Game::processEvents(sf::Time deltaTime)
{
sf::Event event;
while(mWindow.pollEvent(event))
{
std::cout << "loop NO" << counter << std::endl;
++counter;
std::cout << "frame NO" << frame << std::endl;
switch(event.type)
{
case sf::Event::KeyPressed:
handlePlayerInput(event.key.code, true);
std::cout << "subloopPressed" << std::endl;
break;
case sf::Event::KeyReleased:
handlePlayerInput(event.key.code, false);
std::cout << "subloopReleased" << std::endl;
break;
case sf::Event::Closed:
mWindow.close();
break;
}
update(deltaTime);
}
}
void Game::handlePlayerInput(sf::Keyboard::Key key, bool isPressed)
{
if(key == sf::Keyboard::W)
mIsMovingUp = isPressed;
else if(key == sf::Keyboard::S)
mIsMovingDown = isPressed;
else if(key == sf::Keyboard::A)
mIsMovingLeft = isPressed;
else if(key == sf::Keyboard::D)
mIsMovingRight = isPressed;
}
void Game::update(sf::Time deltaTime)
{
sf::Vector2f movement(0.f, 0.f);
if (mIsMovingUp)
movement.y -= 100.f;
if (mIsMovingDown)
movement.y += 100.f;
if (mIsMovingLeft)
movement.x -= 100.f;
if (mIsMovingRight)
movement.x += 100.f;
mPlayer.move(movement * deltaTime.asSeconds());
}
void Game::render()
{
mWindow.clear();
mWindow.draw(mPlayer);
mWindow.display();
}