Warning: Undefined array key "HTTP_ACCEPT_LANGUAGE" in /var/www/vhosts/bilgigunlugum.net/httpdocs/index.php on line 43
SDL Game Programming

SDL3 Oyun Programlama sayfalarımız yayında...

Ana sayfa > Oyun programlama > SDL3 programming > Motion with Vsync, FFPS and time

Motion with Vsync, FFPS and time

In this section, we will examine the program that moves a ball in the main window with the arrow keys using a texture created from a .png image file and controls the ball's movement speed using one of the methods of Vertical Synchronization (VSync), Constant Frame Rate (FPS), and multiplying the movement size by the program loop's repeat time (delta time), or all of the preferred methods at the same time.

In this section, three methods can be used separately or together to limit the movement of objects, depending on the user's request. In the following sections, only the time difference and the movement size calculation method will be preferred.

When the program runs, the following operations are performed in addition to the basic operations:

  1. At the beginning of the program, the following global variables are first defined:
    • unsigned int time_start: Program start time
    • unsigned int time_start_fps: Program while loop time
    • int frame_no: Program while loop repetition count
    • int vsync: Checks whether the Vsync feature is active.
    • bool fps_fixed: Checks whether the fixed FPS feature is active. controls.
    • int fps: FPS limit value
    • bool delta_move: Controls whether time-based movement control is active.
    • unsigned int last_frame_time: Last frame time
    • int move_size: Movement size
    • A variable named ball is defined from the following structure data type:
      struct objects {
        float x;
        float y;
        float width;
        float height;
        float vel_x;
        float vel_y;
      } ball;
      
  2. The init_vars() function assigns initial values ​​to global variables.
  3. In the load_img() function, an image file for the ball is loaded into the texture variable with the IMG_LoadTexture() function.
  4. In the main() function, 0 is assigned to the frame_no variable and the program start time is assigned to the time_start variable with the SDL_GetTicks() function.
  5. At the beginning of the while loop, the active time value is retrieved with the SDL_GetTicks() function and assigned to the time_start_fps variable.
  6. In the process_event() function,
    • When the arrow keys are pressed, the move_size variable value in the direction indicated by the arrow is added to the ball.vel_x or ball.vel_y variable, which has a value of 0. Thus, movement is achieved.
    • When the arrow keys are released, the value of the movement size variable named move_size in the direction indicated by the arrow is subtracted from the ball.vel_x or ball.vel_y variables, and the values ​​of these variables are set to 0. Thus, the movement ends.
    • When the V key is pressed, if vertical synchronization is active, it is passive, if passive, it is active.
    • When the F key is pressed, if the fixed FPS process is active, it is passive, if passive, it is active.
    • When the T key is pressed, if the time-based movement size determination method is active, it is passive, if passive, it is active.
    • When the V, F, T keys are released, the move_size variable takes the value 100 if delta_move is true, and 10 otherwise. The frame_no variable value is 0 and the time_start value is the active time value.
    • When the PageUp key is pressed, if the fps value is less than 100, the fps variable value is increased by 1.
    • When the PageDown key is pressed, if the fps value is greater than 10, the fps variable value is decreased by 1.
    • When the PageUp and PageDown keys are released, the frame_no variable value is 0 and the time_start value is the active time value.
  7. In update_screen() function,
    • The difference between the active time and the previous time of the loop is taken in seconds and assigned to the time_delta variable.
    • The active time is retrieved with the SDL_GetTicks() function and assigned to the last_frame_time variable to be used in the next iteration of the loop.
    • If the time-dependent move option (delta_move) is enabled, the product of ball.vel_x and the loop iteration time (time_delta), otherwise ball.vel_x is assigned to the move_x variable.
    • The value of ball.x is incremented by the move_x value.
    • If ball.x is less than 0 or the sum of ball.x and ball.width is greater than the width of the window, ball.x takes its previous value.
    • If the time-dependent move option (delta_move) is enabled, the product of ball.vel_y and the loop iteration time (time_delta), otherwise ball.vel_y is assigned to the move_y variable
    • The ball.y value is incremented by the move_y value.
    • If ball.y is less than 0 or the sum of ball.y and ball.height is greater than the height of the window, ball.y takes its previous value.
  8. Within the draw_screen() function,
    • The ball is displayed in its position in the main window with the SDL_RenderTexture() function.
    • The time elapsed (in milliseconds) since the program started is assigned to the time_elapsed variable. This value is reset when the V, F, T, PageUp and PageDown keys are pressed.
    • The time_elapsed variable is used to obtain minute, second and millisecond values ​​for the time counter.
    • The number of times the while loop is repeated is divided by the seconds of the elapsed time, and the number of times the loop repeats in a second (average number of frames) is found, and this value is assigned to the float avg_fps variable.
    • The time to be elapsed for a frame is calculated by dividing the value of 1000 by the fps variable value and assigned to the fpms variable.
    • The difference between the start time of the last iteration of the while loop and the active time value is calculated, and the elapsed time value is assigned to the time_passed_ms variable.
    • If the fixed FPS method is active (fps_fixed variable is true) and the time_passed_ms variable value is less than the fpms variable value, the program is used with the SDL_Delay() function for the time equal to the difference between the two variables. its operation is delayed.
    • Time counter values, frame count, FPS value, Vsync, fixed FPS and the activity status of the time-based motion restriction methods are written in the main window title.
    • The frame_no value is incremented by 1.

