starship/game.c

147 lines
4.1 KiB
C
Raw Normal View History

2025-03-01 05:58:12 -05:00
// Copyright 2025 Bill Rossi
//
// This file is part of Starship Futuretime
//
// Starship Futuretime is free software: you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version.
//
// Starship Futuretime is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details.
//
// You should have received a copy of the GNU General Public License along with Starship Futuretime. If not, see <https://www.gnu.org/licenses/>.
#include <stdlib.h>
2025-03-01 11:34:51 -05:00
#include <stdio.h>
2025-03-01 05:58:12 -05:00
#include <stdbool.h>
#include <raylib.h>
2025-03-01 08:53:38 -05:00
2025-03-01 05:58:12 -05:00
#include "game.h"
2025-03-01 08:53:38 -05:00
#include "enemy.h"
2025-03-01 09:03:41 -05:00
#include "bullet.h"
2025-03-01 08:53:38 -05:00
2025-03-01 11:19:00 -05:00
#define FOREACH_ENEMY for (size_t i = 0; i < g->enemies->count; i++)
#define INIT_ENEMY Entity *e = g->enemies->entities[i]
#define FOREACH_BULLET for (size_t i = 0; i < g->bullets->count; i++)
#define INIT_BULLET Entity *e = g->bullets->entities[i]
2025-03-01 05:58:12 -05:00
void initialize_game(Game *g) {
g->should_close = false;
g->player = malloc(sizeof(Player));
2025-03-08 10:25:24 -05:00
initialize_player(g, g->player);
2025-03-01 11:19:00 -05:00
g->enemies = malloc(sizeof(Entities));
g->enemies->entities = malloc(10 * sizeof(Entity*));
g->enemies->count = 0;
g->enemies->capacity = 10;
g->bullets = malloc(sizeof(Entities));
g->bullets->entities = malloc(10 * sizeof(Entity*));
g->bullets->count = 0;
g->bullets->capacity = 10;
2025-03-08 10:08:44 -05:00
2025-03-08 10:25:24 -05:00
g->camera = malloc(sizeof(Camera));
g->camera->zoom = 1.;
g->camera->rotation = 0.;
2025-03-08 10:08:44 -05:00
g->level = malloc(sizeof(Level));
2025-03-08 13:22:05 -05:00
init_level(g, g->level, "levels/test_level.csv");
2025-03-08 13:38:45 -05:00
g->player->position.y = g->level->length * 32;
2025-03-01 08:53:38 -05:00
}
2025-03-01 11:19:00 -05:00
void add_entity(Entities *entities, Entity *e) {
if (entities->count >= entities->capacity) {
entities->capacity *= 2;
entities->entities = realloc(entities->entities, entities->capacity * sizeof(Entity*));
2025-03-01 08:53:38 -05:00
}
2025-03-01 11:19:00 -05:00
entities->entities[entities->count++] = e;
2025-03-01 05:58:12 -05:00
}
2025-03-01 11:34:51 -05:00
void remove_entity(Entities *entities, Entity *e) {
size_t i;
for (i = 0; entities->entities[i] != e; i++) {
if (i > entities->count) {
printf("Couldn't find entity %s\n", e->name);
return;
}
}
for (; i < entities->count - 1; i++) {
entities->entities[i] = entities->entities[i + 1];
}
entities->count--;
}
2025-03-01 05:58:12 -05:00
void handle_input(Game *g) {
handle_player_input(g->player);
2025-03-01 08:53:38 -05:00
if (IsKeyPressed(KEY_S)) {
2025-03-01 11:19:00 -05:00
add_entity(g->enemies, spawn_enemy());
2025-03-01 08:53:38 -05:00
}
2025-03-01 09:03:41 -05:00
if (IsKeyPressed(KEY_SPACE)) {
2025-03-08 10:34:37 -05:00
add_entity(g->bullets, spawn_bullet(g->player->position, g->player->velocity, false));
2025-03-01 09:03:41 -05:00
}
2025-03-01 05:58:12 -05:00
}
2025-03-01 11:40:23 -05:00
void process_bullet_collisions(Game *g, Entity *bullet) {
2025-03-02 18:58:52 -05:00
BulletProperties *bullet_props = bullet->properties;
Rectangle bullet_rec = bullet_props->position;
if (bullet_props->from_enemy) {
if (CheckCollisionPointRec(g->player->position, bullet_rec)) {
2025-03-01 11:40:23 -05:00
remove_entity(g->bullets, bullet);
2025-03-02 18:58:52 -05:00
}
} else {
FOREACH_ENEMY {
INIT_ENEMY;
Rectangle enemy_rec = ((EnemyProperties*)e->properties)->position;
if (CheckCollisionRecs(bullet_rec, enemy_rec)) {
remove_entity(g->bullets, bullet);
remove_entity(g->enemies, e);
}
2025-03-01 11:40:23 -05:00
}
}
}
2025-03-01 05:58:12 -05:00
void run_frame(Game *g) {
2025-03-09 10:49:50 -04:00
float dt = GetFrameTime();
2025-03-01 05:58:12 -05:00
handle_input(g);
2025-03-09 10:49:50 -04:00
tick_player(g->player, dt);
2025-03-01 11:19:00 -05:00
FOREACH_ENEMY {
INIT_ENEMY;
2025-03-01 11:34:51 -05:00
if (entity_expired(e)) {
remove_entity(g->enemies, e);
continue;
}
2025-03-02 18:58:52 -05:00
e->tick(g, e, GetFrameTime());
2025-03-01 11:19:00 -05:00
}
FOREACH_BULLET {
INIT_BULLET;
2025-03-02 18:58:52 -05:00
2025-03-01 11:34:51 -05:00
if (entity_expired(e)) {
remove_entity(g->bullets, e);
continue;
}
2025-03-02 18:58:52 -05:00
e->tick(g, e, GetFrameTime());
2025-03-01 11:40:23 -05:00
process_bullet_collisions(g, e);
2025-03-01 08:53:38 -05:00
}
2025-03-01 05:58:12 -05:00
}
void draw_frame(Game *g) {
BeginDrawing();
ClearBackground(RAYWHITE);
2025-03-08 10:25:24 -05:00
BeginMode2D(*g->camera);
2025-03-08 10:08:44 -05:00
draw_level(g->level);
2025-03-01 05:58:12 -05:00
draw_player(g->player);
2025-03-01 11:19:00 -05:00
FOREACH_ENEMY {
INIT_ENEMY;
e->draw(e);
}
FOREACH_BULLET {
INIT_BULLET;
2025-03-01 08:53:38 -05:00
e->draw(e);
}
2025-03-08 10:25:24 -05:00
EndMode2D();
2025-03-01 05:58:12 -05:00
EndDrawing();
}
void run_until_closing(Game *g) {
while (!WindowShouldClose() && !g->should_close) {
run_frame(g);
draw_frame(g);
}
}