Mercurial > molko
view libcore/core/game.c @ 121:789b23e01f52
misc: reorganize hierarchy, closes #2490
author | David Demelier <markand@malikania.fr> |
---|---|
date | Mon, 05 Oct 2020 13:25:06 +0200 |
parents | src/core/game.c@445035ace840 |
children | 09978921e281 |
line wrap: on
line source
/* * game.c -- main game object * * Copyright (c) 2020 David Demelier <markand@malikania.fr> * * Permission to use, copy, modify, and/or distribute this software for any * purpose with or without fee is hereby granted, provided that the above * copyright notice and this permission notice appear in all copies. * * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. */ #include <assert.h> #include <stddef.h> #include <string.h> #include "game.h" #include "state.h" #include "painter.h" struct game game; static struct action * find_empty_action(void) { static struct action null; for (struct action *a = game.actions; a != &game.actions[GAME_ACTIONS_MAX]; ++a) if (memcmp(a, &null, sizeof (struct action)) == 0) return a; return NULL; } static void clear_actions(void) { for (struct action *a = game.actions; a != &game.actions[GAME_ACTIONS_MAX]; ++a) { /* These actions are removed on state change. */ if (a->flags & ACTION_AUTO_LEAVE) { if (a->finish) a->finish(a); memset(a, 0, sizeof (struct action)); } } } static void handle_actions(const union event *event) { for (struct action *a = game.actions; a != &game.actions[GAME_ACTIONS_MAX]; ++a) if (a->handle) a->handle(a, event); } static void update_actions(unsigned int ticks) { for (size_t i = 0; i < GAME_ACTIONS_MAX; ++i) { struct action *a = &game.actions[i]; if (!a->update) continue; if (a->update(a, ticks)) { if (a->end) a->end(a); if (a->finish) a->finish(a); memset(&game.actions[i], 0, sizeof (struct action)); } } } static void draw_actions(void) { for (size_t i = 0; i < GAME_ACTIONS_MAX; ++i) if (game.actions[i].draw) game.actions[i].draw(&game.actions[i]); } void game_switch(struct state *state, bool quick) { assert(state); if (quick) { game.state = state; game.state->enter(); } else game.state_next = state; } void game_handle(const union event *event) { assert(event); if (game.state && !(game.inhibit & INHIBIT_STATE_INPUT)) game.state->handle(event); handle_actions(event); } void game_update(unsigned int ticks) { if (!(game.inhibit & INHIBIT_STATE_UPDATE)) { /* Change state if any. */ if (game.state_next) { /* Inform the current state we're gonna leave it. */ if (game.state) game.state->leave(); game.state = game.state_next; game.state->enter(); game.state_next = NULL; /* Remove any actions that must be deleted. */ clear_actions(); } if (game.state) game.state->update(ticks); } update_actions(ticks); } void game_draw(void) { if (game.state && !(game.inhibit & INHIBIT_STATE_DRAW)) game.state->draw(); draw_actions(); painter_present(); } void game_add_action(const struct action *action) { assert(action); struct action *pos; if ((pos = find_empty_action())) memcpy(pos, action, sizeof (struct action)); } void game_quit(void) { if (game.state && game.state->leave) game.state->leave(); game.state = NULL; }