main.c file content will be as follows:


#include <stdio.h>
#include <SDL3/SDL.h>
#include <SDL3_image/SDL_image.h>

// Window width and height
#define WINDOW_WIDTH 640
#define WINDOW_HEIGHT 480

// Global variables
int is_running = false;        // Variable that controls the execution of the main loop of the program
SDL_Window *window = NULL;     // Main window variable
SDL_Texture *texture = NULL;   // Texture variable
SDL_Renderer *renderer = NULL; // Renderer variable

unsigned int time_start;       // Program start time
unsigned int time_start_fps;   // Program while loop time
int frame_no;                  // while loop iteration number (frame)
int vsync;                     // Controls whether Vsync is enabled
bool fps_fixed;                // Controls whether Fixed FPS is enabled
int fps;                       // Frame Per Second (FPS) to fix
bool delta_move;               // Controls whether time dependent move is enabled
unsigned int last_frame_time;  // Last frame time
int move_size;                 // Movement size

// Struct declaration
struct objects {
    float x;
    float y;
    float width;
    float height;
    float vel_x;
    float vel_y;
} ball;

// Function prototypes
int init_window(void);         // Create window and renderer
void init_vars(void);          // Initialize variables
void load_img(void);           // Load image to texture
void process_event(void);      // Process events
void update_screen();          // Update values
void draw_screen(void);        // Draw screen
void destroy_window(void);     // Destroy window

int main(int argc, char* argv[])
{
    // Create window and renderer
    is_running = init_window();

    // Initialize variables
    init_vars();

    // Load .png file to texture
    load_img();

    frame_no = 0;
    time_start = SDL_GetTicks();        // Get program start time

    // Main loop
    while (is_running) {
       time_start_fps = SDL_GetTicks(); // Get time for while loop iteration start
       process_event();                 // Processing SDL events (Here keyboard inputs)
       update_screen();                 // Updating variables
       draw_screen();                   // Drawing objects on the window (Rendering)
    }

    // Destroy renderer and SDL window
    destroy_window();

    return 0;
}

// Create window and renderer
int init_window(void)
{
    // Initialize the SDL library.
    if(SDL_Init(SDL_INIT_VIDEO) == false) {
       SDL_Log("SDL init error: %s\n", SDL_GetError());
       return false;
    }

    // Create a window and a 2D rendering context for the window.
    if(!SDL_CreateWindowAndRenderer("SDL3 window", WINDOW_WIDTH, WINDOW_HEIGHT, 0, &window, &renderer)) {
       return false;
    }

    return true;
}

