diff options
author | talha <talha@talhaamir.xyz> | 2024-07-02 10:06:24 +0500 |
---|---|---|
committer | talha <talha@talhaamir.xyz> | 2024-07-02 10:06:24 +0500 |
commit | c04e95ca7d020a400be0582c8e6550af3d5d1cf4 (patch) | |
tree | def60b14460a669ef36a893b818fac7777dde78e /source/main.cpp | |
parent | 5ebf4c228b9d26c2b1b32073487413eea6a4c4f2 (diff) |
initial refactor of text rendering done
Diffstat (limited to 'source/main.cpp')
-rw-r--r-- | source/main.cpp | 367 |
1 files changed, 191 insertions, 176 deletions
diff --git a/source/main.cpp b/source/main.cpp index ecba4c9..4a20754 100644 --- a/source/main.cpp +++ b/source/main.cpp @@ -31,6 +31,10 @@ * but it will be useful for the future */ +/* +* - refactor text rendering functions and see what to do about them. +*/ + typedef uint8_t u8; typedef uint16_t u16; typedef uint32_t u32; @@ -280,6 +284,184 @@ void gl2d_draw_tex_quad(GlShader shader, GlTexturedQuad tex_quad, Vec3 position, glDrawArrays(GL_TRIANGLES, 0, 6); } +// ============ UI Text Rendering ============= +// + +struct TextChar { + Vec2 size; + Vec2 bearing; + s64 advance; +}; + +struct TextState { + u32 text_atlas; + u32 vao; + u32 vbo; + TextChar* char_map; + Mat4* transforms; + s32* char_indexes; +}; + +u32 chunk_size = 32; + +void gl_setup_text(TextState* state, FT_Face font_face) +{ + FT_Set_Pixel_Sizes(font_face, 256, 256); + glPixelStorei(GL_UNPACK_ALIGNMENT, 1); + + glGenTextures(1, &(state->text_atlas)); + glBindTexture(GL_TEXTURE_2D_ARRAY, state->text_atlas); + // generate texture + glTexImage3D( + GL_TEXTURE_2D_ARRAY, + 0, + GL_R8, + 256, + 256, + 128, + 0, + GL_RED, + GL_UNSIGNED_BYTE, + 0 + ); + + // generate characters + for (unsigned char c = 0; c < 128; c++) + { + if (FT_Load_Char(font_face, c, FT_LOAD_RENDER)) + { + printf("Error: Freetype failed to load glyph: %c", c); + } + else + { + glTexSubImage3D( + GL_TEXTURE_2D_ARRAY, + 0, + 0, 0, // x, y offset + int(c), + font_face->glyph->bitmap.width, + font_face->glyph->bitmap.rows, + 1, + GL_RED, + GL_UNSIGNED_BYTE, + font_face->glyph->bitmap.buffer + ); + // set texture options + glTexParameteri(GL_TEXTURE_2D_ARRAY, GL_TEXTURE_WRAP_S, GL_CLAMP_TO_EDGE); + glTexParameteri(GL_TEXTURE_2D_ARRAY, GL_TEXTURE_WRAP_T, GL_CLAMP_TO_EDGE); + glTexParameteri(GL_TEXTURE_2D_ARRAY, GL_TEXTURE_MIN_FILTER, GL_LINEAR); + glTexParameteri(GL_TEXTURE_2D_ARRAY, GL_TEXTURE_MAG_FILTER, GL_LINEAR); + + TextChar tc = { + Vec2{(r32)font_face->glyph->bitmap.width, (r32)font_face->glyph->bitmap.rows}, + Vec2{(r32)font_face->glyph->bitmap_left, (r32)font_face->glyph->bitmap_top}, + font_face->glyph->advance.x + }; + + state->char_map[c] = tc; + } + } + glBindTexture(GL_TEXTURE_2D_ARRAY, 0); + + // @note: this data is used for GL_TRIANGLE_STRIP + // as such the order for vertices for this is AntiCW -> CW -> AntiCW + // that can be seen in this array as it goes from ACW -> CW + r32 vertices[] = { + 0.0f, 1.0f, + 0.0f, 0.0f, + 1.0f, 1.0f, + 1.0f, 0.0f + }; + + glGenVertexArrays(1, &(state->vao)); + glGenBuffers(1, &(state->vbo)); + glBindVertexArray(state->vao); + glBindBuffer(GL_ARRAY_BUFFER, state->vbo); + glBufferData(GL_ARRAY_BUFFER, sizeof(vertices), vertices, GL_STATIC_DRAW); + glEnableVertexAttribArray(0); + glVertexAttribPointer(0, 2, GL_FLOAT, GL_FALSE, 0, 0); + + glBindBuffer(GL_ARRAY_BUFFER, 0); + glBindVertexArray(0); +} + +void gl_render_text(TextState state, u32 sp, const char* text, u32 text_len, Vec3 pos, r32 size, Vec3 color) +{ + // render ui text + glDisable(GL_DEPTH_TEST); + glEnable(GL_BLEND); + glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA); + + glUseProgram(sp); + s32 text_color_loc = glGetUniformLocation(sp, "TextColor"); + glUniform3fv(text_color_loc, 1, color.data); + + glBindVertexArray(state.vao); + glBindTexture(GL_TEXTURE_2D_ARRAY, state.text_atlas); + glBindBuffer(GL_ARRAY_BUFFER, state.vbo); + // @note: not really sure about this + glActiveTexture(GL_TEXTURE0); + + u32 running_index = 0; + r32 startx = pos.x; + r32 starty = pos.y; + r32 zindex = pos.z; + r32 linex = startx; + r32 scale = size / 256.0f; + for (u32 i = 0; i < text_len - 1; i++) + { + char c = text[i]; + TextChar rc = state.char_map[c]; + if (c == '\n') + { + linex = startx; + starty = starty - (rc.size.y * 1.5 * scale); // line_height: 1.5; + continue; + } + else if (c == ' ') + { + linex += (rc.advance >> 6) * scale; + continue; + } + r32 xpos = linex + (scale * rc.bearing.x); + r32 ypos = starty - (256.0f - rc.bearing.y) * scale; + + r32 w = scale * 256.0f; + r32 h = scale * 256.0f; + + Mat4 tm = translation_matrix4m(xpos, ypos, -zindex); + Mat4 sm = scaling_matrix4m(w, h, 0); + Mat4 model_mat = multiply4m(tm, sm); + state.transforms[running_index] = model_mat; + state.char_indexes[running_index] = int(c); + + linex += (rc.advance >> 6) * scale; + running_index++; + if (running_index > chunk_size-1) { + { + r32 letter_transforms_loc = glGetUniformLocation(sp, "LetterTransforms"); + glUniformMatrix4fv(letter_transforms_loc, chunk_size, GL_TRUE, &(state.transforms[0].buffer[0])); + r32 texture_map_loc = glGetUniformLocation(sp, "TextureMap"); + glUniform1iv(texture_map_loc, chunk_size, state.char_indexes); + glDrawArraysInstanced(GL_TRIANGLE_STRIP, 0, 4, chunk_size); + running_index = 0; + memset(state.transforms, 0, chunk_size); + memset(state.char_indexes, 0, chunk_size); + } + } + } + if (running_index > 0) { + r32 letter_transforms_loc = glGetUniformLocation(sp, "LetterTransforms"); + glUniformMatrix4fv(letter_transforms_loc, chunk_size, GL_TRUE, &(state.transforms[0].buffer[0])); + r32 texture_map_loc = glGetUniformLocation(sp, "TextureMap"); + glUniform1iv(texture_map_loc, chunk_size, state.char_indexes); + glDrawArraysInstanced(GL_TRIANGLE_STRIP, 0, 4, chunk_size); + } + + glBindBuffer(GL_ARRAY_BUFFER, 0); + glBindVertexArray(0); + glBindTexture(GL_TEXTURE_2D_ARRAY, 0); +} // =============== CAMERA STUFF ================= // @note: Be sure to update and refactor the camera Mat4 camera_create4m(Vec3 camera_pos, Vec3 camera_look, Vec3 camera_up) @@ -308,12 +490,6 @@ Vec3 camera_look_around(r32 angle_pitch, r32 angle_yaw) return camera_look; } -struct TextChar { - Vec2 size; - Vec2 bearing; - s64 advance; -}; - struct CameraOrtho { r32 left; r32 right; @@ -446,88 +622,12 @@ int main(int argc, char* argv[]) // setup_text // debug only u32 chunk_size = 32; - Mat4* text_transforms = (Mat4*)malloc(chunk_size*sizeof(Mat4)); - s32* char_indexes = (s32*)malloc(chunk_size*sizeof(s32)); - TextChar* all_text_chars = (TextChar*)malloc(128*sizeof(TextChar)); - - FT_Set_Pixel_Sizes(face, 256, 256); - glPixelStorei(GL_UNPACK_ALIGNMENT, 1); + TextState ui_text = TextState{}; + ui_text.transforms = (Mat4*)malloc(chunk_size*sizeof(Mat4)); + ui_text.char_indexes = (s32*)malloc(chunk_size*sizeof(s32)); + ui_text.char_map = (TextChar*)malloc(128*sizeof(TextChar)); - u32 texture; - glGenTextures(1, &texture); - glBindTexture(GL_TEXTURE_2D_ARRAY, texture); - // generate texture - glTexImage3D( - GL_TEXTURE_2D_ARRAY, - 0, - GL_R8, - 256, - 256, - 128, - 0, - GL_RED, - GL_UNSIGNED_BYTE, - 0 - ); - - // generate characters - for (unsigned char c = 0; c < 128; c++) - { - if (FT_Load_Char(face, c, FT_LOAD_RENDER)) - { - printf("Error: Freetype failed to load glyph: %c", c); - } - else - { - glTexSubImage3D( - GL_TEXTURE_2D_ARRAY, - 0, - 0, 0, // x, y offset - int(c), - face->glyph->bitmap.width, - face->glyph->bitmap.rows, - 1, - GL_RED, - GL_UNSIGNED_BYTE, - face->glyph->bitmap.buffer - ); - // set texture options - glTexParameteri(GL_TEXTURE_2D_ARRAY, GL_TEXTURE_WRAP_S, GL_CLAMP_TO_EDGE); - glTexParameteri(GL_TEXTURE_2D_ARRAY, GL_TEXTURE_WRAP_T, GL_CLAMP_TO_EDGE); - glTexParameteri(GL_TEXTURE_2D_ARRAY, GL_TEXTURE_MIN_FILTER, GL_LINEAR); - glTexParameteri(GL_TEXTURE_2D_ARRAY, GL_TEXTURE_MAG_FILTER, GL_LINEAR); - - TextChar tc = { - Vec2{(r32)face->glyph->bitmap.width, (r32)face->glyph->bitmap.rows}, - Vec2{(r32)face->glyph->bitmap_left, (r32)face->glyph->bitmap_top}, - face->glyph->advance.x - }; - - all_text_chars[c] = tc; - } - } - glBindTexture(GL_TEXTURE_2D_ARRAY, 0); - // @note: this data is used for GL_TRIANGLE_STRIP - // as such the order for vertices for this is AntiCW -> CW -> AntiCW - // that can be seen in this array as it goes from ACW -> CW - r32 vertices[] = { - 0.0f, 1.0f, - 0.0f, 0.0f, - 1.0f, 1.0f, - 1.0f, 0.0f - }; - - u32 vao, vbo; - glGenVertexArrays(1, &vao); - glGenBuffers(1, &vbo); - glBindVertexArray(vao); - glBindBuffer(GL_ARRAY_BUFFER, vbo); - glBufferData(GL_ARRAY_BUFFER, sizeof(vertices), vertices, GL_STATIC_DRAW); - glEnableVertexAttribArray(0); - glVertexAttribPointer(0, 2, GL_FLOAT, GL_FALSE, 0, 0); - - glBindBuffer(GL_ARRAY_BUFFER, 0); - glBindVertexArray(0); + gl_setup_text(&ui_text, face); // ==================== END setup_text r32 FOV = 90.0; @@ -567,15 +667,12 @@ int main(int argc, char* argv[]) Vec3{ -1.0, 2.0, -8.0}, // 7: refractive "window" }; + // the renderer will handle these operations s32 proj_loc; glUseProgram(ui_text_sp); proj_loc = glGetUniformLocation(ui_text_sp, "Projection"); - // glUniformMatrix4fv(proj_loc, 1, GL_TRUE, ortho_cam.projection.buffer); glUniformMatrix4fv(proj_loc, 1, GL_TRUE, ortho_proj.buffer); - //glUseProgram(tex_quad_shader.id); - //glUniformMatrix4fv(tex_quad_shader.proj_loc, 1, GL_TRUE, ortho_cam.projection.buffer); - glEnable(GL_DEPTH_TEST); //glEnable(GL_CULL_FACE); //glCullFace(GL_BACK); @@ -714,99 +811,17 @@ int main(int argc, char* argv[]) // OUTPUT glClearColor(1.0f, 0.6f, .6f, 1.0f); glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT); - { - // render ui text - glDisable(GL_DEPTH_TEST); - glEnable(GL_BLEND); - glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA); - - glUseProgram(ui_text_sp); - s32 text_color_loc = glGetUniformLocation(ui_text_sp, "TextColor"); - glUniform3f(text_color_loc, 0.0f, 0.0f, 0.0f); - - glBindVertexArray(vao); - glBindTexture(GL_TEXTURE_2D_ARRAY, texture); - glBindBuffer(GL_ARRAY_BUFFER, vbo); - glActiveTexture(GL_TEXTURE0); - - const char text_to_render[] = "Lorem ipsum dolor sit amet, consectetur adipiscing elit. Nam pellentesque fringilla mauris vitae convallis. \nSuspendisse potenti. Morbi at eros nulla. Morbi pellentesque mollis diam, nec eleifend sapien commodo nec. \nAenean elementum, eros non mattis consectetur, odio nulla posuere orci, sed posuere nibh nisl suscipit orci. Donec lectus dolor, rhoncus at vulputate sit amet, pellentesque non velit. \nVivamus vitae lectus lacinia, feugiat elit non, malesuada diam. Class aptent taciti sociosqu ad litora torquent per conubia nostra, per inceptos himenaeos. \nNulla porttitor cursus nulla, sit amet pulvinar nibh tempus sed. Nulla quis luctus tellus. Suspendisse efficitur tellus vel augue tempor, sed gravida purus consectetur. \nSed congue rutrum elit nec aliquet.\n\n" + const char text_to_render[] = "Lorem ipsum dolor sit amet, consectetur adipiscing elit. Nam pellentesque fringilla mauris vitae convallis. \nSuspendisse potenti. Morbi at eros nulla. Morbi pellentesque mollis diam, nec eleifend sapien commodo nec. \nAenean elementum, eros non mattis consectetur, odio nulla posuere orci, sed posuere nibh nisl suscipit orci. Donec lectus dolor, rhoncus at vulputate sit amet, pellentesque non velit. \nVivamus vitae lectus lacinia, feugiat elit non, malesuada diam. Class aptent taciti sociosqu ad litora torquent per conubia nostra, per inceptos himenaeos. \nNulla porttitor cursus nulla, sit amet pulvinar nibh tempus sed. Nulla quis luctus tellus. Suspendisse efficitur tellus vel augue tempor, sed gravida purus consectetur. \nSed congue rutrum elit nec aliquet.\n\n" "In hac habitasse platea dictumst. Nulla nec pellentesque purus, vitae interdum eros. Pellentesque pretium dui sed erat commodo finibus. \nInteger fringilla tincidunt orci, vel mattis dolor eleifend id. Sed condimentum lorem sit amet enim porttitor dictum nec ut arcu. \nMauris mauris ex, varius in sem eget, semper hendrerit enim. Curabitur vel vehicula metus, vel luctus nunc. \nPraesent et molestie diam, non faucibus ligula. Duis elementum nunc erat, eget dictum mi faucibus id.\n\n" "Nulla non vehicula erat, vitae tincidunt turpis. In tincidunt dolor vel augue tincidunt, vitae feugiat leo facilisis. Nullam auctor sodales sodales. \nDuis augue nibh, scelerisque id faucibus sit amet, ultricies eget lorem. Integer dapibus ultricies dolor sit amet mollis. \nQuisque eros sem, bibendum non convallis quis, vulputate ac erat. Nam laoreet justo id dui feugiat, condimentum elementum dolor pellentesque. \nVivamus imperdiet metus et mattis laoreet. Ut eget scelerisque neque. Praesent mollis, nisl quis volutpat eleifend, purus sem suscipit ex, \na molestie elit risus vel velit. Pellentesque habitant morbi tristique senectus et netus et malesuada fames ac turpis egestas. \nProin vitae nibh ac leo tempor vehicula quis quis nulla. Donec sodales sapien et nulla varius vulputate. \nPhasellus ac eleifend dui.\n\n" "Nulla nec nibh ut mauris accumsan tristique. Nam iaculis pellentesque elit, sit amet tristique erat dignissim ac. \nDuis gravida dolor eget sem fringilla, ac consequat ipsum malesuada. In facilisis mattis nibh in bibendum. \nUt congue gravida bibendum. Phasellus egestas consectetur sem, ornare pellentesque mauris. Ut pretium eleifend dui ut feugiat. \nCurabitur ut faucibus nisi, ac laoreet tortor.\n\n" "Nulla interdum vehicula tempus. Curabitur bibendum nisl eget sem tristique, at aliquam magna rutrum. \nLorem ipsum dolor sit amet, consectetur adipiscing elit. Suspendisse tempus posuere urna et auctor. \nVestibulum lectus risus, aliquet eget placerat nec, iaculis et augue. \nMauris vel nulla eu urna pretium vestibulum in vel elit. Donec sollicitudin dui felis, sed auctor nibh aliquam \nsit amet. Pellentesque eget elementum lacus, suscipit facilisis nisi. Integer tempus velit nulla. \nSed ullamcorper, eros et consequat pharetra, eros ipsum varius odio, vitae commodo lectus libero at libero. \nCurabitur sed elit tincidunt tellus tempor convallis et eu nunc. \nAliquam blandit auctor ipsum, eu mattis leo viverra vel. Integer feugiat dui neque, vitae vulputate lectus congue non. \nFusce molestie vel ligula nec sodales. Integer vestibulum molestie porttitor.\n\n"; - u32 running_index = 0; - r32 startx = 10.0f; - r32 starty = 740.0f; - r32 linex = startx; - r32 zindex = 3.0f; - r32 scale = 1.0f * 22.0f / 256.0f; - r32 max_size = 0.0f; - for (u32 i = 0; i < sizeof(text_to_render) - 1; i++) - { - char c = text_to_render[i]; - TextChar rc = all_text_chars[c]; - if (rc.size.y > max_size) - { - max_size = rc.size.y; - } - if (c == '\n') - { - linex = startx; - starty = starty - (rc.size.y * 1.5 * scale); // line_height: 1.5; - continue; - } - else if (c == ' ') - { - linex += (rc.advance >> 6) * scale; - continue; - } - r32 xpos = linex + (scale * rc.bearing.x); - r32 ypos = starty - (256.0f - rc.bearing.y) * scale; - - r32 w = scale * 256.0f; - r32 h = scale * 256.0f; - - Mat4 tm = translation_matrix4m(xpos, ypos, -zindex); - Mat4 sm = scaling_matrix4m(w, h, 0); - Mat4 model_mat = multiply4m(tm, sm); - text_transforms[running_index] = model_mat; - char_indexes[running_index] = int(c); - - linex += (rc.advance >> 6) * scale; - running_index++; - if (running_index > chunk_size-1) { - { - r32 letter_transforms_loc = glGetUniformLocation(ui_text_sp, "LetterTransforms"); - glUniformMatrix4fv(letter_transforms_loc, chunk_size, GL_TRUE, &text_transforms[0].buffer[0]); - r32 texture_map_loc = glGetUniformLocation(ui_text_sp, "TextureMap"); - glUniform1iv(texture_map_loc, chunk_size, char_indexes); - glDrawArraysInstanced(GL_TRIANGLE_STRIP, 0, 4, chunk_size); - running_index = 0; - memset(text_transforms, 0, chunk_size); - memset(char_indexes, 0, chunk_size); - } - } - } - if (running_index > 0) { - r32 letter_transforms_loc = glGetUniformLocation(ui_text_sp, "LetterTransforms"); - glUniformMatrix4fv(letter_transforms_loc, chunk_size, GL_TRUE, &text_transforms[0].buffer[0]); - r32 texture_map_loc = glGetUniformLocation(ui_text_sp, "TextureMap"); - glUniform1iv(texture_map_loc, chunk_size, char_indexes); - glDrawArraysInstanced(GL_TRIANGLE_STRIP, 0, 4, chunk_size); - } - - glBindBuffer(GL_ARRAY_BUFFER, 0); - glBindVertexArray(0); - glBindTexture(GL_TEXTURE_2D_ARRAY, 0); - } - + gl_render_text(ui_text, ui_text_sp, text_to_render, sizeof(text_to_render), + Vec3{10.0f, 700.0f, 3.0f}, 28.0f, Vec3{0.0f, 0.0f, 0.0f}); SDL_GL_SwapWindow(window); } - free(text_transforms); - free(char_indexes); - free(all_text_chars); // opengl free calls // it is up to the renderer to free everything here // of course I have not bothered to set that up so it does not matter |