fix view not updating with rendered contents
parent
0d8b0b738a
commit
2eaca7a4b9
|
@ -0,0 +1,3 @@
|
||||||
|
TabWidth: 4
|
||||||
|
IndentWidth: 4
|
||||||
|
UseTab: Never
|
103
src/gfx.h
103
src/gfx.h
|
@ -16,6 +16,7 @@
|
||||||
bool keep_running = true;
|
bool keep_running = true;
|
||||||
|
|
||||||
@interface EDGFXView : NSView
|
@interface EDGFXView : NSView
|
||||||
|
@property NSTrackingArea *tracking_area;
|
||||||
@end
|
@end
|
||||||
@interface AppDelegate : NSObject <NSApplicationDelegate>
|
@interface AppDelegate : NSObject <NSApplicationDelegate>
|
||||||
@end
|
@end
|
||||||
|
@ -58,6 +59,8 @@ typedef struct {
|
||||||
EDGFXView *view;
|
EDGFXView *view;
|
||||||
bool keep_running;
|
bool keep_running;
|
||||||
|
|
||||||
|
int mouse_x, mouse_y;
|
||||||
|
|
||||||
// Metal objects
|
// Metal objects
|
||||||
id<MTLDevice> device;
|
id<MTLDevice> device;
|
||||||
CAMetalLayer *metal_layer;
|
CAMetalLayer *metal_layer;
|
||||||
|
@ -70,7 +73,7 @@ typedef struct {
|
||||||
} _metal_gfx_context;
|
} _metal_gfx_context;
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
typedef void (*_gfx_frame_func)();
|
typedef void (*_gfx_frame_func)(int mouse_x, int mouse_y);
|
||||||
typedef struct {
|
typedef struct {
|
||||||
#if defined(__APPLE__)
|
#if defined(__APPLE__)
|
||||||
_metal_gfx_context backend;
|
_metal_gfx_context backend;
|
||||||
|
@ -152,8 +155,58 @@ void gfx_update_buffer(gfx_context_t *cx, size_t buffer_index, const void *data,
|
||||||
return NSViewLayerContentsRedrawOnSetNeedsDisplay;
|
return NSViewLayerContentsRedrawOnSetNeedsDisplay;
|
||||||
}
|
}
|
||||||
|
|
||||||
- (void)insertText:(id)insertString {
|
- (id)initWithFrame:(NSRect)frameRect {
|
||||||
NSLog(@"inserting text: %@", insertString);
|
self = [super initWithFrame:frameRect];
|
||||||
|
NSRect rect = NSMakeRect(
|
||||||
|
0, 0,
|
||||||
|
_gfx_context.frame_width =
|
||||||
|
_gfx_context.backend.window.contentView.frame.size.width,
|
||||||
|
_gfx_context.frame_height =
|
||||||
|
_gfx_context.backend.window.contentView.frame.size.height);
|
||||||
|
|
||||||
|
self.tracking_area = [[NSTrackingArea alloc]
|
||||||
|
initWithRect:rect
|
||||||
|
options:(NSTrackingMouseEnteredAndExited | NSTrackingMouseMoved |
|
||||||
|
NSTrackingActiveInKeyWindow)
|
||||||
|
owner:self
|
||||||
|
userInfo:nil];
|
||||||
|
[self addTrackingArea:self.tracking_area];
|
||||||
|
|
||||||
|
return self;
|
||||||
|
}
|
||||||
|
|
||||||
|
- (void)mouseMoved:(NSEvent *)event {
|
||||||
|
// NSPoint location = NSEvent.mouseLocation;
|
||||||
|
NSPoint location = [self convertPoint:[event locationInWindow]
|
||||||
|
fromView:nil];
|
||||||
|
|
||||||
|
_gfx_context.backend.mouse_x = location.x;
|
||||||
|
_gfx_context.backend.mouse_y = location.y;
|
||||||
|
|
||||||
|
[self setNeedsDisplay:YES];
|
||||||
|
[self displayIfNeeded];
|
||||||
|
[_gfx_context.backend.metal_layer setNeedsDisplay];
|
||||||
|
[_gfx_context.backend.metal_layer displayIfNeeded];
|
||||||
|
}
|
||||||
|
|
||||||
|
- (void)updateTrackingAreas {
|
||||||
|
[self removeTrackingArea:self.tracking_area];
|
||||||
|
[self.tracking_area release];
|
||||||
|
|
||||||
|
NSRect rect = NSMakeRect(
|
||||||
|
0, 0,
|
||||||
|
_gfx_context.frame_width =
|
||||||
|
_gfx_context.backend.window.contentView.frame.size.width,
|
||||||
|
_gfx_context.frame_height =
|
||||||
|
_gfx_context.backend.window.contentView.frame.size.height);
|
||||||
|
|
||||||
|
self.tracking_area = [[NSTrackingArea alloc]
|
||||||
|
initWithRect:rect
|
||||||
|
options:(NSTrackingMouseEnteredAndExited | NSTrackingMouseMoved |
|
||||||
|
NSTrackingActiveInKeyWindow)
|
||||||
|
owner:self
|
||||||
|
userInfo:nil];
|
||||||
|
[self addTrackingArea:self.tracking_area];
|
||||||
}
|
}
|
||||||
@end
|
@end
|
||||||
|
|
||||||
|
@ -168,9 +221,10 @@ static _metal_gfx_context _metal_gfx_init_context(uint32_t width,
|
||||||
NSString *title = @"chat - [Slack Sux]";
|
NSString *title = @"chat - [Slack Sux]";
|
||||||
|
|
||||||
NSRect rect = NSMakeRect(0, 0, width, height);
|
NSRect rect = NSMakeRect(0, 0, width, height);
|
||||||
NSWindow *window = [[NSWindow alloc]
|
NSWindow *window =
|
||||||
initWithContentRect:rect
|
[[NSWindow alloc] initWithContentRect:rect
|
||||||
styleMask:NSWindowStyleMaskTitled | NSWindowStyleMaskClosable |
|
styleMask:NSWindowStyleMaskTitled |
|
||||||
|
NSWindowStyleMaskClosable |
|
||||||
NSWindowStyleMaskMiniaturizable |
|
NSWindowStyleMaskMiniaturizable |
|
||||||
NSWindowStyleMaskResizable
|
NSWindowStyleMaskResizable
|
||||||
backing:NSBackingStoreBuffered
|
backing:NSBackingStoreBuffered
|
||||||
|
@ -267,8 +321,8 @@ static _metal_gfx_context _metal_gfx_init_context(uint32_t width,
|
||||||
MTLBlendOperationAdd;
|
MTLBlendOperationAdd;
|
||||||
ui_rect_pipeline_descriptor.colorAttachments[0].sourceAlphaBlendFactor =
|
ui_rect_pipeline_descriptor.colorAttachments[0].sourceAlphaBlendFactor =
|
||||||
MTLBlendFactorSourceAlpha;
|
MTLBlendFactorSourceAlpha;
|
||||||
ui_rect_pipeline_descriptor.colorAttachments[0].destinationAlphaBlendFactor =
|
ui_rect_pipeline_descriptor.colorAttachments[0]
|
||||||
MTLBlendFactorOneMinusSourceAlpha;
|
.destinationAlphaBlendFactor = MTLBlendFactorOneMinusSourceAlpha;
|
||||||
ui_rect_pipeline_descriptor.colorAttachments[0].sourceRGBBlendFactor =
|
ui_rect_pipeline_descriptor.colorAttachments[0].sourceRGBBlendFactor =
|
||||||
MTLBlendFactorSourceAlpha;
|
MTLBlendFactorSourceAlpha;
|
||||||
ui_rect_pipeline_descriptor.colorAttachments[0].destinationRGBBlendFactor =
|
ui_rect_pipeline_descriptor.colorAttachments[0].destinationRGBBlendFactor =
|
||||||
|
@ -288,9 +342,9 @@ static _metal_gfx_context _metal_gfx_init_context(uint32_t width,
|
||||||
|
|
||||||
exit(1);
|
exit(1);
|
||||||
}
|
}
|
||||||
pushArray(_MTLRenderPipelineState, &pipelines,
|
pushArray(
|
||||||
[device
|
_MTLRenderPipelineState, &pipelines,
|
||||||
newRenderPipelineStateWithDescriptor:ui_rect_pipeline_descriptor
|
[device newRenderPipelineStateWithDescriptor:ui_rect_pipeline_descriptor
|
||||||
error:&pipeline_error]);
|
error:&pipeline_error]);
|
||||||
|
|
||||||
array(_MTLBuffer) buffers = newArray(_MTLBuffer, 8);
|
array(_MTLBuffer) buffers = newArray(_MTLBuffer, 8);
|
||||||
|
@ -332,7 +386,8 @@ static void _metal_gfx_send_events(_metal_gfx_context *cx) {
|
||||||
static void _metal_gfx_present(_metal_gfx_context *cx) {
|
static void _metal_gfx_present(_metal_gfx_context *cx) {
|
||||||
_gfx_context.gpu_glyphs.size = 0;
|
_gfx_context.gpu_glyphs.size = 0;
|
||||||
_gfx_context.gpu_ui_rects.size = 0;
|
_gfx_context.gpu_ui_rects.size = 0;
|
||||||
_gfx_context.frame_func();
|
_gfx_context.frame_func(cx->mouse_x,
|
||||||
|
_gfx_context.frame_height - cx->mouse_y);
|
||||||
|
|
||||||
if (_gfx_context.gpu_glyphs.size > 0) {
|
if (_gfx_context.gpu_glyphs.size > 0) {
|
||||||
gfx_update_buffer(&_gfx_context, 2, _gfx_context.gpu_glyphs.data,
|
gfx_update_buffer(&_gfx_context, 2, _gfx_context.gpu_glyphs.data,
|
||||||
|
@ -354,6 +409,7 @@ static void _metal_gfx_present(_metal_gfx_context *cx) {
|
||||||
gfx_update_buffer(&_gfx_context, 3, &gpu_uniform_params,
|
gfx_update_buffer(&_gfx_context, 3, &gpu_uniform_params,
|
||||||
sizeof(GpuUniformParams));
|
sizeof(GpuUniformParams));
|
||||||
|
|
||||||
|
@autoreleasepool {
|
||||||
id<CAMetalDrawable> drawable = [cx->metal_layer nextDrawable];
|
id<CAMetalDrawable> drawable = [cx->metal_layer nextDrawable];
|
||||||
|
|
||||||
id<MTLCommandBuffer> command_buffer = [cx->command_queue commandBuffer];
|
id<MTLCommandBuffer> command_buffer = [cx->command_queue commandBuffer];
|
||||||
|
@ -365,13 +421,14 @@ static void _metal_gfx_present(_metal_gfx_context *cx) {
|
||||||
MTLClearColorMake(0, 0, 0, 1);
|
MTLClearColorMake(0, 0, 0, 1);
|
||||||
render_pass_desc.colorAttachments[0].storeAction = MTLStoreActionStore;
|
render_pass_desc.colorAttachments[0].storeAction = MTLStoreActionStore;
|
||||||
|
|
||||||
id<MTLRenderCommandEncoder> encoder =
|
id<MTLRenderCommandEncoder> encoder = [command_buffer
|
||||||
[command_buffer renderCommandEncoderWithDescriptor:render_pass_desc];
|
renderCommandEncoderWithDescriptor:render_pass_desc];
|
||||||
|
|
||||||
if (_gfx_context.gpu_ui_rects.size > 0) {
|
if (_gfx_context.gpu_ui_rects.size > 0) {
|
||||||
// UI Rects
|
// UI Rects
|
||||||
[encoder setRenderPipelineState:cx->pipelines.data[1]];
|
[encoder setRenderPipelineState:cx->pipelines.data[1]];
|
||||||
// FIXME: allow these to be described by the user instead of hardcoded
|
// FIXME: allow these to be described by the user instead of
|
||||||
|
// hardcoded
|
||||||
[encoder setVertexBuffer:cx->buffers.data[0]
|
[encoder setVertexBuffer:cx->buffers.data[0]
|
||||||
offset:0
|
offset:0
|
||||||
atIndex:0]; // vertices
|
atIndex:0]; // vertices
|
||||||
|
@ -381,7 +438,6 @@ static void _metal_gfx_present(_metal_gfx_context *cx) {
|
||||||
[encoder setVertexBuffer:cx->buffers.data[3]
|
[encoder setVertexBuffer:cx->buffers.data[3]
|
||||||
offset:0
|
offset:0
|
||||||
atIndex:2]; // uniforms
|
atIndex:2]; // uniforms
|
||||||
// TODO: get instance count properly
|
|
||||||
[encoder drawIndexedPrimitives:MTLPrimitiveTypeTriangle
|
[encoder drawIndexedPrimitives:MTLPrimitiveTypeTriangle
|
||||||
indexCount:6
|
indexCount:6
|
||||||
indexType:MTLIndexTypeUInt16
|
indexType:MTLIndexTypeUInt16
|
||||||
|
@ -393,7 +449,8 @@ static void _metal_gfx_present(_metal_gfx_context *cx) {
|
||||||
if (_gfx_context.gpu_glyphs.size > 0) {
|
if (_gfx_context.gpu_glyphs.size > 0) {
|
||||||
// UI Text
|
// UI Text
|
||||||
[encoder setRenderPipelineState:cx->pipelines.data[0]];
|
[encoder setRenderPipelineState:cx->pipelines.data[0]];
|
||||||
// FIXME: allow these to be described by the user instead of hardcoded
|
// FIXME: allow these to be described by the user instead of
|
||||||
|
// hardcoded
|
||||||
[encoder setVertexBuffer:cx->buffers.data[0]
|
[encoder setVertexBuffer:cx->buffers.data[0]
|
||||||
offset:0
|
offset:0
|
||||||
atIndex:0]; // vertices
|
atIndex:0]; // vertices
|
||||||
|
@ -404,7 +461,6 @@ static void _metal_gfx_present(_metal_gfx_context *cx) {
|
||||||
offset:0
|
offset:0
|
||||||
atIndex:2]; // uniforms
|
atIndex:2]; // uniforms
|
||||||
[encoder setFragmentTexture:cx->textures.data[0] atIndex:0];
|
[encoder setFragmentTexture:cx->textures.data[0] atIndex:0];
|
||||||
// TODO: get instance count properly
|
|
||||||
[encoder drawIndexedPrimitives:MTLPrimitiveTypeTriangle
|
[encoder drawIndexedPrimitives:MTLPrimitiveTypeTriangle
|
||||||
indexCount:6
|
indexCount:6
|
||||||
indexType:MTLIndexTypeUInt16
|
indexType:MTLIndexTypeUInt16
|
||||||
|
@ -414,11 +470,13 @@ static void _metal_gfx_present(_metal_gfx_context *cx) {
|
||||||
}
|
}
|
||||||
|
|
||||||
[encoder endEncoding];
|
[encoder endEncoding];
|
||||||
[command_buffer presentDrawable:drawable];
|
[command_buffer presentDrawable:drawable
|
||||||
|
afterMinimumDuration:1.0 / 144.0];
|
||||||
[command_buffer commit];
|
[command_buffer commit];
|
||||||
|
|
||||||
[command_buffer waitUntilScheduled];
|
[command_buffer waitUntilScheduled];
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
static size_t _metal_gfx_push_texture_buffer(_metal_gfx_context *cx,
|
static size_t _metal_gfx_push_texture_buffer(_metal_gfx_context *cx,
|
||||||
uint32_t width, uint32_t height,
|
uint32_t width, uint32_t height,
|
||||||
|
@ -496,7 +554,8 @@ void gfx_run_events(gfx_context_t *cx) {
|
||||||
size_t gfx_push_texture_buffer(gfx_context_t *cx, uint32_t width,
|
size_t gfx_push_texture_buffer(gfx_context_t *cx, uint32_t width,
|
||||||
uint32_t height, const void *data, size_t len) {
|
uint32_t height, const void *data, size_t len) {
|
||||||
#if defined(__APPLE__)
|
#if defined(__APPLE__)
|
||||||
return _metal_gfx_push_texture_buffer(&cx->backend, width, height, data, len);
|
return _metal_gfx_push_texture_buffer(&cx->backend, width, height, data,
|
||||||
|
len);
|
||||||
#else
|
#else
|
||||||
#error "Unsupported graphics backend"
|
#error "Unsupported graphics backend"
|
||||||
#endif
|
#endif
|
||||||
|
@ -596,8 +655,8 @@ void *gfx_init_context(_gfx_frame_func frame_func, uint32_t width,
|
||||||
gfx_allocate_vertex_buffer(&_gfx_context, _gfx_context.gpu_glyphs.capacity *
|
gfx_allocate_vertex_buffer(&_gfx_context, _gfx_context.gpu_glyphs.capacity *
|
||||||
sizeof(GpuGlyph));
|
sizeof(GpuGlyph));
|
||||||
gfx_allocate_vertex_buffer(&_gfx_context, sizeof(GpuUniformParams));
|
gfx_allocate_vertex_buffer(&_gfx_context, sizeof(GpuUniformParams));
|
||||||
gfx_allocate_vertex_buffer(&_gfx_context, _gfx_context.gpu_ui_rects.capacity *
|
gfx_allocate_vertex_buffer(
|
||||||
sizeof(GpuUiRect));
|
&_gfx_context, _gfx_context.gpu_ui_rects.capacity * sizeof(GpuUiRect));
|
||||||
|
|
||||||
return &_gfx_context;
|
return &_gfx_context;
|
||||||
}
|
}
|
||||||
|
|
30
src/main.c
30
src/main.c
|
@ -48,7 +48,8 @@ void ed_init(_gfx_frame_func frame_func) {
|
||||||
ttf_buffer));
|
ttf_buffer));
|
||||||
|
|
||||||
stbtt_fontinfo font;
|
stbtt_fontinfo font;
|
||||||
stbtt_InitFont(&font, ttf_buffer, stbtt_GetFontOffsetForIndex(ttf_buffer, 0));
|
stbtt_InitFont(&font, ttf_buffer,
|
||||||
|
stbtt_GetFontOffsetForIndex(ttf_buffer, 0));
|
||||||
|
|
||||||
const int font_bitmap_size = 512;
|
const int font_bitmap_size = 512;
|
||||||
const int rasterized_font_height = _FONT_HEIGHT;
|
const int rasterized_font_height = _FONT_HEIGHT;
|
||||||
|
@ -77,8 +78,8 @@ void ed_init(_gfx_frame_func frame_func) {
|
||||||
int y = 0;
|
int y = 0;
|
||||||
for (size_t i = 33; i < 33 + 96; ++i) {
|
for (size_t i = 33; i < 33 + 96; ++i) {
|
||||||
int width, height, xoff, yoff;
|
int width, height, xoff, yoff;
|
||||||
uint8_t *bitmap = stbtt_GetCodepointBitmap(&font, scale, scale, (int)i,
|
uint8_t *bitmap = stbtt_GetCodepointBitmap(
|
||||||
&width, &height, &xoff, &yoff);
|
&font, scale, scale, (int)i, &width, &height, &xoff, &yoff);
|
||||||
|
|
||||||
if (x + width >= font_bitmap_size) {
|
if (x + width >= font_bitmap_size) {
|
||||||
x = 0;
|
x = 0;
|
||||||
|
@ -89,7 +90,8 @@ void ed_init(_gfx_frame_func frame_func) {
|
||||||
for (size_t xx = 0; xx < (size_t)width; ++xx) {
|
for (size_t xx = 0; xx < (size_t)width; ++xx) {
|
||||||
size_t yyy = y;
|
size_t yyy = y;
|
||||||
for (size_t yy = 0; yy < (size_t)height; ++yy) {
|
for (size_t yy = 0; yy < (size_t)height; ++yy) {
|
||||||
font_bitmap[xxx + yyy * font_bitmap_size] = bitmap[xx + yy * width];
|
font_bitmap[xxx + yyy * font_bitmap_size] =
|
||||||
|
bitmap[xx + yy * width];
|
||||||
|
|
||||||
yyy += 1;
|
yyy += 1;
|
||||||
}
|
}
|
||||||
|
@ -120,16 +122,27 @@ void render_ui_rect(float position[2], float size[2], float color[4]) {
|
||||||
gfx_queue_ui_rect(state.gfx_cx, position, size, 0, color);
|
gfx_queue_ui_rect(state.gfx_cx, position, size, 0, color);
|
||||||
}
|
}
|
||||||
|
|
||||||
void ed_frame() {
|
void ed_frame(int mouse_x, int mouse_y) {
|
||||||
state.ui_cx.frame_elements.data[0].size.computed_size[0] =
|
state.ui_cx.frame_elements.data[0].size.computed_size[0] =
|
||||||
state.gfx_cx->frame_width;
|
state.gfx_cx->frame_width;
|
||||||
state.ui_cx.frame_elements.data[0].size.computed_size[1] =
|
state.ui_cx.frame_elements.data[0].size.computed_size[1] =
|
||||||
state.gfx_cx->frame_height;
|
state.gfx_cx->frame_height;
|
||||||
|
|
||||||
|
uint8_t buffer[256] = {};
|
||||||
|
snprintf(buffer, 256, "Mouse X: %d, Mouse Y: %d, Mouse %s", mouse_x,
|
||||||
|
mouse_y, state.ui_cx.input.mouse_left_down ? "Down" : "Up");
|
||||||
|
|
||||||
|
render_ui_rect((float[2]){mouse_x, mouse_y}, (float[2]){32, 32},
|
||||||
|
(float[4]){1, 1, 1, 1});
|
||||||
|
|
||||||
ui_element(&state.ui_cx, _String("channel sidebar"), UI_AXIS_VERTICAL,
|
ui_element(&state.ui_cx, _String("channel sidebar"), UI_AXIS_VERTICAL,
|
||||||
ui_make_size(ui_children_sum, ui_fill), UI_FLAG_DRAW_BACKGROUND);
|
ui_make_size(ui_children_sum, ui_fill), UI_FLAG_DRAW_BACKGROUND);
|
||||||
ui_push_parent(&state.ui_cx);
|
ui_push_parent(&state.ui_cx);
|
||||||
{
|
{
|
||||||
|
ui_element(&state.ui_cx, _String(buffer), UI_AXIS_HORIZONTAL,
|
||||||
|
ui_make_size(ui_fit_text, ui_fit_text),
|
||||||
|
UI_FLAG_DRAW_BACKGROUND | UI_FLAG_DRAW_TEXT);
|
||||||
|
|
||||||
ui_element(&state.ui_cx, _String("#dev-general"), UI_AXIS_HORIZONTAL,
|
ui_element(&state.ui_cx, _String("#dev-general"), UI_AXIS_HORIZONTAL,
|
||||||
ui_make_size(ui_fit_text, ui_fit_text),
|
ui_make_size(ui_fit_text, ui_fit_text),
|
||||||
UI_FLAG_DRAW_BACKGROUND | UI_FLAG_DRAW_TEXT);
|
UI_FLAG_DRAW_BACKGROUND | UI_FLAG_DRAW_TEXT);
|
||||||
|
@ -144,6 +157,13 @@ void ed_frame() {
|
||||||
|
|
||||||
ui_update_cache(&state.ui_cx, 0);
|
ui_update_cache(&state.ui_cx, 0);
|
||||||
ui_prune(&state.ui_cx);
|
ui_prune(&state.ui_cx);
|
||||||
|
|
||||||
|
ui_update_input(&state.ui_cx, (ui_context_input){
|
||||||
|
.mouse_x = mouse_x,
|
||||||
|
.mouse_y = mouse_y,
|
||||||
|
.mouse_left_down = false,
|
||||||
|
.mouse_right_down = false,
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
int main(int argc, char *argv[]) {
|
int main(int argc, char *argv[]) {
|
||||||
|
|
14
src/string.h
14
src/string.h
|
@ -1,12 +1,13 @@
|
||||||
#ifndef ED_STRING_INCLUDED
|
#ifndef ED_STRING_INCLUDED
|
||||||
#define ED_STRING_INCLUDED
|
#define ED_STRING_INCLUDED
|
||||||
|
|
||||||
#include <stdlib.h>
|
#include <stdbool.h>
|
||||||
#include <stddef.h>
|
#include <stddef.h>
|
||||||
#include <stdint.h>
|
#include <stdint.h>
|
||||||
#include <stdbool.h>
|
#include <stdlib.h>
|
||||||
|
|
||||||
#define _String(text) ((string) { .data = (uint8_t*) text, .len = sizeof(text), .owned = false })
|
#define _String(text) \
|
||||||
|
((string){.data = (uint8_t *)text, .len = sizeof(text), .owned = false})
|
||||||
typedef struct {
|
typedef struct {
|
||||||
uint8_t *data;
|
uint8_t *data;
|
||||||
size_t len;
|
size_t len;
|
||||||
|
@ -14,13 +15,14 @@ typedef struct {
|
||||||
bool owned;
|
bool owned;
|
||||||
} string;
|
} string;
|
||||||
|
|
||||||
|
|
||||||
#ifdef ED_STRING_IMPLEMENTATION
|
#ifdef ED_STRING_IMPLEMENTATION
|
||||||
bool string_eq(string a, string b) {
|
bool string_eq(string a, string b) {
|
||||||
if (a.len != b.len) return false;
|
if (a.len != b.len)
|
||||||
|
return false;
|
||||||
|
|
||||||
for (size_t i = 0; i < a.len; ++i) {
|
for (size_t i = 0; i < a.len; ++i) {
|
||||||
if (a.data[i] != b.data[i]) return false;
|
if (a.data[i] != b.data[i])
|
||||||
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
|
|
38
src/ui.h
38
src/ui.h
|
@ -69,6 +69,12 @@ typedef struct {
|
||||||
uint32_t computed_pos[2];
|
uint32_t computed_pos[2];
|
||||||
} ui_size;
|
} ui_size;
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
bool hovering;
|
||||||
|
bool clicked;
|
||||||
|
bool dragging;
|
||||||
|
} ui_interaction;
|
||||||
|
|
||||||
// UI Element data persisted across frames
|
// UI Element data persisted across frames
|
||||||
typedef struct {
|
typedef struct {
|
||||||
string label;
|
string label;
|
||||||
|
@ -106,11 +112,22 @@ typedef struct {
|
||||||
} ui_element_frame_data;
|
} ui_element_frame_data;
|
||||||
arrayTemplate(ui_element_frame_data);
|
arrayTemplate(ui_element_frame_data);
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
bool mouse_left_down;
|
||||||
|
bool mouse_right_down;
|
||||||
|
|
||||||
|
uint32_t mouse_x;
|
||||||
|
uint32_t mouse_y;
|
||||||
|
} ui_context_input;
|
||||||
|
|
||||||
typedef struct {
|
typedef struct {
|
||||||
ed_ht cached_elements;
|
ed_ht cached_elements;
|
||||||
array(ui_element_frame_data) frame_elements;
|
array(ui_element_frame_data) frame_elements;
|
||||||
array(ui_element_frame_data) frame_floating_elements;
|
array(ui_element_frame_data) frame_floating_elements;
|
||||||
|
|
||||||
|
ui_context_input input;
|
||||||
|
ui_context_input last_input;
|
||||||
|
|
||||||
size_t frame_index;
|
size_t frame_index;
|
||||||
uint32_t canvas_size[2];
|
uint32_t canvas_size[2];
|
||||||
|
|
||||||
|
@ -276,9 +293,9 @@ static void _ui_compute_children_layout(ui_context *cx,
|
||||||
uint32_t child_size[2] = {0, 0};
|
uint32_t child_size[2] = {0, 0};
|
||||||
|
|
||||||
// NOTE: the number of fills for the opposite axis of this box needs to be 1
|
// NOTE: the number of fills for the opposite axis of this box needs to be 1
|
||||||
// because it will never get incremented in the loop below and cause a divide
|
// because it will never get incremented in the loop below and cause a
|
||||||
// by zero and the number of fills for the axis of the box needs to start at
|
// divide by zero and the number of fills for the axis of the box needs to
|
||||||
// zero or else it will be n+1 causing incorrect sizes
|
// start at zero or else it will be n+1 causing incorrect sizes
|
||||||
uint32_t num_fills[2] = {1, 1};
|
uint32_t num_fills[2] = {1, 1};
|
||||||
num_fills[elm->size.axis] = 0;
|
num_fills[elm->size.axis] = 0;
|
||||||
|
|
||||||
|
@ -473,11 +490,13 @@ void ui_prune(ui_context *cx) {
|
||||||
if (cx->cached_elements.key_slots[i].key.data != NULL) {
|
if (cx->cached_elements.key_slots[i].key.data != NULL) {
|
||||||
string key = cx->cached_elements.key_slots[i].key;
|
string key = cx->cached_elements.key_slots[i].key;
|
||||||
|
|
||||||
// if this element hasn't been created in the past 5 frames, remove it
|
// if this element hasn't been created in the past 5 frames, remove
|
||||||
|
// it
|
||||||
ui_element_cache_data *cached = ht_get(&cx->cached_elements, key);
|
ui_element_cache_data *cached = ht_get(&cx->cached_elements, key);
|
||||||
if (cached && cached->last_instantiated_index < cx->frame_index - 5) {
|
if (cached &&
|
||||||
// fprintf(stderr, "removing %.*s from cache, cache index: %zu, frame
|
cached->last_instantiated_index < cx->frame_index - 5) {
|
||||||
// index: %zu\n", (int)key.len, key.data,
|
// fprintf(stderr, "removing %.*s from cache, cache index: %zu,
|
||||||
|
// frame index: %zu\n", (int)key.len, key.data,
|
||||||
// cached->last_instantiated_index, cx->frame_index);
|
// cached->last_instantiated_index, cx->frame_index);
|
||||||
|
|
||||||
ht_remove(&cx->cached_elements, key);
|
ht_remove(&cx->cached_elements, key);
|
||||||
|
@ -503,5 +522,10 @@ void ui_prune(ui_context *cx) {
|
||||||
cx->current_parent = 0;
|
cx->current_parent = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void ui_update_input(ui_context *cx, ui_context_input new_input) {
|
||||||
|
cx->last_input = cx->input;
|
||||||
|
cx->input = new_input;
|
||||||
|
}
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
#endif
|
#endif
|
||||||
|
|
Loading…
Reference in New Issue