899 lines
28 KiB
C++
899 lines
28 KiB
C++
/*
|
|
* Mr. 4th Dimention - Allen Webster
|
|
*
|
|
* 12.17.2014
|
|
*
|
|
* Rendering layer for project codename "4ed"
|
|
*
|
|
*/
|
|
|
|
// TOP
|
|
|
|
internal i32_Rect
|
|
rect_clamp_to_rect(i32_Rect rect, i32_Rect clamp_box){
|
|
if (rect.x0 < clamp_box.x0) rect.x0 = clamp_box.x0;
|
|
if (rect.y0 < clamp_box.y0) rect.y0 = clamp_box.y0;
|
|
if (rect.x1 > clamp_box.x1) rect.x1 = clamp_box.x1;
|
|
if (rect.y1 > clamp_box.y1) rect.y1 = clamp_box.y1;
|
|
|
|
return rect;
|
|
}
|
|
|
|
inline i32_Rect
|
|
rect_clamp_to_rect(i32 left, i32 top, i32 right, i32 bottom, i32_Rect clamp_box){
|
|
return rect_clamp_to_rect(i32R(left, top, right, bottom), clamp_box);
|
|
}
|
|
|
|
inline i32_Rect
|
|
rect_from_target(Render_Target *target){
|
|
return i32R(0, 0, target->width, target->height);
|
|
}
|
|
|
|
#if SOFTWARE_RENDER
|
|
internal void
|
|
draw_clear(Render_Target *target, u32 color){
|
|
u8 *pixel_line = (u8*)target->pixel_data;
|
|
for (i32 pixel_y = 0; pixel_y < target->height; ++pixel_y){
|
|
u32 *pixel = (u32*)pixel_line;
|
|
for (i32 pixel_x = 0; pixel_x < target->width; ++pixel_x){
|
|
*pixel++ = color;
|
|
}
|
|
pixel_line += target->pitch;
|
|
}
|
|
}
|
|
|
|
internal void
|
|
draw_vertical_line(Render_Target *target,
|
|
i32 x, i32 top, i32 bottom,
|
|
u32 color){
|
|
if (x >= 0 && x < target->width){
|
|
if (top < 0){
|
|
top = 0;
|
|
}
|
|
if (bottom >= target->height){
|
|
bottom = target->height - 1;
|
|
}
|
|
|
|
if (top <= bottom){
|
|
i32 y_range = bottom - top;
|
|
u8 *pixel_line = (u8*)target->pixel_data + top*target->pitch;
|
|
for (i32 pixel_y = 0; pixel_y <= y_range; ++pixel_y){
|
|
u32 *pixel = (u32*)pixel_line + x;
|
|
*pixel = color;
|
|
pixel_line += target->pitch;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
internal void
|
|
draw_horizontal_line(Render_Target *target,
|
|
i32 y, i32 left, i32 right,
|
|
u32 color){
|
|
if (y >= 0 && y < target->height){
|
|
if (left < 0){
|
|
left = 0;
|
|
}
|
|
if (right >= target->width){
|
|
right = target->width - 1;
|
|
}
|
|
|
|
if (left <= right){
|
|
i32 x_range = right - left;
|
|
u8 *pixel_line = (u8*)target->pixel_data + y*target->pitch;
|
|
u32 *pixel = (u32*)pixel_line + left;
|
|
for (i32 pixel_x = 0; pixel_x <= x_range; ++pixel_x){
|
|
*pixel = color;
|
|
++pixel;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
internal void
|
|
draw_rectangle_blend_2corner_unclamped(Render_Target *target,
|
|
Blit_Rect rect, u32 color){
|
|
if (rect.x_start < rect.x_end && rect.y_start < rect.y_end){
|
|
i32 y_range = rect.y_end - rect.y_start;
|
|
i32 x_range = rect.x_end - rect.x_start;
|
|
|
|
u8 a,r,g,b;
|
|
a = (u8)(color >> 24);
|
|
r = (u8)(color >> 16);
|
|
g = (u8)(color >> 8);
|
|
b = (u8)(color >> 0);
|
|
|
|
real32 blend = (a/255.f);
|
|
real32 pr, pg, pb;
|
|
pr = r*blend;
|
|
pg = g*blend;
|
|
pb = b*blend;
|
|
|
|
real32 inv_blend = 1.f - blend;
|
|
|
|
u8 *pixel_line = (u8*)target->pixel_data + rect.y_start*target->pitch;
|
|
for (i32 pixel_y = 0; pixel_y < y_range; ++pixel_y){
|
|
u32 *pixel = (u32*)pixel_line + rect.x_start;
|
|
for (i32 pixel_x = 0; pixel_x < x_range; ++pixel_x){
|
|
u8 dr,dg,db;
|
|
dr = (u8)(*pixel >> 16);
|
|
dg = (u8)(*pixel >> 8);
|
|
db = (u8)(*pixel >> 0);
|
|
|
|
dr = (u8)(dr*inv_blend + pr);
|
|
dg = (u8)(dg*inv_blend + pg);
|
|
db = (u8)(db*inv_blend + pb);
|
|
|
|
*pixel = (dr << 16) | (dg << 8) | (db);
|
|
++pixel;
|
|
}
|
|
pixel_line += target->pitch;
|
|
}
|
|
}
|
|
}
|
|
|
|
internal void
|
|
draw_rectangle_noblend_2corner_unclamped(Render_Target *target,
|
|
Blit_Rect rect, u32 color){
|
|
if (rect.x_start < rect.x_end && rect.y_start < rect.y_end){
|
|
i32 y_range = rect.y_end - rect.y_start;
|
|
i32 x_range = rect.x_end - rect.x_start;
|
|
|
|
u8 *pixel_line = (u8*)target->pixel_data + rect.y_start*target->pitch;
|
|
for (i32 pixel_y = 0; pixel_y < y_range; ++pixel_y){
|
|
u32 *pixel = (u32*)pixel_line + rect.x_start;
|
|
for (i32 pixel_x = 0; pixel_x < x_range; ++pixel_x){
|
|
*pixel++ = color;
|
|
}
|
|
pixel_line += target->pitch;
|
|
}
|
|
}
|
|
}
|
|
|
|
// NOTE(allen): uses of this can be replaced with draw_rectangle_2corner_unclamped
|
|
// if it is guaranteed that clip_box will be within the target area.
|
|
inline void
|
|
draw_rectangle_2corner_clipped(Render_Target *target,
|
|
i32 left, i32 top, i32 right, i32 bottom,
|
|
u32 color, Blit_Rect clip_box){
|
|
clip_box = rect_clamp_to_rect(clip_box, rect_from_target(target));
|
|
Blit_Rect rect = rect_clamp_to_rect(left, top, right, bottom, clip_box);
|
|
draw_rectangle_noblend_2corner_unclamped(target, rect, color);
|
|
}
|
|
|
|
inline void
|
|
draw_rectangle_2corner(Render_Target *target,
|
|
i32 left, i32 top, i32 right, i32 bottom,
|
|
u32 color){
|
|
Blit_Rect rect = rect_clamp_to_rect(left, top, right, bottom, rect_from_target(target));
|
|
draw_rectangle_noblend_2corner_unclamped(target, rect, color);
|
|
}
|
|
|
|
inline void
|
|
draw_rectangle_clipped(Render_Target *target,
|
|
i32 x, i32 y, i32 w, i32 h,
|
|
u32 color, Blit_Rect clip_box){
|
|
draw_rectangle_2corner_clipped(target, x, y, x+w, y+h, color, clip_box);
|
|
}
|
|
|
|
inline void
|
|
draw_rectangle(Render_Target *target,
|
|
i32 x, i32 y, i32 w, i32 h,
|
|
u32 color){
|
|
draw_rectangle_2corner(target, x, y, x+w, y+h, color);
|
|
}
|
|
|
|
internal void
|
|
draw_rectangle_outline_unclamped(Render_Target *target,
|
|
i32 x, i32 y, i32 w, i32 h,
|
|
u32 color, Blit_Rect rect){
|
|
if (rect.x_start <= rect.x_end && rect.y_start <= rect.y_end){
|
|
if (rect.y_start == y){
|
|
draw_horizontal_line(target,
|
|
rect.y_start, rect.x_start, rect.x_end-1,
|
|
color);
|
|
}
|
|
|
|
if (rect.y_end == y+h){
|
|
draw_horizontal_line(target,
|
|
rect.y_end-1, rect.x_start, rect.x_end-1,
|
|
color);
|
|
}
|
|
|
|
if (rect.x_start == x){
|
|
draw_vertical_line(target,
|
|
rect.x_start, rect.y_start, rect.y_end-1,
|
|
color);
|
|
}
|
|
|
|
if (rect.x_end == x+w){
|
|
draw_vertical_line(target,
|
|
rect.x_end-1, rect.y_start, rect.y_end-1,
|
|
color);
|
|
}
|
|
}
|
|
}
|
|
|
|
internal void
|
|
draw_rectangle_outline_clipped(Render_Target *target,
|
|
i32 x, i32 y, i32 w, i32 h,
|
|
u32 color, Blit_Rect clip_box){
|
|
clip_box = rect_clamp_to_rect(clip_box, rect_from_target(target));
|
|
Blit_Rect rect = rect_clamp_to_rect(x, y, x+w, y+h, clip_box);
|
|
draw_rectangle_outline_unclamped(target, x, y, w, h, color, rect);
|
|
}
|
|
|
|
internal void
|
|
draw_rectangle_outline(Render_Target *target,
|
|
i32 x, i32 y, i32 w, i32 h,
|
|
u32 color){
|
|
Blit_Rect rect = rect_clamp_to_rect(x, y, x+w, y+h, rect_from_target(target));
|
|
draw_rectangle_outline_unclamped(target, x, y, w, h, color, rect);
|
|
}
|
|
|
|
// TODO(allen): eliminate this?
|
|
internal i32
|
|
font_init(){
|
|
return 1;
|
|
}
|
|
|
|
inline internal i32
|
|
font_predict_size(i32 pt_size){
|
|
return pt_size*pt_size*128;
|
|
}
|
|
|
|
internal i32
|
|
font_load(Font *font_out, i32 pt_size,
|
|
void *font_block, i32 font_block_size,
|
|
i32 *memory_used_out){
|
|
i32 result = 1;
|
|
File_Data file;
|
|
file = system_load_file((u8*)"liberation-mono.ttf");
|
|
|
|
if (!file.data){
|
|
result = 0;
|
|
}
|
|
|
|
else{
|
|
stbtt_fontinfo font;
|
|
if (!stbtt_InitFont(&font, (u8*)file.data, 0)){
|
|
result = 0;
|
|
}
|
|
else{
|
|
i32 ascent, descent, line_gap;
|
|
real32 scale;
|
|
|
|
stbtt_GetFontVMetrics(&font, &ascent, &descent, &line_gap);
|
|
scale = stbtt_ScaleForPixelHeight(&font, (real32)pt_size);
|
|
|
|
real32 scaled_ascent, scaled_descent, scaled_line_gap;
|
|
|
|
scaled_ascent = scale*ascent;
|
|
scaled_descent = scale*descent;
|
|
scaled_line_gap = scale*line_gap;
|
|
|
|
font_out->height = (i32)(scaled_ascent - scaled_descent + scaled_line_gap);
|
|
font_out->ascent = (i32)(scaled_ascent);
|
|
font_out->descent = (i32)(scaled_descent);
|
|
font_out->line_skip = (i32)(scaled_line_gap);
|
|
|
|
i32 max_advance = 0;
|
|
|
|
bool32 block_full = 0, block_overfull = 0;
|
|
u8 *memory_cursor = (u8*)font_block;
|
|
for (u16 code_point = 0; code_point < 128; ++code_point){
|
|
i32 glyph_index;
|
|
|
|
if (block_full){
|
|
block_overfull = 1;
|
|
font_out->glyphs[code_point] = {};
|
|
continue;
|
|
}
|
|
|
|
else{
|
|
glyph_index = stbtt_FindGlyphIndex(&font, code_point);
|
|
if (glyph_index != 0){
|
|
font_out->glyphs[code_point].exists = 1;
|
|
|
|
i32 left, right, top, bottom;
|
|
stbtt_GetGlyphBitmapBox(&font, glyph_index, scale, scale, &left, &top, &right, &bottom);
|
|
|
|
i32 glyph_width, glyph_height;
|
|
i32 data_width, data_height;
|
|
glyph_width = right - left;
|
|
glyph_height = bottom - top;
|
|
data_width = glyph_width + 2;
|
|
data_height = glyph_height + 2;
|
|
|
|
font_out->glyphs[code_point].minx = left;
|
|
font_out->glyphs[code_point].maxx = right;
|
|
font_out->glyphs[code_point].miny = top;
|
|
font_out->glyphs[code_point].maxy = bottom;
|
|
font_out->glyphs[code_point].width = data_width;
|
|
font_out->glyphs[code_point].height = data_height;
|
|
|
|
i32 advance, left_bearing;
|
|
stbtt_GetGlyphHMetrics(&font, glyph_index, &advance, &left_bearing);
|
|
|
|
font_out->glyphs[code_point].left_shift = (i32)(scale*left_bearing);
|
|
|
|
if (advance > max_advance){
|
|
max_advance = advance;
|
|
}
|
|
|
|
u8 *data = memory_cursor;
|
|
memory_cursor = memory_cursor + data_width*data_height;
|
|
|
|
i32 size_after_glyph = (i32)((u8*)memory_cursor - (u8*)font_block);
|
|
if (size_after_glyph >= font_block_size){
|
|
block_full = 1;
|
|
if (size_after_glyph > font_block_size){
|
|
block_overfull = 1;
|
|
}
|
|
}
|
|
else{
|
|
font_out->glyphs[code_point].data = data;
|
|
u8 *data_start = data + data_width + 1;
|
|
stbtt_MakeGlyphBitmap(&font, data_start,
|
|
glyph_width, glyph_height, data_width,
|
|
scale, scale,
|
|
glyph_index);
|
|
}
|
|
}
|
|
else{
|
|
font_out->glyphs[code_point] = {};
|
|
}
|
|
}
|
|
}
|
|
|
|
font_out->advance = Ceil(max_advance*scale);
|
|
}
|
|
system_free_file(file);
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
internal void
|
|
font_draw_glyph_subpixel(Render_Target *target,
|
|
Font *font, u16 character,
|
|
real32 x, real32 y, u32 color,
|
|
Blit_Rect clip){
|
|
if (clip.x_start <= clip.x_end && clip.y_start <= clip.y_end){
|
|
Glyph_Data glyph = font->glyphs[character];
|
|
|
|
Vec2 s_origin = V2(x - 1.f, y - 1.f);
|
|
|
|
i32 left_most = (i32)(x);
|
|
i32 top_most = (i32)(y);
|
|
|
|
real32 xe = x + glyph.width - 2;
|
|
real32 ye = y + glyph.height - 2;
|
|
i32 right_most = (i32)(xe)+(xe>0)*((i32)xe != xe);
|
|
i32 bottom_most = (i32)(ye)+(ye>0)*((i32)ye != ye);
|
|
|
|
if (left_most < clip.x_start){
|
|
left_most = clip.x_start;
|
|
}
|
|
|
|
if (top_most < clip.y_start){
|
|
top_most = clip.y_start;
|
|
}
|
|
|
|
if (right_most >= clip.x_end){
|
|
right_most = clip.x_end - 1;
|
|
}
|
|
|
|
if (bottom_most >= clip.y_end){
|
|
bottom_most = clip.y_end - 1;
|
|
}
|
|
|
|
u8 *dest_data = (u8*)target->pixel_data;
|
|
u8 *src_data = (u8*)glyph.data;
|
|
|
|
i32 width = glyph.width;
|
|
i32 target_pitch = target->pitch;
|
|
|
|
real32 inv_255 = 1.f / 255.f;
|
|
|
|
real32 cr,cg,cb,ca;
|
|
cb = (real32)((color) & 0xFF);
|
|
cg = (real32)((color >> 8) & 0xFF);
|
|
cr = (real32)((color >> 16) & 0xFF);
|
|
ca = (real32)((color >> 24) & 0xFF);
|
|
|
|
i32 lox_start = (i32)(left_most - s_origin.x);
|
|
i32 loy_start = (i32)(top_most - s_origin.y);
|
|
|
|
real32 lX = left_most - s_origin.x - lox_start;
|
|
real32 lY = top_most - s_origin.y - loy_start;
|
|
|
|
real32 inv_lX = 1.f - lX;
|
|
real32 inv_lY = 1.f - lY;
|
|
|
|
i32 loy = loy_start * width;
|
|
|
|
u8 *dest_line = (dest_data + top_most*target_pitch);
|
|
for (i32 y = top_most; y <= bottom_most; ++y){
|
|
u8 src_a[4];
|
|
|
|
i32 lox_loy = lox_start + loy;
|
|
src_a[0] = *(src_data + (lox_loy));
|
|
src_a[2] = *(src_data + (lox_loy + width));
|
|
|
|
u32 *dest_pixel = (u32*)(dest_line) + left_most;
|
|
|
|
for (i32 x = left_most; x <= right_most; ++x){
|
|
src_a[1] = *(src_data + (lox_loy + 1));
|
|
src_a[3] = *(src_data + (lox_loy + 1 + width));
|
|
|
|
real32 alpha = (src_a[0]*(inv_lX) + src_a[1]*(lX))*(inv_lY) +
|
|
(src_a[2]*(inv_lX) + src_a[3]*(lX))*(lY);
|
|
alpha *= inv_255;
|
|
|
|
u32 dp = *dest_pixel;
|
|
|
|
real32 dr,dg,db,da;
|
|
db = (real32)((dp) & 0xFF);
|
|
dg = (real32)((dp >> 8) & 0xFF);
|
|
dr = (real32)((dp >> 16) & 0xFF);
|
|
da = (real32)((dp >> 24) & 0xFF);
|
|
|
|
db = db + (cb - db)*alpha;
|
|
dg = dg + (cg - dg)*alpha;
|
|
dr = dr + (cr - dr)*alpha;
|
|
da = da + (ca - da)*alpha;
|
|
|
|
*dest_pixel = ((u8)db) | (((u8)dg) << 8) | (((u8)dr) << 16) | (((u8)da) << 24);
|
|
|
|
++dest_pixel;
|
|
++lox_loy;
|
|
|
|
src_a[0] = src_a[1];
|
|
src_a[2] = src_a[3];
|
|
}
|
|
loy += width;
|
|
dest_line += target_pitch;
|
|
}
|
|
}
|
|
}
|
|
|
|
internal void
|
|
font_draw_glyph(Render_Target *target, Font *font, u16 character,
|
|
real32 x, real32 y, u32 color){
|
|
Glyph_Data glyph = font->glyphs[character];
|
|
|
|
i32 left = glyph.minx;
|
|
i32 right = glyph.maxx;
|
|
i32 width = right - left;
|
|
|
|
real32 x_shift, y_shift;
|
|
x_shift = glyph.left_shift + (real32)width / font->advance;
|
|
y_shift = (real32)font->ascent + glyph.miny;
|
|
|
|
x += x_shift;
|
|
y += y_shift;
|
|
|
|
i32 xi, yi;
|
|
xi = (i32)(x);
|
|
yi = (i32)(y);
|
|
|
|
Blit_Rect rect = rect_clamp_to_rect(xi, yi, xi+glyph.width-1, yi+glyph.height-1, rect_from_target(target));
|
|
font_draw_glyph_subpixel(target, font, character, x, y, color, rect);
|
|
}
|
|
|
|
internal void
|
|
font_draw_glyph_clipped(Render_Target *target,
|
|
Font *font, u16 character,
|
|
real32 x, real32 y, u32 color,
|
|
Blit_Rect clip_box){
|
|
Glyph_Data glyph = font->glyphs[character];
|
|
|
|
i32 left = glyph.minx;
|
|
i32 right = glyph.maxx;
|
|
i32 width = right - left;
|
|
|
|
real32 x_shift, y_shift;
|
|
x_shift = glyph.left_shift + (real32)width / font->advance;
|
|
y_shift = (real32)font->ascent + glyph.miny;
|
|
|
|
x += x_shift;
|
|
y += y_shift;
|
|
|
|
i32 xi, yi;
|
|
xi = (i32)(x);
|
|
yi = (i32)(y);
|
|
|
|
clip_box = rect_clamp_to_rect(clip_box, rect_from_target(target));
|
|
Blit_Rect rect = rect_clamp_to_rect(xi, yi, xi+glyph.width-1, yi+glyph.height-1, clip_box);
|
|
font_draw_glyph_subpixel(target, font, character, x, y, color, rect);
|
|
}
|
|
|
|
#else
|
|
|
|
inline void
|
|
draw_set_clip(Render_Target *target, i32_Rect clip_box){
|
|
glScissor(clip_box.x0,
|
|
target->height - clip_box.y1,
|
|
clip_box.x1 - clip_box.x0,
|
|
clip_box.y1 - clip_box.y0);
|
|
}
|
|
|
|
inline void
|
|
draw_push_clip(Render_Target *target, i32_Rect clip_box){
|
|
Assert(target->clip_top == -1 ||
|
|
fits_inside(clip_box, target->clip_boxes[target->clip_top]));
|
|
Assert(target->clip_top+1 < ArrayCount(target->clip_boxes));
|
|
target->clip_boxes[++target->clip_top] = clip_box;
|
|
draw_set_clip(target, clip_box);
|
|
}
|
|
|
|
inline void
|
|
draw_pop_clip(Render_Target *target){
|
|
Assert(target->clip_top > 0);
|
|
--target->clip_top;
|
|
draw_set_clip(target, target->clip_boxes[target->clip_top]);
|
|
}
|
|
|
|
inline void
|
|
draw_bind_texture(Render_Target *target, i32 texid){
|
|
if (target->bound_texture != texid){
|
|
glBindTexture(GL_TEXTURE_2D, texid);
|
|
target->bound_texture = texid;
|
|
}
|
|
}
|
|
|
|
internal void
|
|
draw_set_color(Render_Target *target, u32 color){
|
|
if (target->color != color){
|
|
target->color = color;
|
|
Vec4 c = unpack_color4(color);
|
|
glColor4f(c.r, c.g, c.b, c.a);
|
|
}
|
|
}
|
|
|
|
internal void
|
|
draw_rectangle(Render_Target *target, i32_Rect rect, u32 color){
|
|
draw_set_color(target, color);
|
|
draw_bind_texture(target, 0);
|
|
glBegin(GL_QUADS);
|
|
{
|
|
glVertex2i(rect.x0, rect.y0);
|
|
glVertex2i(rect.x0, rect.y1);
|
|
glVertex2i(rect.x1, rect.y1);
|
|
glVertex2i(rect.x1, rect.y0);
|
|
}
|
|
glEnd();
|
|
}
|
|
|
|
internal void
|
|
draw_rectangle(Render_Target *target, real32_Rect rect, u32 color){
|
|
draw_set_color(target, color);
|
|
draw_bind_texture(target, 0);
|
|
glBegin(GL_QUADS);
|
|
{
|
|
glVertex2f(rect.x0, rect.y0);
|
|
glVertex2f(rect.x0, rect.y1);
|
|
glVertex2f(rect.x1, rect.y1);
|
|
glVertex2f(rect.x1, rect.y0);
|
|
}
|
|
glEnd();
|
|
}
|
|
|
|
internal void
|
|
draw_triangle_3corner(Render_Target *target,
|
|
real32 x0, real32 y0,
|
|
real32 x1, real32 y1,
|
|
real32 x2, real32 y2,
|
|
u32 color){
|
|
draw_set_color(target, color);
|
|
draw_bind_texture(target, 0);
|
|
glBegin(GL_TRIANGLES);
|
|
{
|
|
glVertex2f(x0, y0);
|
|
glVertex2f(x1, y1);
|
|
glVertex2f(x2, y2);
|
|
}
|
|
glEnd();
|
|
}
|
|
|
|
internal void
|
|
draw_gradient_2corner_clipped(Render_Target *target, real32_Rect rect,
|
|
Vec4 color_left, Vec4 color_right){
|
|
Vec4 cl = color_left;
|
|
Vec4 cr = color_right;
|
|
|
|
draw_bind_texture(target, 0);
|
|
glBegin(GL_QUADS);
|
|
{
|
|
glColor4f(cl.r, cl.g, cl.b, cl.a);
|
|
glVertex2f(rect.x0, rect.y0);
|
|
glVertex2f(rect.x0, rect.y1);
|
|
|
|
glColor4f(cr.r, cr.g, cr.b, cr.a);
|
|
glVertex2f(rect.x1, rect.y1);
|
|
glVertex2f(rect.x1, rect.y0);
|
|
}
|
|
glEnd();
|
|
}
|
|
|
|
inline void
|
|
draw_gradient_2corner_clipped(Render_Target *target, real32 l, real32 t, real32 r, real32 b,
|
|
Vec4 color_left, Vec4 color_right){
|
|
draw_gradient_2corner_clipped(target, real32R(l,t,r,b), color_left, color_right);
|
|
}
|
|
|
|
internal void
|
|
draw_rectangle_outline(Render_Target *target, real32_Rect rect, u32 color){
|
|
real32_Rect r;
|
|
r.x0 = rect.x0 + .5f;
|
|
r.y0 = rect.y0 + .5f;
|
|
r.x1 = rect.x1 - .5f;
|
|
r.y1 = rect.y1 - .5f;
|
|
|
|
draw_set_color(target, color);
|
|
draw_bind_texture(target, 0);
|
|
glBegin(GL_LINE_STRIP);
|
|
{
|
|
glVertex2f(r.x0, r.y0);
|
|
glVertex2f(r.x1, r.y0);
|
|
glVertex2f(r.x1, r.y1);
|
|
glVertex2f(r.x0, r.y1);
|
|
glVertex2f(r.x0, r.y0);
|
|
}
|
|
glEnd();
|
|
}
|
|
|
|
inline void
|
|
draw_rectangle_outline(Render_Target *target, i32_Rect rect, u32 color){
|
|
draw_rectangle_outline(target, real32R(rect), color);
|
|
}
|
|
|
|
internal void
|
|
draw_margin(Render_Target *target, i32_Rect outer, i32_Rect inner, u32 color){
|
|
draw_rectangle(target, i32R(outer.x0, outer.y0, outer.x1, inner.y0), color);
|
|
draw_rectangle(target, i32R(outer.x0, inner.y1, outer.x1, outer.y1), color);
|
|
draw_rectangle(target, i32R(outer.x0, inner.y0, inner.x0, inner.y1), color);
|
|
draw_rectangle(target, i32R(inner.x1, inner.y0, outer.x1, inner.y1), color);
|
|
}
|
|
|
|
// TODO(allen): eliminate this?
|
|
internal i32
|
|
font_init(){
|
|
return 1;
|
|
}
|
|
|
|
inline internal i32
|
|
font_predict_size(i32 pt_size){
|
|
return pt_size*pt_size*128;
|
|
}
|
|
|
|
internal i32
|
|
font_load(Font *font_out, char *filename, i32 pt_size,
|
|
void *font_block, i32 font_block_size,
|
|
i32 *memory_used_out, i32 tab_width){
|
|
i32 result = 1;
|
|
File_Data file;
|
|
file = system_load_file((u8*)filename);
|
|
|
|
if (!file.data){
|
|
result = 0;
|
|
}
|
|
|
|
else{
|
|
stbtt_fontinfo font;
|
|
if (!stbtt_InitFont(&font, (u8*)file.data, 0)){
|
|
result = 0;
|
|
}
|
|
else{
|
|
i32 ascent, descent, line_gap;
|
|
real32 scale;
|
|
|
|
stbtt_GetFontVMetrics(&font, &ascent, &descent, &line_gap);
|
|
scale = stbtt_ScaleForPixelHeight(&font, (real32)pt_size);
|
|
|
|
real32 scaled_ascent, scaled_descent, scaled_line_gap;
|
|
|
|
scaled_ascent = scale*ascent;
|
|
scaled_descent = scale*descent;
|
|
scaled_line_gap = scale*line_gap;
|
|
|
|
font_out->height = (i32)(scaled_ascent - scaled_descent + scaled_line_gap);
|
|
font_out->ascent = (i32)(scaled_ascent);
|
|
font_out->descent = (i32)(scaled_descent);
|
|
font_out->line_skip = (i32)(scaled_line_gap);
|
|
|
|
u8 *memory_cursor = (u8*)font_block;
|
|
Assert(pt_size*pt_size*128 <= font_block_size);
|
|
|
|
i32 tex_width, tex_height;
|
|
tex_width = pt_size*128;
|
|
tex_height = pt_size*2;
|
|
|
|
font_out->tex_width = tex_width;
|
|
font_out->tex_height = tex_height;
|
|
|
|
if (stbtt_BakeFontBitmap((u8*)file.data, 0, (real32)pt_size,
|
|
memory_cursor, tex_width, tex_height, 0, 128, font_out->chardata) <= 0){
|
|
result = 0;
|
|
}
|
|
|
|
else{
|
|
GLuint font_tex;
|
|
glGenTextures(1, &font_tex);
|
|
glBindTexture(GL_TEXTURE_2D, font_tex);
|
|
|
|
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_REPEAT);
|
|
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_REPEAT);
|
|
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR);
|
|
|
|
glTexImage2D(GL_TEXTURE_2D, 0, GL_ALPHA, tex_width, tex_height, 0, GL_ALPHA, GL_UNSIGNED_BYTE, memory_cursor);
|
|
|
|
font_out->tex = font_tex;
|
|
glBindTexture(GL_TEXTURE_2D, 0);
|
|
}
|
|
|
|
font_out->chardata['\r'] = font_out->chardata[' '];
|
|
font_out->chardata['\n'] = font_out->chardata[' '];
|
|
font_out->chardata['\t'] = font_out->chardata[' '];
|
|
font_out->chardata['\t'].xadvance *= tab_width;
|
|
|
|
i32 max_advance = 0;
|
|
for (u16 code_point = 0; code_point < 128; ++code_point){
|
|
if (stbtt_FindGlyphIndex(&font, code_point) != 0){
|
|
font_out->glyphs[code_point].exists = 1;
|
|
i32 advance = CEIL32(font_out->chardata[code_point].xadvance);
|
|
if (max_advance < advance){
|
|
max_advance = advance;
|
|
}
|
|
}
|
|
}
|
|
font_out->advance = max_advance - 1;
|
|
}
|
|
system_free_file(file);
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
internal void
|
|
font_set_tabwidth(Font *font, i32 tab_width){
|
|
font->chardata['\t'].xadvance *= font->chardata[' '].xadvance * tab_width;
|
|
}
|
|
|
|
internal void
|
|
font_draw_glyph_mono(Render_Target *target, Font *font, u16 character,
|
|
real32 x, real32 y, real32 advance, u32 color){
|
|
real32 x_shift, y_shift;
|
|
i32 left = font->chardata[character].x0;
|
|
i32 right = font->chardata[character].x1;
|
|
i32 width = (right - left);
|
|
x_shift = (real32)(advance - width) * .5f - font->chardata[character].xoff;
|
|
y_shift = (real32)font->ascent;
|
|
|
|
x += x_shift;
|
|
y += y_shift;
|
|
|
|
stbtt_aligned_quad q;
|
|
stbtt_GetBakedQuadUnrounded(font->chardata, font->tex_width, font->tex_height, character, &x, &y, &q, 1);
|
|
|
|
draw_set_color(target, color);
|
|
draw_bind_texture(target, font->tex);
|
|
glBegin(GL_QUADS);
|
|
{
|
|
glTexCoord2f(q.s0, q.t1); glVertex2f(q.x0, q.y1);
|
|
glTexCoord2f(q.s1, q.t1); glVertex2f(q.x1, q.y1);
|
|
glTexCoord2f(q.s1, q.t0); glVertex2f(q.x1, q.y0);
|
|
glTexCoord2f(q.s0, q.t0); glVertex2f(q.x0, q.y0);
|
|
}
|
|
glEnd();
|
|
}
|
|
|
|
inline void
|
|
font_draw_glyph_mono(Render_Target *target, Font *font, u16 character,
|
|
real32 x, real32 y, u32 color){
|
|
font_draw_glyph_mono(target, font, character, x, y, (real32)font->advance, color);
|
|
}
|
|
|
|
internal void
|
|
font_draw_glyph(Render_Target *target, Font *font, u16 character,
|
|
real32 x, real32 y, u32 color){
|
|
real32 x_shift, y_shift;
|
|
x_shift = 0;
|
|
y_shift = (real32)font->ascent;
|
|
|
|
x += x_shift;
|
|
y += y_shift;
|
|
|
|
stbtt_aligned_quad q;
|
|
stbtt_GetBakedQuadUnrounded(font->chardata, font->tex_width, font->tex_height, character, &x, &y, &q, 1);
|
|
|
|
draw_set_color(target, color);
|
|
draw_bind_texture(target, font->tex);
|
|
glBegin(GL_QUADS);
|
|
{
|
|
glTexCoord2f(q.s0, q.t1); glVertex2f(q.x0, q.y1);
|
|
glTexCoord2f(q.s1, q.t1); glVertex2f(q.x1, q.y1);
|
|
glTexCoord2f(q.s1, q.t0); glVertex2f(q.x1, q.y0);
|
|
glTexCoord2f(q.s0, q.t0); glVertex2f(q.x0, q.y0);
|
|
}
|
|
glEnd();
|
|
}
|
|
|
|
inline real32
|
|
font_get_glyph_width(Font *font, u16 character){
|
|
return font->chardata[character].xadvance;
|
|
}
|
|
|
|
internal i32
|
|
draw_string(Render_Target *target, Font *font, char *str,
|
|
i32 x_, i32 y, u32 color){
|
|
real32 x = (real32)x_;
|
|
for (i32 i = 0; str[i]; ++i){
|
|
char c = str[i];
|
|
font_draw_glyph(target, font, c,
|
|
x, (real32)y, color);
|
|
x += font_get_glyph_width(font, c);
|
|
}
|
|
return CEIL32(x);
|
|
}
|
|
|
|
internal real32
|
|
draw_string_mono(Render_Target *target, Font *font, char *str,
|
|
real32 x, real32 y, real32 advance, u32 color){
|
|
for (i32 i = 0; str[i]; ++i){
|
|
font_draw_glyph_mono(target, font, str[i],
|
|
x, y, advance, color);
|
|
x += advance;
|
|
}
|
|
return x;
|
|
}
|
|
|
|
internal i32
|
|
draw_string(Render_Target *target, Font *font, String str,
|
|
i32 x_, i32 y, u32 color){
|
|
real32 x = (real32)x_;
|
|
for (i32 i = 0; i < str.size; ++i){
|
|
char c = str.str[i];
|
|
font_draw_glyph(target, font, c,
|
|
x, (real32)y, color);
|
|
x += font_get_glyph_width(font, c);
|
|
}
|
|
return CEIL32(x);
|
|
}
|
|
|
|
internal real32
|
|
draw_string_mono(Render_Target *target, Font *font, String str,
|
|
real32 x, real32 y, real32 advance, u32 color){
|
|
for (i32 i = 0; i < str.size; ++i){
|
|
font_draw_glyph_mono(target, font, str.str[i],
|
|
x, y, advance, color);
|
|
x += advance;
|
|
}
|
|
return x;
|
|
}
|
|
|
|
internal real32
|
|
font_get_max_width(Font *font, char *characters){
|
|
stbtt_bakedchar *chardata = font->chardata;
|
|
real32 cx, x = 0;
|
|
for (i32 i = 0; characters[i]; ++i){
|
|
cx = chardata[characters[i]].xadvance;
|
|
if (x < cx) x = cx;
|
|
}
|
|
return x;
|
|
}
|
|
|
|
internal real32
|
|
font_get_string_width(Font *font, String string){
|
|
real32 result = 0;
|
|
for (i32 i = 0; i < string.size; ++i){
|
|
font_get_glyph_width(font, string.str[i]);
|
|
}
|
|
return result;
|
|
}
|
|
|
|
#endif
|
|
|
|
// BOTTOM
|