/*
* Copyright (C) 2021 P. J. McDermott
*
* This file is part of Dodge Balls
*
* Dodge Balls 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.
*
* Dodge Balls 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 Dodge Balls. If not, see .
*/
#include
#include
#include "ball.h"
#include "collision.h"
#include "map.h"
#include "tileset.h"
#include "output.h"
struct db_ball {
int cx;
int cy;
double x;
double y;
int r;
double a;
int d;
int sr;
double s;
struct db_tileset *tilesets;
int gid;
struct db_map *map;
struct db_ball *next;
};
struct db_ball *
db_ball_new(int x, int y, int r, int a, int d, int sr, double s,
struct db_tileset *tilesets, int gid, struct db_map *map,
struct db_ball *prev)
{
struct db_ball *ball;
ball = calloc(1, sizeof(*ball));
if (ball == NULL) {
db_err("Failed to allocate memory");
return NULL;
}
db_dbg("Ball at (%d,%d)", x, y);
if (sr == 0) {
ball->x = x;
ball->y = y;
} else {
ball->cx = x;
ball->cy = y;
ball->x = x + cos(a * (M_PI / 180)) * sr;
ball->y = y + sin(a * (M_PI / 180)) * sr;
}
ball->r = r;
ball->a = a;
ball->d = d;
ball->sr = sr;
ball->s = s;
ball->tilesets = tilesets;
ball->gid = gid;
ball->map = map;
if (prev != NULL) {
prev->next = ball;
}
return ball;
}
void
db_balls_move(struct db_ball *ball)
{
if (ball->sr == 0) {
ball->x += cos(ball->a * (M_PI / 180));
ball->y += sin(ball->a * (M_PI / 180));
} else {
ball->a = fmod(ball->a + ball->d * ball->s, 360);
ball->x = ball->cx + cos(ball->a * (M_PI / 180)) * ball->sr;
ball->y = ball->cy + sin(ball->a * (M_PI / 180)) * ball->sr;
db_dbg("Spinning ball angle %f (speed %f, direction %d)",
ball->a, ball->s, ball->d);
}
if (ball->next != NULL) {
db_balls_move(ball->next);
}
}
static void
_db_ball_bounce(struct db_ball *ball, double col_x, double col_y)
{
double col_a;
col_a = atan2(col_y - ball->y, col_x - ball->x) * 180 / M_PI;
ball->a = fmod(2 * col_a - ball->a - 180, 360);
}
void
db_balls_collisions(struct db_ball *ball)
{
SDL_Rect wall;
int map_tw;
int map_th;
int x_min;
int x_max;
int y_min;
int y_max;
SDL_bool bounce;
int x;
int y;
double col_x;
double col_y;
struct db_ball *other;
wall.w = map_tw = db_map_get_tilewidth (ball->map);
wall.h = map_th = db_map_get_tileheight(ball->map);
x_min = floor((ball->x - ball->r) / map_tw);
x_max = ceil ((ball->x + ball->r) / map_tw);
y_min = floor((ball->y - ball->r) / map_th);
y_max = ceil ((ball->y + ball->r) / map_th);
bounce = SDL_FALSE;
for (y = y_min; y < y_max; ++y) {
for (x = x_min; x < x_max; ++x) {
if (db_map_tile_ball_collides(ball->map, x, y)) {
wall.x = x * map_tw;
wall.y = y * map_th;
if (db_col_cir_rect(ball->x, ball->y, ball->r,
&wall, &col_x, &col_y)){
bounce = SDL_TRUE;
}
}
}
}
if (bounce) {
_db_ball_bounce(ball, col_x, col_y);
}
/* Spinning balls shouldn't collide with any other balls */
if (ball->sr == 0) {
for (other = ball->next; other != NULL; other = other->next) {
if (other->sr == 0 && db_col_pt_cir_cir(
ball->x, ball->y, ball->r,
other->x, other->y, other->r,
&col_x, &col_y)) {
_db_ball_bounce(ball , col_x, col_y);
_db_ball_bounce(other, col_x, col_y);
}
}
}
if (ball->next != NULL) {
db_balls_collisions(ball->next);
}
}
int
db_balls_player_collisions(struct db_ball *ball,
int player_x, int player_y, int player_r)
{
if (db_col_cir_cir(ball->x, ball->y, ball->r,
player_x, player_y, player_r)) {
return 1;
}
if (ball->next != NULL) {
return db_balls_player_collisions(ball->next,
player_x, player_y, player_r);
} else {
return 0;
}
}
void
db_balls_render(struct db_ball *ball, SDL_Renderer *renderer)
{
SDL_Rect dstrect;
dstrect.x = ball->x - ball->r;
dstrect.y = ball->y - ball->r;
dstrect.w = ball->r * 2;
dstrect.h = ball->r * 2;
db_dbg("Rendering ball (%dx%d) at (%d,%d)",
dstrect.w, dstrect.h, dstrect.x, dstrect.y);
db_tile_render(ball->tilesets, renderer, ball->gid, &dstrect);
if (ball->next != NULL) {
db_balls_render(ball->next, renderer);
}
}