aboutsummaryrefslogblamecommitdiffstats
path: root/src/renderer.c
blob: c742e18b0d96205a570be87483c7b6ea179b238a (plain) (tree)



















































































































































































































































                                                                           
/*
 * Copyright (c) 2019-2020, yzrh <yzrh@noema.org>
 *
 * SPDX-License-Identifier: Apache-2.0
 */

#include <stdbool.h>

#include <SDL2/SDL.h>
#include <SDL2/SDL_ttf.h>

#include "renderer.h"
#include "screen.h"
#include "game.h"

void
draw_clear(SDL_Renderer *renderer)
{
	SDL_SetRenderDrawColor(renderer,
		COLOUR_BACKGROUND_R,
		COLOUR_BACKGROUND_G,
		COLOUR_BACKGROUND_B,
		255);
	SDL_RenderClear(renderer);
}

void
draw_colour(SDL_Renderer *renderer, Snake_Colour colour)
{
	int r;
	int g;
	int b;

	switch (colour) {
		case SNAKE_FOREGROUND:
			r = COLOUR_FOREGROUND_R;
			g = COLOUR_FOREGROUND_G;
			b = COLOUR_FOREGROUND_B;
			break;
		case SNAKE_BACKGROUND:
			r = COLOUR_BACKGROUND_R;
			g = COLOUR_BACKGROUND_G;
			b = COLOUR_BACKGROUND_B;
			break;
		case SNAKE_BACKGROUND_SHADE:
			r = COLOUR_BACKGROUND_SHADE_R;
			g = COLOUR_BACKGROUND_SHADE_G;
			b = COLOUR_BACKGROUND_SHADE_B;
			break;
		case SNAKE_RED:
			r = COLOUR_RED_R;
			g = COLOUR_RED_G;
			b = COLOUR_RED_B;
			break;
		case SNAKE_GREEN:
			r = COLOUR_GREEN_R;
			g = COLOUR_GREEN_G;
			b = COLOUR_GREEN_B;
			break;
		case SNAKE_BLUE:
			r = COLOUR_RED_R;
			g = COLOUR_RED_G;
			b = COLOUR_RED_B;
			break;
		case SNAKE_YELLOW:
			r = COLOUR_RED_R;
			g = COLOUR_RED_G;
			b = COLOUR_RED_B;
			break;
		case SNAKE_MAGENTA:
			r = COLOUR_RED_R;
			g = COLOUR_RED_G;
			b = COLOUR_RED_B;
			break;
		case SNAKE_CYAN:
			r = COLOUR_RED_R;
			g = COLOUR_RED_G;
			b = COLOUR_RED_B;
			break;
		case SNAKE_WHITE:
			r = COLOUR_RED_R;
			g = COLOUR_RED_G;
			b = COLOUR_RED_B;
			break;
		case SNAKE_BLACK:
			r = COLOUR_RED_R;
			g = COLOUR_RED_G;
			b = COLOUR_RED_B;
			break;
		default:
			break;
	}

	SDL_SetRenderDrawColor(renderer,
		r,
		g,
		b,
		255);
}

void
draw_text(SDL_Renderer *renderer, TTF_Font *font,
	int text_x, int text_y, char *text,
	bool centre, bool select, SDL_Color fg, SDL_Color bg,
	Snake_Text *message)
{
	if (!message->cache) {
		if (message->texture != NULL)
			SDL_DestroyTexture(message->texture);

		SDL_Surface *surface =
			TTF_RenderUTF8_Shaded(font, text, fg, bg);
		message->texture =
			SDL_CreateTextureFromSurface(renderer, surface);
		SDL_FreeSurface(surface);

		message->pos.x = text_x;
		message->pos.y = text_y;
		TTF_SizeText(font, text, &message->pos.w, &message->pos.h);
		if (centre)
			message->pos.x -= message->pos.w / 2;
		message->pos.y -= message->pos.h;

		message->cache = 1;
	}

	SDL_RenderCopy(renderer, message->texture, NULL, &message->pos);

	if (select) {
		SDL_SetRenderDrawColor(renderer,
			COLOUR_FOREGROUND_R,
			COLOUR_FOREGROUND_G,
			COLOUR_FOREGROUND_B,
			255);
		SDL_RenderDrawLine(renderer,
			message->pos.x, message->pos.y,
			message->pos.x, message->pos.y + message->pos.h);
		SDL_RenderDrawLine(renderer,
			message->pos.x, message->pos.y,
			message->pos.x + message->pos.w, message->pos.y);
		SDL_RenderDrawLine(renderer,
			message->pos.x, message->pos.y + message->pos.h,
			message->pos.x + message->pos.w,
				message->pos.y + message->pos.h);
		SDL_RenderDrawLine(renderer,
			message->pos.x + message->pos.w, message->pos.y,
			message->pos.x + message->pos.w,
				message->pos.y + message->pos.h);
	}
}

