8 #include "./renderer.h"
19 static vec_t effective_ratio(const SDL_Rect *view_port);
20 static vec_t effective_scale(const SDL_Rect *view_port);
21 static vec_t camera_point(const camera_t *camera,
22 const SDL_Rect *view_port,
24 static rect_t camera_rect(const camera_t *camera,
25 const SDL_Rect *view_port,
27 static triangle_t camera_triangle(const camera_t *camera,
28 const SDL_Rect *view_port,
31 camera_t *create_camera(point_t position)
33 camera_t *camera = malloc(sizeof(camera_t));
36 throw_error(ERROR_TYPE_LIBC);
40 camera->position = position;
41 camera->debug_mode = 0;
42 camera->blackwhite_mode = 0;
47 void destroy_camera(camera_t *camera)
55 int camera_fill_rect(const camera_t *camera,
64 SDL_RenderGetViewport(render, &view_port);
66 const SDL_Rect sdl_rect = rect_for_sdl(
67 camera_rect(camera, &view_port, rect));
69 const SDL_Color sdl_color = color_for_sdl(camera->blackwhite_mode ? color_desaturate(color) : color);
71 if (SDL_SetRenderDrawColor(render, sdl_color.r, sdl_color.g, sdl_color.b, sdl_color.a) < 0) {
72 throw_error(ERROR_TYPE_SDL2);
76 if (camera->debug_mode) {
77 if (SDL_RenderDrawRect(render, &sdl_rect) < 0) {
78 throw_error(ERROR_TYPE_SDL2);
82 if (SDL_RenderFillRect(render, &sdl_rect) < 0) {
83 throw_error(ERROR_TYPE_SDL2);
91 int camera_draw_rect(const camera_t * camera,
100 SDL_RenderGetViewport(render, &view_port);
102 const SDL_Rect sdl_rect = rect_for_sdl(
103 camera_rect(camera, &view_port, rect));
105 const SDL_Color sdl_color = color_for_sdl(camera->blackwhite_mode ? color_desaturate(color) : color);
107 if (SDL_SetRenderDrawColor(render, sdl_color.r, sdl_color.g, sdl_color.b, sdl_color.a) < 0) {
108 throw_error(ERROR_TYPE_SDL2);
112 if (SDL_RenderDrawRect(render, &sdl_rect) < 0) {
113 throw_error(ERROR_TYPE_SDL2);
120 int camera_draw_triangle(const camera_t *camera,
121 SDL_Renderer *render,
129 SDL_RenderGetViewport(render, &view_port);
131 const SDL_Color sdl_color = color_for_sdl(camera->blackwhite_mode ? color_desaturate(color) : color);
133 if (SDL_SetRenderDrawColor(render, sdl_color.r, sdl_color.g, sdl_color.b, sdl_color.a) < 0) {
134 throw_error(ERROR_TYPE_SDL2);
138 if (draw_triangle(render, camera_triangle(camera, &view_port, t)) < 0) {
145 int camera_fill_triangle(const camera_t *camera,
146 SDL_Renderer *render,
154 SDL_RenderGetViewport(render, &view_port);
156 const SDL_Color sdl_color = color_for_sdl(camera->blackwhite_mode ? color_desaturate(color) : color);
158 if (SDL_SetRenderDrawColor(render, sdl_color.r, sdl_color.g, sdl_color.b, sdl_color.a) < 0) {
159 throw_error(ERROR_TYPE_SDL2);
163 if (fill_triangle(render, camera_triangle(camera, &view_port, t)) < 0) {
170 int camera_clear_background(const camera_t *camera,
171 SDL_Renderer *render,
174 const SDL_Color sdl_color = color_for_sdl(camera->blackwhite_mode ? color_desaturate(color) : color);
176 if (SDL_SetRenderDrawColor(render, sdl_color.r, sdl_color.g, sdl_color.b, sdl_color.a) < 0) {
177 throw_error(ERROR_TYPE_SDL2);
181 if (SDL_RenderClear(render) < 0) {
182 throw_error(ERROR_TYPE_SDL2);
189 void camera_center_at(camera_t *camera, point_t position)
192 camera->position = position;
195 void camera_toggle_debug_mode(camera_t *camera)
198 camera->debug_mode = !camera->debug_mode;
201 void camera_toggle_blackwhite_mode(camera_t *camera)
204 camera->blackwhite_mode = !camera->blackwhite_mode;
207 int camera_is_point_visible(const camera_t *camera, SDL_Renderer *renderer, point_t p)
210 SDL_RenderGetViewport(renderer, &view_port);
212 return rect_contains_point(
213 rect_from_sdl(&view_port),
214 camera_point(camera, &view_port, p));
217 /* ---------- Private Function ---------- */
219 static vec_t effective_ratio(const SDL_Rect *view_port)
221 if ((float) view_port->w / RATIO_X > (float) view_port->h / RATIO_Y) {
222 return vec(RATIO_X, (float) view_port->h / ((float) view_port->w / RATIO_X));
224 return vec((float) view_port->w / ((float) view_port->h / RATIO_Y), RATIO_Y);
228 static vec_t effective_scale(const SDL_Rect *view_port)
230 return vec_entry_div(
231 vec((float) view_port->w, (float) view_port->h),
232 vec_scala_mult(effective_ratio(view_port), 50.0f));
235 static vec_t camera_point(const camera_t *camera,
236 const SDL_Rect *view_port,
242 vec_sum(p, vec_neg(camera->position)),
243 effective_scale(view_port)),
244 vec((float) view_port->w * 0.5f,
245 (float) view_port->h * 0.5f));
248 static triangle_t camera_triangle(const camera_t *camera,
249 const SDL_Rect *view_port,
253 camera_point(camera, view_port, t.p1),
254 camera_point(camera, view_port, t.p2),
255 camera_point(camera, view_port, t.p3));
258 static rect_t camera_rect(const camera_t *camera,
259 const SDL_Rect *view_port,
262 return rect_from_vecs(
266 vec(rect.x, rect.y)),
268 effective_scale(view_port),
269 vec(rect.w, rect.h)));