breakhack/src/particle_engine.c

241 lines
5.0 KiB
C
Raw Normal View History

2018-02-16 18:11:26 +01:00
/*
* BreakHack - A dungeone crawler RPG
* Copyright (C) 2018 Linus Probert <linus.probert@gmail.com>
*
* This program 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.
*
* This program 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 this program. If not, see <http://www.gnu.org/licenses/>.
*/
2018-02-03 23:39:49 +01:00
#include <stdlib.h>
2018-02-19 15:55:37 +01:00
#include <stdarg.h>
2018-02-03 23:39:49 +01:00
#include "particle_engine.h"
#include "linkedlist.h"
#include "util.h"
#include "defines.h"
#include "vector2d.h"
#include "random.h"
2018-02-03 23:39:49 +01:00
typedef struct Particle_t {
Position pos;
Vector2d velocity;
Dimension dim;
unsigned int movetime;
unsigned int lifetime;
SDL_Color color;
} Particle;
typedef struct Engine_t {
LinkedList *particles;
} Engine;
static Engine *engine = NULL;
static void
check_engine(void)
{
if (!engine)
fatal("Particle engine not initiated");
}
void
particle_engine_init(void)
{
if (engine != NULL)
fatal("Engine already initiated");
engine = ec_malloc(sizeof(Engine));
engine->particles = linkedlist_create();
}
void
2018-02-09 13:55:57 +01:00
particle_engine_bloodspray(Position pos, Dimension dim, unsigned int count)
2018-02-03 23:39:49 +01:00
{
check_engine();
2018-02-09 13:55:57 +01:00
if (count > 100)
count = 100;
for (unsigned int i = 0; i < count; ++i) {
2018-02-03 23:46:17 +01:00
int x, y, xv, yv, w, h;
unsigned int mt, lt;
Particle *p;
x = get_random(dim.width) + pos.x;
y = get_random(dim.height) + pos.y;
2018-02-03 23:39:49 +01:00
xv = get_random(200) - 100;
yv = get_random(200) - 100;
2018-02-03 23:39:49 +01:00
mt = get_random(10) + 10;
lt = get_random(120) + 60;
2018-02-03 23:39:49 +01:00
w = get_random(3) + 2;
h = get_random(3) + 2;
2018-02-03 23:39:49 +01:00
p = ec_malloc(sizeof(Particle));
p->pos = (Position) { x, y };
p->velocity = (Vector2d) { xv, yv };
p->movetime = mt;
p->lifetime = lt;
p->dim = (Dimension) { w, h };
p->color = (SDL_Color) { 255, 0, 0, 255 };
linkedlist_append(&engine->particles, p);
}
}
2018-02-19 15:55:37 +01:00
static void
create_explosion(Position pos, Dimension dim, size_t c_count, ...)
{
SDL_Color *colors = ec_malloc(c_count * sizeof(SDL_Color));
va_list(args);
va_start(args, c_count);
for (size_t i = 0; i < c_count; ++i) {
colors[i] = va_arg(args, SDL_Color);
}
va_end(args);
for (unsigned int i = 0; i < 150; ++i) {
int x, y, xv, yv;
unsigned int lt;
Particle *p;
x = get_random(dim.width) + pos.x;
y = get_random(dim.height) + pos.y;
xv = get_random(600) - 300;
yv = get_random(600) - 300;
lt = get_random(10);
p = ec_malloc(sizeof(Particle));
p->pos = (Position) { x, y };
p->velocity = (Vector2d) { xv, yv };
p->movetime = lt;
p->lifetime = lt;
p->dim = (Dimension) { 2, 2 };
p->color = colors[get_random(c_count-1)];
linkedlist_append(&engine->particles, p);
}
}
void
particle_engine_fire_explosion(Position pos, Dimension dim)
{
static SDL_Color red = { 255, 0, 0, 255 };
static SDL_Color yellow = { 255, 255, 0, 255 };
check_engine();
create_explosion(pos, dim, 3, yellow, yellow, red);
}
void
particle_engine_eldritch_explosion(Position pos, Dimension dim)
{
static SDL_Color green = { 0, 255, 0, 255 };
check_engine();
create_explosion(pos, dim, 1, green);
}
2018-02-03 23:39:49 +01:00
static void
move_particle(Particle *particle, float deltaTime)
{
if (!particle->movetime)
return;
particle->pos.x += particle->velocity.x * deltaTime;
particle->pos.y += particle->velocity.y * deltaTime;
}
void
particle_engine_update(float deltaTime)
{
check_engine();
LinkedList *current, *last;
current = engine->particles;
last = NULL;
while (current) {
Particle *particle = current->data;
2018-02-03 23:39:49 +01:00
if (particle->movetime)
particle->movetime--;
if (particle->lifetime > 0) {
particle->lifetime--;
move_particle(current->data, deltaTime);
last = current;
current = current->next;
} else {
if (!last) {
engine->particles = current->next;
free(current->data);
free(current);
current = engine->particles;
} else {
last->next = current->next;
free(current->data);
free(current);
current = last->next;
}
}
}
}
static void
render_particle(Particle *p, Camera *cam)
{
Position pos = camera_to_camera_position(cam, &p->pos);
SDL_Rect box = { pos.x, pos.y, p->dim.width, p->dim.height };
SDL_SetRenderDrawColor(cam->renderer,
p->color.r,
p->color.g,
2018-02-19 15:55:37 +01:00
p->color.b,
2018-02-03 23:39:49 +01:00
p->color.a);
SDL_RenderFillRect(cam->renderer, &box);
}
void
particle_engine_render(Camera *cam)
{
check_engine();
LinkedList *particles = engine->particles;
while (particles) {
render_particle(particles->data, cam);
particles = particles->next;
}
}
void
particle_engine_clear(void)
{
check_engine();
while (engine->particles)
free(linkedlist_pop(&engine->particles));
}
2018-02-03 23:39:49 +01:00
void
particle_engine_close(void)
{
check_engine();
while (engine->particles)
free(linkedlist_pop(&engine->particles));
free(engine);
engine = NULL;
}