Hash :
1ae6ad0e
Author :
Thomas de Grivel
Date :
2024-07-15T19:59:52
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122
/* kc3
* Copyright 2022,2023,2024 kmx.io <contact@kmx.io>
*
* Permission is hereby granted to use this software granted the above
* copyright notice and this permission paragraph are included in all
* copies and substantial portions of this software.
*
* THIS SOFTWARE IS PROVIDED "AS-IS" WITHOUT ANY GUARANTEE OF
* PURPOSE AND PERFORMANCE. IN NO EVENT WHATSOEVER SHALL THE
* AUTHOR BE CONSIDERED LIABLE FOR THE USE AND PERFORMANCE OF
* THIS SOFTWARE.
*/
#include <math.h>
#include <libkc3/kc3.h>
#include "../window_sdl2.h"
#include "../gl_camera.h"
#include "../mat4.h"
#include "../gl_object.h"
#include "../gl_sphere.h"
#include "../gl_sprite.h"
#include "earth.h"
#define EARTH_CAMERA_ROTATION_Z_SPEED 0.1
#define EARTH_SEGMENTS_U 200
#define EARTH_SEGMENTS_V 100
s_gl_sprite g_sprite_earth = {0};
bool earth_load (s_sequence *seq)
{
s_map *map;
s_mat4 matrix;
s_gl_camera *camera;
s_gl_sphere *sphere;
const f32 sphere_radius = 5.0;
s_window_sdl2 *window;
assert(seq);
window = seq->window;
assert(window);
camera = gl_camera_new(window->w, window->h);
if (! camera)
return false;
sphere = gl_sphere_new(EARTH_SEGMENTS_U, EARTH_SEGMENTS_V);
if (! sphere)
return false;
mat4_init_scale(&matrix, sphere_radius, sphere_radius,
sphere_radius);
gl_object_transform(&sphere->object, &matrix);
gl_object_update(&sphere->object);
if (! tag_map(&seq->tag, 3))
return false;
map = &seq->tag.data.map;
tag_init_sym( map->key + 0, sym_1("camera"));
tag_init_ptr_free( map->value + 0, camera);
tag_init_sym( map->key + 1, sym_1("camera_rot_x_speed"));
tag_init_f64( map->value + 1, 0.01);
tag_init_sym( map->key + 2, sym_1("sphere"));
tag_init_struct_with_data(map->value + 2, sym_1("GL.Sphere"),
sphere);
return true;
}
bool earth_render (s_sequence *seq)
{
s_gl_camera *camera;
f64 *camera_rot_x_speed;
s_map *map;
s_gl_sphere *sphere;
s_window_sdl2 *window;
assert(seq);
window = seq->window;
assert(window);
if (! seq || seq->tag.type != TAG_MAP ||
seq->tag.data.map.count != 3) {
err_puts("earth_render: invalid seq->tag");
return false;
}
map = &seq->tag.data.map;
if (map->value[0].type != TAG_PTR_FREE ||
map->value[1].type != TAG_F64 ||
map->value[2].type != TAG_STRUCT) {
err_puts("earth_render: invalid map");
return false;
}
camera = map->value[0].data.ptr_free.p;
camera_rot_x_speed = &map->value[1].data.f64;
sphere = map->value[2].data.struct_.data;
gl_camera_set_aspect_ratio(camera, window->w, window->h);
camera->rotation.x += seq->dt * (*camera_rot_x_speed) *
M_PI * 2.0f;
if (camera->rotation.x > M_PI || camera->rotation.x < 0)
*camera_rot_x_speed *= -1.0;
camera->rotation.z += seq->dt * EARTH_CAMERA_ROTATION_Z_SPEED *
M_PI * 2.0f;
assert(glGetError() == GL_NO_ERROR);
gl_camera_render(camera);
assert(glGetError() == GL_NO_ERROR);
glClearColor(0.0f, 0.0f, 0.0f, 1.0f);
assert(glGetError() == GL_NO_ERROR);
glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT);
assert(glGetError() == GL_NO_ERROR);
assert(glGetError() == GL_NO_ERROR);
assert(glGetError() == GL_NO_ERROR);
assert(glGetError() == GL_NO_ERROR);
glEnable(GL_DEPTH_TEST);
assert(glGetError() == GL_NO_ERROR);
gl_camera_bind_texture(camera,
gl_sprite_texture(&g_sprite_earth, 0));
assert(glGetError() == GL_NO_ERROR);
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER,
GL_LINEAR_MIPMAP_LINEAR);
assert(glGetError() == GL_NO_ERROR);
gl_sphere_render(sphere);
glDisable(GL_DEPTH_TEST);
return true;
}
bool earth_unload (s_sequence *seq)
{
(void) seq;
return true;
}