// Initialization function that runs only once at the beginning of the program
void init_vars(void)
{
    time_start = 0;
    time_start_fps = 0;
    frame_no = 0;
    vsync = 0;
    fps = 30;
    fps_fixed = false;
    delta_move = false;
    last_frame_time = 0;
    move_size = 10;

    ball.x = 0;
    ball.y = 0;
    ball.width = 20;
    ball.height = 20;
    ball.vel_x = 0;
    ball.vel_y = 0;
}

// Load image to texture
void load_img(void)
{
    // Loading image to texture
    texture = IMG_LoadTexture(renderer, "ball.png");

	if(texture == NULL) {
       SDL_Log("IMG_LoadTexture error: %s\n", SDL_GetError());
	}
}

// Function to control SDL events and process keyboard inputs
void process_event(void)
{
    SDL_Event event;

    // Creating a loop to process user inputs
    while (SDL_PollEvent(&event)) {
       switch (event.type) {
          case SDL_EVENT_QUIT: // Logout action by the user (x button at the top right of the window)
               is_running = false; // Terminates the execution of the program main loop.
               break;

          case SDL_EVENT_KEY_DOWN: // Key pressed.
               if(event.key.repeat == 0) {
                  switch(event.key.key) {
                     case SDLK_UP: ball.vel_y -= move_size; break;
                     case SDLK_DOWN: ball.vel_y += move_size; break;
                     case SDLK_LEFT: ball.vel_x -= move_size; break;
                     case SDLK_RIGHT: ball.vel_x += move_size; break;
                     case SDLK_V: // Enable vertical synchronization (VSync)
                          vsync = vsync==SDL_RENDERER_VSYNC_DISABLED ? 1 : SDL_RENDERER_VSYNC_DISABLED;
                          SDL_SetRenderVSync(renderer, vsync);
                          break;
                     case SDLK_F: // Enable Fixed Frame Rate (FPS)
                          fps_fixed = !fps_fixed;
                          break;
                     case SDLK_T: // Enable calculating movement with time difference
                          delta_move = !delta_move;
                          break;

                     case SDLK_PAGEUP:   // Increase fixed fps limit by 1
                          if(fps<100) fps++;
                          break;
                     case SDLK_PAGEDOWN: // Decrease fixed fps limit by 1
                          if(fps>10) fps--;
                          break;

                     case SDLK_ESCAPE: is_running = false; break;
                  }
               }
              break;

          case SDL_EVENT_KEY_UP: // Key released.
               if(event.key.repeat == 0) {
                  switch(event.key.key) {
                     case SDLK_UP: ball.vel_y += move_size; break;
                     case SDLK_DOWN: ball.vel_y -= move_size; break;
                     case SDLK_LEFT: ball.vel_x += move_size; break;
                     case SDLK_RIGHT: ball.vel_x -= move_size; break;

                     case SDLK_V:
                     case SDLK_F:
                     case SDLK_T:
                          move_size = delta_move ? 100 : 10;
                     case SDLK_PAGEUP:
                     case SDLK_PAGEDOWN:
                          frame_no = 0;
                          time_start = SDL_GetTicks();
                          break;
                  }
               }
               break;
       }
    }
}