void
draw_wall(SDL_Renderer *renderer)
{
	SDL_RenderDrawLine(renderer,
		0, 0,
		0, SCREEN_HEIGHT);
	SDL_RenderDrawLine(renderer,
		0, 0,
		SCREEN_HEIGHT, 0);
	SDL_RenderDrawLine(renderer,
		0, SCREEN_HEIGHT - 1,
		SCREEN_HEIGHT, SCREEN_HEIGHT - 1);
	SDL_RenderDrawLine(renderer,
		SCREEN_HEIGHT, 0,
		SCREEN_HEIGHT, SCREEN_HEIGHT);
}

void
draw_snake(SDL_Renderer *renderer, Snake_Pos **snake)
{
	Snake_Pos *ptr = *snake;

	SDL_Rect pos;
	while (ptr != NULL) {
		pos.x = ptr->x;
		pos.y = ptr->y;
		pos.w = SCREEN_UNIT;
		pos.h = SCREEN_UNIT;
		SDL_RenderFillRect(renderer, &pos);
		ptr = ptr->next;
	}
}

void
draw_fruit(SDL_Renderer *renderer, int *fruit)
{
	int rx = fruit[0] + SCREEN_UNIT / 2;
	int ry = fruit[1] + SCREEN_UNIT / 2;

	/* Midpoint circle algorithm */
	int x = SCREEN_UNIT / 2;
	int y = 0;
	int dx = 1;
	int dy = 1;
	int err = dx - ((SCREEN_UNIT / 2) << 1);
	while (x >= y) {
		SDL_RenderDrawLine(renderer,
			rx - y, ry + x,
			rx + y, ry + x);
		SDL_RenderDrawLine(renderer,
			rx - x, ry + y,
			rx + x, ry + y);
		SDL_RenderDrawLine(renderer,
			rx - x, ry - y,
			rx + x, ry - y);
		SDL_RenderDrawLine(renderer,
			rx - y, ry - x,
			rx + y, ry - x);

		if (err <= 0) {
			y++;
			err += dy;
			dy += 2;
		}

		if (err > 0) {
			x--;
			dx += 2;
			err += dx - ((SCREEN_UNIT / 2) << 1);
		}
	}
}

void
draw_pipe(SDL_Renderer *renderer, Snake_Pos **pipe)
{
	Snake_Pos *ptr = *pipe;

	SDL_Rect pos;
	while (ptr != NULL) {
		pos.x = ptr->x;
		pos.y = 0;
		pos.w = SCREEN_UNIT;
		if (ptr->x + SCREEN_UNIT > SCREEN_HEIGHT)
			pos.w -= ptr->x + SCREEN_UNIT - SCREEN_HEIGHT;
		pos.h = ptr->y - PIPE_GAP / 2 * SCREEN_UNIT;
		SDL_RenderFillRect(renderer, &pos);
		pos.y = ptr->y + PIPE_GAP / 2 * SCREEN_UNIT;
		pos.h = SCREEN_HEIGHT - ptr->y - SCREEN_UNIT;
		SDL_RenderFillRect(renderer, &pos);
		ptr = ptr->next;
	}
}