From 26fb1fcafa552c600a70b2680c868f7355b138e4 Mon Sep 17 00:00:00 2001 From: omniscient <17525998+omnisci3nce@users.noreply.github.com> Date: Wed, 12 Jun 2024 00:16:51 +1000 Subject: wip: uvsphere mesh gen and pbr shader first pass --- src/maths/primitives.c | 64 +++++++++++++++++++++++++++++++++++++++++++++++--- src/maths/primitives.h | 2 +- 2 files changed, 62 insertions(+), 4 deletions(-) (limited to 'src/maths') diff --git a/src/maths/primitives.c b/src/maths/primitives.c index c0af140..3931b93 100644 --- a/src/maths/primitives.c +++ b/src/maths/primitives.c @@ -112,7 +112,7 @@ geometry_data geo_create_cuboid(f32x3 extents) { .vertices = vertices, .has_indices = true, .indices = indices, // FIXME: make darray methods that return stack allocated struct - .colour = vec3(0, 0, 0), + .colour = (rgba){ 0, 0, 0, 1 } }; return geo; @@ -133,6 +133,64 @@ geometry_data geo_create_cuboid(f32x3 extents) { // --- Spheres -geometry_data geo_create_uvsphere(f32 radius, f32 north_south_lines, f32 east_west_lines) { - // TODO +vec3 spherical_to_cartesian_coords(f32 rho, f32 theta, f32 phi) { + f32 x = rho * sin(phi) * cos(theta); + f32 y = rho * sin(phi) * sin(theta); + f32 z = rho * cos(phi); + return vec3(x, y, z); +} + +geometry_data geo_create_uvsphere(f32 radius, u32 north_south_lines, u32 east_west_lines) { + assert(east_west_lines >= 3); // sphere will be degenerate and look gacked without at least 3 + assert(north_south_lines >= 3); + + vertex_darray* vertices = vertex_darray_new(2 + (east_west_lines - 1) * north_south_lines); + + // Create a UV sphere with spherical coordinates + // a point P on the unit sphere can be represented P(r, theta, phi) + // for each vertex we must convert that to a cartesian R3 coordinate + + // Top point + vertex top = { .static_3d = { .position = vec3(0, 0, radius), + .normal = vec3(0, 0, radius), + .tex_coords = vec2(0, 0) } }; + + // parallels + for (u32 i = 0; i < (east_west_lines - 1); i++) { + // phi should range from 0 to pi + f32 phi = PI * ((i + 1) / east_west_lines); + + // meridians + for (u32 j = 0; j < east_west_lines; j++) { + // theta should range from 0 to 2PI + f32 theta = TAU * (j / north_south_lines); + vec3 position = spherical_to_cartesian_coords(radius, theta, phi); + f32 d = vec3_len(position); + assert(d == radius); // all points on the sphere should be 'radius' away from the origin + vertex v = { .static_3d = { + .position = position, + .normal = position, // normal vector on sphere is same as position + .tex_coords = vec2(0, 0) // TODO + } }; + vertex_darray_push(vertices, v); + } + } + + // Bottom point + vertex bot = { .static_3d = { .position = vec3(0, 0, -radius), + .normal = vec3(0, 0, -radius), + .tex_coords = vec2(0, 0) } }; + + // TODO: generate indices for for each flat quad on the UV sphere and triangles + // where they meet the north and south poles + u32_darray* indices = u32_darray_new(1); + + geometry_data geo = { + .format = VERTEX_STATIC_3D, + .vertices = vertices, + .has_indices = true, + .indices = indices, + .colour = RED_800, + }; + return geo; } diff --git a/src/maths/primitives.h b/src/maths/primitives.h index 3e0cc5f..be2c6ff 100644 --- a/src/maths/primitives.h +++ b/src/maths/primitives.h @@ -9,5 +9,5 @@ geometry_data geo_create_plane(f32x2 extents); geometry_data geo_create_cuboid(f32x3 extents); geometry_data geo_create_cylinder(f32 radius, f32 height, u32 resolution); -geometry_data geo_create_uvsphere(f32 radius, f32 north_south_lines, f32 east_west_lines); +geometry_data geo_create_uvsphere(f32 radius, u32 north_south_lines, u32 east_west_lines); geometry_data geo_create_icosphere(f32 radius, f32 n_subdivisions); \ No newline at end of file -- cgit v1.2.3-70-g09d2