// Updates objects in the main window
void update_screen(void)
{
    // Get the difference between the active time and the previous time of loop in seconds
    float time_delta = (SDL_GetTicks() - last_frame_time) / 1000.0;
    // Assign the active time to use in the next iteration of the loop
    last_frame_time = SDL_GetTicks();

    // Calculate moving size
    // if time dependent movement option (delta_move) is on, multiplication of ball.vel_x and
    // the loop repeat time (delta time), otherwise ball.vel_x assigned to move_x variable.
    float move_x = delta_move ? (ball.vel_x * time_delta) : ball.vel_x;

    // Ball x coordinate increased by move_x
    ball.x += move_x;
    // If ball.x is lower than 0 or, sum of ball.x and ball.width is greater than windows width
    // ball.x gets its previous value.
    if((ball.x<0) || (ball.x + ball.width > WINDOW_WIDTH)) {
	   ball.x -= move_x;
    }

    // if time dependent movement option (delta_move) is on, multiplication of ball.vel_y and
    // the loop repeat time (delta time), otherwise ball.vel_y assigned to move_y variable.
    float move_y = delta_move ? (ball.vel_y * time_delta) : ball.vel_y;

    // Ball y coordinate increased by move_y
    ball.y += move_y;
    // If ball.yx is lower than 0 or, sum of ball.y and ball.height is greater than windows height
    // ball.y gets its previous value.
    if((ball.y<0) || (ball.y + ball.height > WINDOW_HEIGHT)) {
       ball.y -= move_y;
    }
}

// Render function used to draw game objects in the main window
void draw_screen(void)
{
    SDL_SetRenderDrawColor(renderer, 0, 0, 0, 255); // Set the color used for drawing operations.
    SDL_RenderClear(renderer); // Clear the current rendering target with the drawing color.

    SDL_FRect rect_dst = { ball.x, ball.y, 20, 20 };
    // Load texture content to main window
	SDL_RenderTexture(renderer, texture, NULL, &rect_dst);

    // Loading all objects drawn one by one to the back buffer to the front buffer at once and loading them onto the screen
    // This process prevents each drawn object from being displayed on the screen one by one.
    SDL_RenderPresent(renderer); // Update on screen all operations performed since the previous call

    // The time elapsed since the start of the program in milliseconds
    // It is reset when the V, F, T, PageUp and PageDown keys are pressed.
    unsigned int time_elapsed = SDL_GetTicks()-time_start;

    int milliseconds = time_elapsed;   // The elapsed time in milliseconds
    int seconds = milliseconds / 1000; // The elapsed time in seconds
    milliseconds %= 1000;              // The millisecond part of the counter
    int minutes = seconds / 60;        // The elapsed time in minutes
    seconds %= 60;                     // The second part of the counter
    minutes %= 60;                     // The minute part of the counter

    // Calculating FPS
    // The number of times the while loop repeats is divided by the seconds value of the elapsed time,
    // The number of times the loop repeats in a second (average number of frames) is found.
    float avg_fps = frame_no / (time_elapsed / 1000.f);

    // fpms = 16.66 (miliseconds allocated for 1 frame (one iteration of the while loop))
    // time_start_fps value is reset at the beginning of each iteration of the while loop.
    // The time elapsed since the beginning of the last while loop execution (miliseconds) - time_start_fps = SDL_GetTicks()
    float fpms = (float) 1000 / fps;
    int time_passed_ms = SDL_GetTicks() - time_start_fps;

    if(fps_fixed) {
       if(time_passed_ms < fpms) { // 16.66
          // Wait until the elapsed time reaches 16.66
          SDL_Delay(fpms - time_passed_ms);
       }
    }

    char cdizi[200];
    sprintf(cdizi, "%.2d:%.2d:%.4d - Frame: %d - FPS: %.2f - Vsync: %s - Fixed FPS: %s - Time move: %s",
                    minutes, seconds, milliseconds, frame_no, avg_fps,
                    vsync==1 ? "On" : "Off", fps_fixed ? "On" : "Off", delta_move ? "On" : "Off");
    SDL_SetWindowTitle(window, cdizi);

    frame_no++;
}

// Destroy Renderer and SDL window, exit from SDL3
void destroy_window(void)
{
    SDL_DestroyTexture(texture);
    SDL_DestroyRenderer(renderer);
    SDL_DestroyWindow(window);
    SDL_Quit();
}