From 2eaca7a4b9a902bbcf2ebe96da37044bd67bf506 Mon Sep 17 00:00:00 2001 From: Patrick Cleavelin Date: Tue, 16 Apr 2024 16:14:14 -0500 Subject: [PATCH] fix view not updating with rendered contents --- .clang-format | 3 + src/gfx.h | 795 +++++++++++++++++++++++++++----------------------- src/main.c | 210 +++++++------ src/string.h | 18 +- src/ui.h | 716 +++++++++++++++++++++++---------------------- 5 files changed, 925 insertions(+), 817 deletions(-) create mode 100644 .clang-format diff --git a/.clang-format b/.clang-format new file mode 100644 index 0000000..d544a1c --- /dev/null +++ b/.clang-format @@ -0,0 +1,3 @@ +TabWidth: 4 +IndentWidth: 4 +UseTab: Never diff --git a/src/gfx.h b/src/gfx.h index 8d9e96c..e1bf18d 100644 --- a/src/gfx.h +++ b/src/gfx.h @@ -16,6 +16,7 @@ bool keep_running = true; @interface EDGFXView : NSView +@property NSTrackingArea *tracking_area; @end @interface AppDelegate : NSObject @end @@ -23,68 +24,70 @@ bool keep_running = true; @end #define wrapIdArray(T) \ - typedef id _##T; \ - arrayTemplate(_##T); + typedef id _##T; \ + arrayTemplate(_##T); wrapIdArray(MTLRenderPipelineState); wrapIdArray(MTLBuffer); wrapIdArray(MTLTexture); typedef struct { - float position[4]; - float size[4]; - float border_size[4]; - float color[4]; + float position[4]; + float size[4]; + float border_size[4]; + float color[4]; } GpuUiRect; arrayTemplate(GpuUiRect); typedef struct { - float atlas_position[2]; - float size[2]; - float position[2]; - float y_offset; - float _haha_alignment; + float atlas_position[2]; + float size[2]; + float position[2]; + float y_offset; + float _haha_alignment; } GpuGlyph; arrayTemplate(GpuGlyph); typedef struct { - float screen_size[2]; + float screen_size[2]; } GpuUniformParams; #if defined(__APPLE__) typedef struct { - NSApplication *application; - NSWindow *window; - EDGFXView *view; - bool keep_running; + NSApplication *application; + NSWindow *window; + EDGFXView *view; + bool keep_running; - // Metal objects - id device; - CAMetalLayer *metal_layer; - id library; - id command_queue; + int mouse_x, mouse_y; - array(_MTLRenderPipelineState) pipelines; - array(_MTLBuffer) buffers; - array(_MTLTexture) textures; + // Metal objects + id device; + CAMetalLayer *metal_layer; + id library; + id command_queue; + + array(_MTLRenderPipelineState) pipelines; + array(_MTLBuffer) buffers; + array(_MTLTexture) textures; } _metal_gfx_context; #endif -typedef void (*_gfx_frame_func)(); +typedef void (*_gfx_frame_func)(int mouse_x, int mouse_y); typedef struct { #if defined(__APPLE__) - _metal_gfx_context backend; + _metal_gfx_context backend; #else #error "Unsupported platform" #endif - uint32_t frame_width; - uint32_t frame_height; - _gfx_frame_func frame_func; + uint32_t frame_width; + uint32_t frame_height; + _gfx_frame_func frame_func; - array(GpuUiRect) gpu_ui_rects; - array(GpuGlyph) gpu_glyphs; - array(GpuGlyph) glyph_cache; + array(GpuUiRect) gpu_ui_rects; + array(GpuGlyph) gpu_glyphs; + array(GpuGlyph) glyph_cache; } gfx_context_t; static gfx_context_t _gfx_context; @@ -100,394 +103,449 @@ void gfx_update_buffer(gfx_context_t *cx, size_t buffer_index, const void *data, @implementation AppDelegate - (NSApplicationTerminateReply)applicationShouldTerminate: (NSApplication *)sender { - keep_running = false; + keep_running = false; - return NSTerminateCancel; + return NSTerminateCancel; } - (BOOL)applicationShouldTerminateAfterLastWindowClosed: (NSApplication *)sender { - keep_running = false; + keep_running = false; - return NO; + return NO; } @end @implementation WindowDelegate - (BOOL)windowShouldClose:(NSApplication *)sender { - keep_running = false; - return YES; + keep_running = false; + return YES; } - (void)windowDidResize:(NSNotification *)notification { - _gfx_context.frame_width = - _gfx_context.backend.window.contentView.frame.size.width; - _gfx_context.frame_height = - _gfx_context.backend.window.contentView.frame.size.height; + _gfx_context.frame_width = + _gfx_context.backend.window.contentView.frame.size.width; + _gfx_context.frame_height = + _gfx_context.backend.window.contentView.frame.size.height; - CGFloat scale = _gfx_context.backend.metal_layer.contentsScale; - [_gfx_context.backend.metal_layer - setDrawableSize:CGSizeMake(_gfx_context.frame_width * scale, - _gfx_context.frame_height * scale)]; + CGFloat scale = _gfx_context.backend.metal_layer.contentsScale; + [_gfx_context.backend.metal_layer + setDrawableSize:CGSizeMake(_gfx_context.frame_width * scale, + _gfx_context.frame_height * scale)]; - _gfx_context.backend.view.needsDisplay = true; + _gfx_context.backend.view.needsDisplay = true; } @end @implementation EDGFXView - (BOOL)isOpaque { - return YES; + return YES; } - (void)updateLayer { - _metal_gfx_present(&_gfx_context.backend); + _metal_gfx_present(&_gfx_context.backend); } - (BOOL)wantsLayer { - return YES; + return YES; } - (BOOL)wantsUpdateLayer { - return YES; + return YES; } - (NSViewLayerContentsRedrawPolicy)layerContentsRedrawPolicy { - return NSViewLayerContentsRedrawOnSetNeedsDisplay; + return NSViewLayerContentsRedrawOnSetNeedsDisplay; } -- (void)insertText:(id)insertString { - NSLog(@"inserting text: %@", insertString); +- (id)initWithFrame:(NSRect)frameRect { + 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 static _metal_gfx_context _metal_gfx_init_context(uint32_t width, uint32_t height) { - NSApplication *application = [NSApplication sharedApplication]; - if (application == NULL) { - fprintf(stderr, "NSApplication:sharedApplication failed\n"); - exit(1); - } - - NSString *title = @"chat - [Slack Sux]"; - - NSRect rect = NSMakeRect(0, 0, width, height); - NSWindow *window = [[NSWindow alloc] - initWithContentRect:rect - styleMask:NSWindowStyleMaskTitled | NSWindowStyleMaskClosable | - NSWindowStyleMaskMiniaturizable | - NSWindowStyleMaskResizable - backing:NSBackingStoreBuffered - defer:NO]; - EDGFXView *view = [[EDGFXView alloc] initWithFrame:rect]; - [view updateTrackingAreas]; - [window setTitle:title]; - [window setContentView:view]; - [window setDelegate:[[WindowDelegate alloc] init]]; - [window makeKeyAndOrderFront:NULL]; - - // TODO: make this work - // if (application.mainMenu == NULL) { - // NSMenu *menu = [[NSMenu alloc] initWithTitle:@"an_editor"]; - // if (menu == NULL) { - // fprintf(stderr, "failed to create application menu\n"); - // exit(1); - // } - // application.mainMenu = menu; - // } - [application setDelegate:[[AppDelegate alloc] init]]; - [application setActivationPolicy:NSApplicationActivationPolicyRegular]; - [application setPresentationOptions:NSApplicationPresentationDefault]; - [application finishLaunching]; - - id device = MTLCreateSystemDefaultDevice(); - - CAMetalLayer *metal_layer = [CAMetalLayer layer]; - metal_layer.device = device; - metal_layer.pixelFormat = MTLPixelFormatRGBA8Unorm; - metal_layer.frame = CGRectMake(0, 0, width, height); - metal_layer.needsDisplayOnBoundsChange = YES; - metal_layer.presentsWithTransaction = YES; - metal_layer.autoresizingMask = kCALayerWidthSizable | kCALayerHeightSizable; - - // TODO: set this to the display dpi scale - metal_layer.contentsScale = 2.0; - view.wantsLayer = YES; - [view.layer addSublayer:metal_layer]; - view.layerContentsRedrawPolicy = NSViewLayerContentsRedrawOnSetNeedsDisplay; - - NSError *libraryError = NULL; - NSURL *libraryURL = [[NSBundle mainBundle] URLForResource:@"./shaders" - withExtension:@"metallib"]; - if (libraryURL == NULL) { - fprintf(stderr, "Couldn't find library file\n"); - exit(1); - } - - id library = [device newLibraryWithURL:libraryURL - error:&libraryError]; - - if (library == NULL) { - if (libraryError.description != NULL) { - NSLog(@"Error description: %@\n", libraryError.description); + NSApplication *application = [NSApplication sharedApplication]; + if (application == NULL) { + fprintf(stderr, "NSApplication:sharedApplication failed\n"); + exit(1); } - exit(1); - } + NSString *title = @"chat - [Slack Sux]"; - id command_queue = [device newCommandQueue]; - id vertex_func = [library newFunctionWithName:@"vs_main"]; - id fragment_func = [library newFunctionWithName:@"fs_main"]; - id ui_rect_vertex_func = - [library newFunctionWithName:@"ui_rect_vs"]; - id ui_rect_fragment_func = - [library newFunctionWithName:@"ui_rect_fs"]; + NSRect rect = NSMakeRect(0, 0, width, height); + NSWindow *window = + [[NSWindow alloc] initWithContentRect:rect + styleMask:NSWindowStyleMaskTitled | + NSWindowStyleMaskClosable | + NSWindowStyleMaskMiniaturizable | + NSWindowStyleMaskResizable + backing:NSBackingStoreBuffered + defer:NO]; + EDGFXView *view = [[EDGFXView alloc] initWithFrame:rect]; + [view updateTrackingAreas]; + [window setTitle:title]; + [window setContentView:view]; + [window setDelegate:[[WindowDelegate alloc] init]]; + [window makeKeyAndOrderFront:NULL]; - MTLRenderPipelineDescriptor *pipeline_descriptor = - [[MTLRenderPipelineDescriptor alloc] init]; - [pipeline_descriptor setVertexFunction:vertex_func]; - [pipeline_descriptor setFragmentFunction:fragment_func]; - pipeline_descriptor.colorAttachments[0].pixelFormat = - MTLPixelFormatRGBA8Unorm; - pipeline_descriptor.colorAttachments[0].alphaBlendOperation = - MTLBlendOperationAdd; - pipeline_descriptor.colorAttachments[0].sourceAlphaBlendFactor = - MTLBlendFactorSourceAlpha; - pipeline_descriptor.colorAttachments[0].destinationAlphaBlendFactor = - MTLBlendFactorOneMinusSourceAlpha; - pipeline_descriptor.colorAttachments[0].sourceRGBBlendFactor = - MTLBlendFactorSourceAlpha; - pipeline_descriptor.colorAttachments[0].destinationRGBBlendFactor = - MTLBlendFactorOneMinusSourceAlpha; - pipeline_descriptor.colorAttachments[0].blendingEnabled = true; + // TODO: make this work + // if (application.mainMenu == NULL) { + // NSMenu *menu = [[NSMenu alloc] initWithTitle:@"an_editor"]; + // if (menu == NULL) { + // fprintf(stderr, "failed to create application menu\n"); + // exit(1); + // } + // application.mainMenu = menu; + // } + [application setDelegate:[[AppDelegate alloc] init]]; + [application setActivationPolicy:NSApplicationActivationPolicyRegular]; + [application setPresentationOptions:NSApplicationPresentationDefault]; + [application finishLaunching]; - MTLRenderPipelineDescriptor *ui_rect_pipeline_descriptor = - [[MTLRenderPipelineDescriptor alloc] init]; - [ui_rect_pipeline_descriptor setVertexFunction:ui_rect_vertex_func]; - [ui_rect_pipeline_descriptor setFragmentFunction:ui_rect_fragment_func]; - ui_rect_pipeline_descriptor.colorAttachments[0].pixelFormat = - MTLPixelFormatRGBA8Unorm; - ui_rect_pipeline_descriptor.colorAttachments[0].alphaBlendOperation = - MTLBlendOperationAdd; - ui_rect_pipeline_descriptor.colorAttachments[0].sourceAlphaBlendFactor = - MTLBlendFactorSourceAlpha; - ui_rect_pipeline_descriptor.colorAttachments[0].destinationAlphaBlendFactor = - MTLBlendFactorOneMinusSourceAlpha; - ui_rect_pipeline_descriptor.colorAttachments[0].sourceRGBBlendFactor = - MTLBlendFactorSourceAlpha; - ui_rect_pipeline_descriptor.colorAttachments[0].destinationRGBBlendFactor = - MTLBlendFactorOneMinusSourceAlpha; - ui_rect_pipeline_descriptor.colorAttachments[0].blendingEnabled = true; + id device = MTLCreateSystemDefaultDevice(); - NSError *pipeline_error = NULL; - array(_MTLRenderPipelineState) pipelines = - newArray(_MTLRenderPipelineState, 2); - pushArray(_MTLRenderPipelineState, &pipelines, - [device newRenderPipelineStateWithDescriptor:pipeline_descriptor - error:&pipeline_error]); - if (pipeline_error != NULL) { - if (pipeline_error.description != NULL) { - NSLog(@"Error description: %@\n", pipeline_error.description); + CAMetalLayer *metal_layer = [CAMetalLayer layer]; + metal_layer.device = device; + metal_layer.pixelFormat = MTLPixelFormatRGBA8Unorm; + metal_layer.frame = CGRectMake(0, 0, width, height); + metal_layer.needsDisplayOnBoundsChange = YES; + metal_layer.presentsWithTransaction = YES; + metal_layer.autoresizingMask = kCALayerWidthSizable | kCALayerHeightSizable; + + // TODO: set this to the display dpi scale + metal_layer.contentsScale = 2.0; + view.wantsLayer = YES; + [view.layer addSublayer:metal_layer]; + view.layerContentsRedrawPolicy = NSViewLayerContentsRedrawOnSetNeedsDisplay; + + NSError *libraryError = NULL; + NSURL *libraryURL = [[NSBundle mainBundle] URLForResource:@"./shaders" + withExtension:@"metallib"]; + if (libraryURL == NULL) { + fprintf(stderr, "Couldn't find library file\n"); + exit(1); } - exit(1); - } - pushArray(_MTLRenderPipelineState, &pipelines, - [device - newRenderPipelineStateWithDescriptor:ui_rect_pipeline_descriptor + id library = [device newLibraryWithURL:libraryURL + error:&libraryError]; + + if (library == NULL) { + if (libraryError.description != NULL) { + NSLog(@"Error description: %@\n", libraryError.description); + } + + exit(1); + } + + id command_queue = [device newCommandQueue]; + id vertex_func = [library newFunctionWithName:@"vs_main"]; + id fragment_func = [library newFunctionWithName:@"fs_main"]; + id ui_rect_vertex_func = + [library newFunctionWithName:@"ui_rect_vs"]; + id ui_rect_fragment_func = + [library newFunctionWithName:@"ui_rect_fs"]; + + MTLRenderPipelineDescriptor *pipeline_descriptor = + [[MTLRenderPipelineDescriptor alloc] init]; + [pipeline_descriptor setVertexFunction:vertex_func]; + [pipeline_descriptor setFragmentFunction:fragment_func]; + pipeline_descriptor.colorAttachments[0].pixelFormat = + MTLPixelFormatRGBA8Unorm; + pipeline_descriptor.colorAttachments[0].alphaBlendOperation = + MTLBlendOperationAdd; + pipeline_descriptor.colorAttachments[0].sourceAlphaBlendFactor = + MTLBlendFactorSourceAlpha; + pipeline_descriptor.colorAttachments[0].destinationAlphaBlendFactor = + MTLBlendFactorOneMinusSourceAlpha; + pipeline_descriptor.colorAttachments[0].sourceRGBBlendFactor = + MTLBlendFactorSourceAlpha; + pipeline_descriptor.colorAttachments[0].destinationRGBBlendFactor = + MTLBlendFactorOneMinusSourceAlpha; + pipeline_descriptor.colorAttachments[0].blendingEnabled = true; + + MTLRenderPipelineDescriptor *ui_rect_pipeline_descriptor = + [[MTLRenderPipelineDescriptor alloc] init]; + [ui_rect_pipeline_descriptor setVertexFunction:ui_rect_vertex_func]; + [ui_rect_pipeline_descriptor setFragmentFunction:ui_rect_fragment_func]; + ui_rect_pipeline_descriptor.colorAttachments[0].pixelFormat = + MTLPixelFormatRGBA8Unorm; + ui_rect_pipeline_descriptor.colorAttachments[0].alphaBlendOperation = + MTLBlendOperationAdd; + ui_rect_pipeline_descriptor.colorAttachments[0].sourceAlphaBlendFactor = + MTLBlendFactorSourceAlpha; + ui_rect_pipeline_descriptor.colorAttachments[0] + .destinationAlphaBlendFactor = MTLBlendFactorOneMinusSourceAlpha; + ui_rect_pipeline_descriptor.colorAttachments[0].sourceRGBBlendFactor = + MTLBlendFactorSourceAlpha; + ui_rect_pipeline_descriptor.colorAttachments[0].destinationRGBBlendFactor = + MTLBlendFactorOneMinusSourceAlpha; + ui_rect_pipeline_descriptor.colorAttachments[0].blendingEnabled = true; + + NSError *pipeline_error = NULL; + array(_MTLRenderPipelineState) pipelines = + newArray(_MTLRenderPipelineState, 2); + pushArray(_MTLRenderPipelineState, &pipelines, + [device newRenderPipelineStateWithDescriptor:pipeline_descriptor + error:&pipeline_error]); + if (pipeline_error != NULL) { + if (pipeline_error.description != NULL) { + NSLog(@"Error description: %@\n", pipeline_error.description); + } + + exit(1); + } + pushArray( + _MTLRenderPipelineState, &pipelines, + [device newRenderPipelineStateWithDescriptor:ui_rect_pipeline_descriptor error:&pipeline_error]); - array(_MTLBuffer) buffers = newArray(_MTLBuffer, 8); - array(_MTLTexture) textures = newArray(_MTLTexture, 8); + array(_MTLBuffer) buffers = newArray(_MTLBuffer, 8); + array(_MTLTexture) textures = newArray(_MTLTexture, 8); - if (pipeline_error != NULL) { - if (pipeline_error.description != NULL) { - NSLog(@"Error description: %@\n", pipeline_error.description); + if (pipeline_error != NULL) { + if (pipeline_error.description != NULL) { + NSLog(@"Error description: %@\n", pipeline_error.description); + } + + exit(1); } - exit(1); - } + return (_metal_gfx_context){ + .application = application, + .window = window, + .view = view, + .keep_running = true, - return (_metal_gfx_context){ - .application = application, - .window = window, - .view = view, - .keep_running = true, - - .device = device, - .metal_layer = metal_layer, - .library = library, - .command_queue = command_queue, - .pipelines = pipelines, - .buffers = buffers, - .textures = textures, - }; + .device = device, + .metal_layer = metal_layer, + .library = library, + .command_queue = command_queue, + .pipelines = pipelines, + .buffers = buffers, + .textures = textures, + }; } static void _metal_gfx_send_events(_metal_gfx_context *cx) { - NSEvent *event = [cx->application nextEventMatchingMask:NSEventMaskAny - untilDate:[NSDate distantPast] - inMode:NSDefaultRunLoopMode - dequeue:YES]; + NSEvent *event = [cx->application nextEventMatchingMask:NSEventMaskAny + untilDate:[NSDate distantPast] + inMode:NSDefaultRunLoopMode + dequeue:YES]; - [cx->application sendEvent:event]; + [cx->application sendEvent:event]; } static void _metal_gfx_present(_metal_gfx_context *cx) { - _gfx_context.gpu_glyphs.size = 0; - _gfx_context.gpu_ui_rects.size = 0; - _gfx_context.frame_func(); + _gfx_context.gpu_glyphs.size = 0; + _gfx_context.gpu_ui_rects.size = 0; + _gfx_context.frame_func(cx->mouse_x, + _gfx_context.frame_height - cx->mouse_y); - if (_gfx_context.gpu_glyphs.size > 0) { - gfx_update_buffer(&_gfx_context, 2, _gfx_context.gpu_glyphs.data, - _gfx_context.gpu_glyphs.size * sizeof(GpuGlyph)); - } - if (_gfx_context.gpu_ui_rects.size > 0) { - gfx_update_buffer(&_gfx_context, 4, _gfx_context.gpu_ui_rects.data, - _gfx_context.gpu_ui_rects.size * sizeof(GpuUiRect)); - } + if (_gfx_context.gpu_glyphs.size > 0) { + gfx_update_buffer(&_gfx_context, 2, _gfx_context.gpu_glyphs.data, + _gfx_context.gpu_glyphs.size * sizeof(GpuGlyph)); + } + if (_gfx_context.gpu_ui_rects.size > 0) { + gfx_update_buffer(&_gfx_context, 4, _gfx_context.gpu_ui_rects.data, + _gfx_context.gpu_ui_rects.size * sizeof(GpuUiRect)); + } - GpuUniformParams gpu_uniform_params = { - .screen_size = - { - (float)_gfx_context.frame_width, - (float)_gfx_context.frame_height, - }, - }; + GpuUniformParams gpu_uniform_params = { + .screen_size = + { + (float)_gfx_context.frame_width, + (float)_gfx_context.frame_height, + }, + }; - gfx_update_buffer(&_gfx_context, 3, &gpu_uniform_params, - sizeof(GpuUniformParams)); + gfx_update_buffer(&_gfx_context, 3, &gpu_uniform_params, + sizeof(GpuUniformParams)); - id drawable = [cx->metal_layer nextDrawable]; + @autoreleasepool { + id drawable = [cx->metal_layer nextDrawable]; - id command_buffer = [cx->command_queue commandBuffer]; - MTLRenderPassDescriptor *render_pass_desc = - [MTLRenderPassDescriptor renderPassDescriptor]; - render_pass_desc.colorAttachments[0].texture = drawable.texture; - render_pass_desc.colorAttachments[0].loadAction = MTLLoadActionClear; - render_pass_desc.colorAttachments[0].clearColor = - MTLClearColorMake(0, 0, 0, 1); - render_pass_desc.colorAttachments[0].storeAction = MTLStoreActionStore; + id command_buffer = [cx->command_queue commandBuffer]; + MTLRenderPassDescriptor *render_pass_desc = + [MTLRenderPassDescriptor renderPassDescriptor]; + render_pass_desc.colorAttachments[0].texture = drawable.texture; + render_pass_desc.colorAttachments[0].loadAction = MTLLoadActionClear; + render_pass_desc.colorAttachments[0].clearColor = + MTLClearColorMake(0, 0, 0, 1); + render_pass_desc.colorAttachments[0].storeAction = MTLStoreActionStore; - id encoder = - [command_buffer renderCommandEncoderWithDescriptor:render_pass_desc]; + id encoder = [command_buffer + renderCommandEncoderWithDescriptor:render_pass_desc]; - if (_gfx_context.gpu_ui_rects.size > 0) { - // UI Rects - [encoder setRenderPipelineState:cx->pipelines.data[1]]; - // FIXME: allow these to be described by the user instead of hardcoded - [encoder setVertexBuffer:cx->buffers.data[0] - offset:0 - atIndex:0]; // vertices - [encoder setVertexBuffer:cx->buffers.data[4] - offset:0 - atIndex:1]; // ui rects - [encoder setVertexBuffer:cx->buffers.data[3] - offset:0 - atIndex:2]; // uniforms - // TODO: get instance count properly - [encoder drawIndexedPrimitives:MTLPrimitiveTypeTriangle - indexCount:6 - indexType:MTLIndexTypeUInt16 - indexBuffer:cx->buffers.data[1] - indexBufferOffset:0 - instanceCount:_gfx_context.gpu_ui_rects.size]; - } + if (_gfx_context.gpu_ui_rects.size > 0) { + // UI Rects + [encoder setRenderPipelineState:cx->pipelines.data[1]]; + // FIXME: allow these to be described by the user instead of + // hardcoded + [encoder setVertexBuffer:cx->buffers.data[0] + offset:0 + atIndex:0]; // vertices + [encoder setVertexBuffer:cx->buffers.data[4] + offset:0 + atIndex:1]; // ui rects + [encoder setVertexBuffer:cx->buffers.data[3] + offset:0 + atIndex:2]; // uniforms + [encoder drawIndexedPrimitives:MTLPrimitiveTypeTriangle + indexCount:6 + indexType:MTLIndexTypeUInt16 + indexBuffer:cx->buffers.data[1] + indexBufferOffset:0 + instanceCount:_gfx_context.gpu_ui_rects.size]; + } - if (_gfx_context.gpu_glyphs.size > 0) { - // UI Text - [encoder setRenderPipelineState:cx->pipelines.data[0]]; - // FIXME: allow these to be described by the user instead of hardcoded - [encoder setVertexBuffer:cx->buffers.data[0] - offset:0 - atIndex:0]; // vertices - [encoder setVertexBuffer:cx->buffers.data[2] - offset:0 - atIndex:1]; // glyph data - [encoder setVertexBuffer:cx->buffers.data[3] - offset:0 - atIndex:2]; // uniforms - [encoder setFragmentTexture:cx->textures.data[0] atIndex:0]; - // TODO: get instance count properly - [encoder drawIndexedPrimitives:MTLPrimitiveTypeTriangle - indexCount:6 - indexType:MTLIndexTypeUInt16 - indexBuffer:cx->buffers.data[1] - indexBufferOffset:0 - instanceCount:_gfx_context.gpu_glyphs.size]; - } + if (_gfx_context.gpu_glyphs.size > 0) { + // UI Text + [encoder setRenderPipelineState:cx->pipelines.data[0]]; + // FIXME: allow these to be described by the user instead of + // hardcoded + [encoder setVertexBuffer:cx->buffers.data[0] + offset:0 + atIndex:0]; // vertices + [encoder setVertexBuffer:cx->buffers.data[2] + offset:0 + atIndex:1]; // glyph data + [encoder setVertexBuffer:cx->buffers.data[3] + offset:0 + atIndex:2]; // uniforms + [encoder setFragmentTexture:cx->textures.data[0] atIndex:0]; + [encoder drawIndexedPrimitives:MTLPrimitiveTypeTriangle + indexCount:6 + indexType:MTLIndexTypeUInt16 + indexBuffer:cx->buffers.data[1] + indexBufferOffset:0 + instanceCount:_gfx_context.gpu_glyphs.size]; + } - [encoder endEncoding]; - [command_buffer presentDrawable:drawable]; - [command_buffer commit]; + [encoder endEncoding]; + [command_buffer presentDrawable:drawable + afterMinimumDuration:1.0 / 144.0]; + [command_buffer commit]; - [command_buffer waitUntilScheduled]; + [command_buffer waitUntilScheduled]; + } } static size_t _metal_gfx_push_texture_buffer(_metal_gfx_context *cx, uint32_t width, uint32_t height, const void *data, size_t len) { - MTLTextureDescriptor *texture_desc = [MTLTextureDescriptor - texture2DDescriptorWithPixelFormat:MTLPixelFormatR8Unorm - width:width - height:height - mipmapped:false]; - _MTLTexture texture = [cx->device newTextureWithDescriptor:texture_desc]; + MTLTextureDescriptor *texture_desc = [MTLTextureDescriptor + texture2DDescriptorWithPixelFormat:MTLPixelFormatR8Unorm + width:width + height:height + mipmapped:false]; + _MTLTexture texture = [cx->device newTextureWithDescriptor:texture_desc]; - MTLRegion region = MTLRegionMake2D(0, 0, width, height); - [texture replaceRegion:region - mipmapLevel:0 - slice:0 - withBytes:data - bytesPerRow:width * sizeof(uint8_t) - bytesPerImage:len]; + MTLRegion region = MTLRegionMake2D(0, 0, width, height); + [texture replaceRegion:region + mipmapLevel:0 + slice:0 + withBytes:data + bytesPerRow:width * sizeof(uint8_t) + bytesPerImage:len]; - pushArray(_MTLTexture, &cx->textures, texture); - return cx->textures.size - 1; + pushArray(_MTLTexture, &cx->textures, texture); + return cx->textures.size - 1; } static void _metal_gfx_resize_texture_buffer(_metal_gfx_context *cx, uint32_t width, size_t texture_index, uint32_t height) { - [cx->textures.data[texture_index] setPurgeableState:MTLPurgeableStateEmpty]; - [cx->textures.data[texture_index] release]; + [cx->textures.data[texture_index] setPurgeableState:MTLPurgeableStateEmpty]; + [cx->textures.data[texture_index] release]; - MTLTextureDescriptor *texture_desc = [MTLTextureDescriptor - texture2DDescriptorWithPixelFormat:MTLPixelFormatR8Unorm - width:width - height:height - mipmapped:false]; - cx->textures.data[texture_index] = - [cx->device newTextureWithDescriptor:texture_desc]; + MTLTextureDescriptor *texture_desc = [MTLTextureDescriptor + texture2DDescriptorWithPixelFormat:MTLPixelFormatR8Unorm + width:width + height:height + mipmapped:false]; + cx->textures.data[texture_index] = + [cx->device newTextureWithDescriptor:texture_desc]; } size_t _metal_gfx_push_vertex_buffer(_metal_gfx_context *cx, const void *data, size_t len) { - pushArray(_MTLBuffer, &cx->buffers, - [cx->device newBufferWithBytes:data - length:len - options:MTLResourceStorageModeShared]); + pushArray(_MTLBuffer, &cx->buffers, + [cx->device newBufferWithBytes:data + length:len + options:MTLResourceStorageModeShared]); - return cx->buffers.size - 1; + return cx->buffers.size - 1; } static size_t _metal_gfx_allocate_vertex_buffer(_metal_gfx_context *cx, size_t len) { - pushArray(_MTLBuffer, &cx->buffers, - [cx->device newBufferWithLength:len - options:MTLResourceStorageModeShared]); + pushArray(_MTLBuffer, &cx->buffers, + [cx->device newBufferWithLength:len + options:MTLResourceStorageModeShared]); - return cx->buffers.size - 1; + return cx->buffers.size - 1; } static void _metal_gfx_update_buffer(_metal_gfx_context *cx, size_t buffer_index, const void *data, size_t len) { - void *buffer_contents = [cx->buffers.data[buffer_index] contents]; + void *buffer_contents = [cx->buffers.data[buffer_index] contents]; - // FIXME: actually check to see if this will fit in the buffer - memcpy(buffer_contents, data, len); + // FIXME: actually check to see if this will fit in the buffer + memcpy(buffer_contents, data, len); } #endif void gfx_run_events(gfx_context_t *cx) { #if defined(__APPLE__) - return _metal_gfx_send_events(&cx->backend); + return _metal_gfx_send_events(&cx->backend); #else #error "Unsupported graphics backend" #endif @@ -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, uint32_t height, const void *data, size_t len) { #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 #error "Unsupported graphics backend" #endif @@ -504,7 +563,7 @@ size_t gfx_push_texture_buffer(gfx_context_t *cx, uint32_t width, size_t gfx_push_vertex_buffer(gfx_context_t *cx, const void *data, size_t len) { #if defined(__APPLE__) - return _metal_gfx_push_vertex_buffer(&cx->backend, data, len); + return _metal_gfx_push_vertex_buffer(&cx->backend, data, len); #else #error "Unsupported graphics backend" #endif @@ -512,7 +571,7 @@ size_t gfx_push_vertex_buffer(gfx_context_t *cx, const void *data, size_t len) { size_t gfx_allocate_vertex_buffer(gfx_context_t *cx, size_t len) { #if defined(__APPLE__) - return _metal_gfx_allocate_vertex_buffer(&cx->backend, len); + return _metal_gfx_allocate_vertex_buffer(&cx->backend, len); #else #error "Unsupported graphics backend" #endif @@ -522,84 +581,84 @@ size_t gfx_allocate_vertex_buffer(gfx_context_t *cx, size_t len) { void gfx_update_buffer(gfx_context_t *cx, size_t buffer_index, const void *data, size_t len) { #if defined(__APPLE__) - return _metal_gfx_update_buffer(&cx->backend, buffer_index, data, len); + return _metal_gfx_update_buffer(&cx->backend, buffer_index, data, len); #else #error "Unsupported graphics backend" #endif } void gfx_queue_char(gfx_context_t *cx, uint8_t character, float position[2]) { - if (character >= 32) { - GpuGlyph glyph = cx->glyph_cache.data[character - 32]; + if (character >= 32) { + GpuGlyph glyph = cx->glyph_cache.data[character - 32]; - glyph.position[0] = position[0]; - glyph.position[1] = position[1]; + glyph.position[0] = position[0]; + glyph.position[1] = position[1]; - pushArray(GpuGlyph, &cx->gpu_glyphs, glyph); - } + pushArray(GpuGlyph, &cx->gpu_glyphs, glyph); + } } void gfx_queue_text(gfx_context_t *cx, string text, float position[2]) { - float x = 0; - for (size_t i = 0; i < text.len; ++i) { - if (text.data[i] >= 32) { - GpuGlyph glyph = cx->glyph_cache.data[text.data[i] - 32]; + float x = 0; + for (size_t i = 0; i < text.len; ++i) { + if (text.data[i] >= 32) { + GpuGlyph glyph = cx->glyph_cache.data[text.data[i] - 32]; - glyph.position[0] = x + position[0]; - glyph.position[1] = position[1]; - x += glyph.size[0] / 2 + 4; + glyph.position[0] = x + position[0]; + glyph.position[1] = position[1]; + x += glyph.size[0] / 2 + 4; - pushArray(GpuGlyph, &cx->gpu_glyphs, glyph); + pushArray(GpuGlyph, &cx->gpu_glyphs, glyph); + } } - } } void gfx_queue_ui_rect(gfx_context_t *cx, float position[2], float size[2], float border_size, float color[4]) { - GpuUiRect rect = - (GpuUiRect){.position = {position[0], position[1]}, - .size = {size[0], size[1]}, - .border_size = {border_size, border_size}, - .color = {color[0], color[1], color[2], color[3]}}; + GpuUiRect rect = + (GpuUiRect){.position = {position[0], position[1]}, + .size = {size[0], size[1]}, + .border_size = {border_size, border_size}, + .color = {color[0], color[1], color[2], color[3]}}; - pushArray(GpuUiRect, &cx->gpu_ui_rects, rect); + pushArray(GpuUiRect, &cx->gpu_ui_rects, rect); } void *gfx_init_context(_gfx_frame_func frame_func, uint32_t width, uint32_t height) { - __auto_type backend = + __auto_type backend = #if defined(__APPLE__) - _metal_gfx_init_context(width, height); + _metal_gfx_init_context(width, height); #else #error "Unsupported graphics backend" #endif - _gfx_context.backend = backend; - _gfx_context.frame_func = frame_func; - _gfx_context.frame_width = width; - _gfx_context.frame_height = height; + _gfx_context.backend = backend; + _gfx_context.frame_func = frame_func; + _gfx_context.frame_width = width; + _gfx_context.frame_height = height; - _gfx_context.gpu_ui_rects = newArray(GpuUiRect, 2000); - _gfx_context.gpu_glyphs = newArray(GpuGlyph, 8192 * 32); - _gfx_context.glyph_cache = newArray(GpuGlyph, 97); + _gfx_context.gpu_ui_rects = newArray(GpuUiRect, 2000); + _gfx_context.gpu_glyphs = newArray(GpuGlyph, 8192 * 32); + _gfx_context.glyph_cache = newArray(GpuGlyph, 97); - float vertices[] = { - // positions texture coords - -1.0f, 1.0f, 0.0f, 0.0f, 1.0f, 1.0f, 1.0f, 0.0f, - 1.0f, -1.0f, 1.0f, 1.0f, -1.0f, -1.0f, 0.0f, 1.0f, - }; - const uint16_t indices[] = {0, 1, 2, 0, 2, 3}; + float vertices[] = { + // positions texture coords + -1.0f, 1.0f, 0.0f, 0.0f, 1.0f, 1.0f, 1.0f, 0.0f, + 1.0f, -1.0f, 1.0f, 1.0f, -1.0f, -1.0f, 0.0f, 1.0f, + }; + const uint16_t indices[] = {0, 1, 2, 0, 2, 3}; - // NOTE: the order of these matter - gfx_push_vertex_buffer(&_gfx_context, vertices, sizeof(vertices)); - gfx_push_vertex_buffer(&_gfx_context, indices, sizeof(indices)); - gfx_allocate_vertex_buffer(&_gfx_context, _gfx_context.gpu_glyphs.capacity * - sizeof(GpuGlyph)); - gfx_allocate_vertex_buffer(&_gfx_context, sizeof(GpuUniformParams)); - gfx_allocate_vertex_buffer(&_gfx_context, _gfx_context.gpu_ui_rects.capacity * - sizeof(GpuUiRect)); + // NOTE: the order of these matter + gfx_push_vertex_buffer(&_gfx_context, vertices, sizeof(vertices)); + gfx_push_vertex_buffer(&_gfx_context, indices, sizeof(indices)); + gfx_allocate_vertex_buffer(&_gfx_context, _gfx_context.gpu_glyphs.capacity * + sizeof(GpuGlyph)); + gfx_allocate_vertex_buffer(&_gfx_context, sizeof(GpuUniformParams)); + gfx_allocate_vertex_buffer( + &_gfx_context, _gfx_context.gpu_ui_rects.capacity * sizeof(GpuUiRect)); - return &_gfx_context; + return &_gfx_context; } #endif diff --git a/src/main.c b/src/main.c index 9cd3a68..e83c32d 100644 --- a/src/main.c +++ b/src/main.c @@ -25,131 +25,151 @@ // static Arena *context_arena = &default_arena; void *context_alloc(size_t size) { - // assert(context_arena); - // return arena_alloc(context_arena, size); - return malloc(size); + // assert(context_arena); + // return arena_alloc(context_arena, size); + return malloc(size); } static struct { - bool should_exit; + bool should_exit; - ui_context ui_cx; - gfx_context_t *gfx_cx; + ui_context ui_cx; + gfx_context_t *gfx_cx; } state; void ed_init(_gfx_frame_func frame_func) { - state.gfx_cx = gfx_init_context(frame_func, 640, 480); - state.ui_cx = ui_init_context(); + state.gfx_cx = gfx_init_context(frame_func, 640, 480); + state.ui_cx = ui_init_context(); - // TODO: grab default font from the system - uint8_t ttf_buffer[1 << 20]; - assert("failed to load font" && - load_file(_String("./bin/JetBrainsMono-Medium.ttf"), 1 << 20, - ttf_buffer)); + // TODO: grab default font from the system + uint8_t ttf_buffer[1 << 20]; + assert("failed to load font" && + load_file(_String("./bin/JetBrainsMono-Medium.ttf"), 1 << 20, + ttf_buffer)); - stbtt_fontinfo font; - stbtt_InitFont(&font, ttf_buffer, stbtt_GetFontOffsetForIndex(ttf_buffer, 0)); + stbtt_fontinfo font; + stbtt_InitFont(&font, ttf_buffer, + stbtt_GetFontOffsetForIndex(ttf_buffer, 0)); - const int font_bitmap_size = 512; - const int rasterized_font_height = _FONT_HEIGHT; - uint8_t *font_bitmap = - context_alloc(font_bitmap_size * font_bitmap_size * sizeof(uint8_t)); + const int font_bitmap_size = 512; + const int rasterized_font_height = _FONT_HEIGHT; + uint8_t *font_bitmap = + context_alloc(font_bitmap_size * font_bitmap_size * sizeof(uint8_t)); - int ascent, descent, line_gap; - float scale = stbtt_ScaleForPixelHeight(&font, rasterized_font_height * 2); - stbtt_GetFontVMetrics(&font, &ascent, &descent, &line_gap); + int ascent, descent, line_gap; + float scale = stbtt_ScaleForPixelHeight(&font, rasterized_font_height * 2); + stbtt_GetFontVMetrics(&font, &ascent, &descent, &line_gap); - for (size_t xx = 0; xx < rasterized_font_height / 2; ++xx) { - for (size_t yy = 0; yy < rasterized_font_height; ++yy) { - font_bitmap[xx + yy * font_bitmap_size] = 0; + for (size_t xx = 0; xx < rasterized_font_height / 2; ++xx) { + for (size_t yy = 0; yy < rasterized_font_height; ++yy) { + font_bitmap[xx + yy * font_bitmap_size] = 0; + } } - } - // manually add glyph for SPACE - pushArray(GpuGlyph, &state.gfx_cx->glyph_cache, - ((GpuGlyph){ - .atlas_position = {0}, - .size = {rasterized_font_height / 4, 1}, - .position = {0}, - .y_offset = -rasterized_font_height, - })); - - int x = rasterized_font_height / 4; - int y = 0; - for (size_t i = 33; i < 33 + 96; ++i) { - int width, height, xoff, yoff; - uint8_t *bitmap = stbtt_GetCodepointBitmap(&font, scale, scale, (int)i, - &width, &height, &xoff, &yoff); - - if (x + width >= font_bitmap_size) { - x = 0; - y += (int)((float)(ascent - descent + line_gap) * scale); - } - - size_t xxx = x; - for (size_t xx = 0; xx < (size_t)width; ++xx) { - size_t yyy = y; - for (size_t yy = 0; yy < (size_t)height; ++yy) { - font_bitmap[xxx + yyy * font_bitmap_size] = bitmap[xx + yy * width]; - - yyy += 1; - } - - xxx += 1; - } - + // manually add glyph for SPACE pushArray(GpuGlyph, &state.gfx_cx->glyph_cache, ((GpuGlyph){ - .atlas_position = {(float)(x), (float)(y)}, - .size = {(float)width, (float)height}, - .position = {(float)x, (float)y}, - .y_offset = (float)(yoff), + .atlas_position = {0}, + .size = {rasterized_font_height / 4, 1}, + .position = {0}, + .y_offset = -rasterized_font_height, })); - x += width; - } + int x = rasterized_font_height / 4; + int y = 0; + for (size_t i = 33; i < 33 + 96; ++i) { + int width, height, xoff, yoff; + uint8_t *bitmap = stbtt_GetCodepointBitmap( + &font, scale, scale, (int)i, &width, &height, &xoff, &yoff); - gfx_push_texture_buffer( - state.gfx_cx, font_bitmap_size, font_bitmap_size, font_bitmap, - font_bitmap_size * font_bitmap_size * sizeof(uint8_t)); + if (x + width >= font_bitmap_size) { + x = 0; + y += (int)((float)(ascent - descent + line_gap) * scale); + } + + size_t xxx = x; + for (size_t xx = 0; xx < (size_t)width; ++xx) { + size_t yyy = y; + for (size_t yy = 0; yy < (size_t)height; ++yy) { + font_bitmap[xxx + yyy * font_bitmap_size] = + bitmap[xx + yy * width]; + + yyy += 1; + } + + xxx += 1; + } + + pushArray(GpuGlyph, &state.gfx_cx->glyph_cache, + ((GpuGlyph){ + .atlas_position = {(float)(x), (float)(y)}, + .size = {(float)width, (float)height}, + .position = {(float)x, (float)y}, + .y_offset = (float)(yoff), + })); + + x += width; + } + + gfx_push_texture_buffer( + state.gfx_cx, font_bitmap_size, font_bitmap_size, font_bitmap, + font_bitmap_size * font_bitmap_size * sizeof(uint8_t)); } void render_ui_text(string text, float position[2]) { - gfx_queue_text(state.gfx_cx, text, position); + gfx_queue_text(state.gfx_cx, text, position); } 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() { - state.ui_cx.frame_elements.data[0].size.computed_size[0] = - state.gfx_cx->frame_width; - state.ui_cx.frame_elements.data[0].size.computed_size[1] = - state.gfx_cx->frame_height; +void ed_frame(int mouse_x, int mouse_y) { + state.ui_cx.frame_elements.data[0].size.computed_size[0] = + state.gfx_cx->frame_width; + state.ui_cx.frame_elements.data[0].size.computed_size[1] = + state.gfx_cx->frame_height; - ui_element(&state.ui_cx, _String("channel sidebar"), UI_AXIS_VERTICAL, - ui_make_size(ui_children_sum, ui_fill), UI_FLAG_DRAW_BACKGROUND); - ui_push_parent(&state.ui_cx); - { - ui_element(&state.ui_cx, _String("#dev-general"), 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-help"), UI_AXIS_HORIZONTAL, - ui_make_size(ui_fit_text, ui_fit_text), - UI_FLAG_DRAW_BACKGROUND | UI_FLAG_DRAW_TEXT); - } - ui_pop_parent(&state.ui_cx); + 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"); - ui_compute_layout(&state.ui_cx, 0); - ui_render(&state.ui_cx, render_ui_text, render_ui_rect); + render_ui_rect((float[2]){mouse_x, mouse_y}, (float[2]){32, 32}, + (float[4]){1, 1, 1, 1}); - ui_update_cache(&state.ui_cx, 0); - ui_prune(&state.ui_cx); + ui_element(&state.ui_cx, _String("channel sidebar"), UI_AXIS_VERTICAL, + ui_make_size(ui_children_sum, ui_fill), UI_FLAG_DRAW_BACKGROUND); + 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_make_size(ui_fit_text, ui_fit_text), + UI_FLAG_DRAW_BACKGROUND | UI_FLAG_DRAW_TEXT); + ui_element(&state.ui_cx, _String("#dev-help"), UI_AXIS_HORIZONTAL, + ui_make_size(ui_fit_text, ui_fit_text), + UI_FLAG_DRAW_BACKGROUND | UI_FLAG_DRAW_TEXT); + } + ui_pop_parent(&state.ui_cx); + + ui_compute_layout(&state.ui_cx, 0); + ui_render(&state.ui_cx, render_ui_text, render_ui_rect); + + ui_update_cache(&state.ui_cx, 0); + 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[]) { - ed_init(ed_frame); + ed_init(ed_frame); - while (keep_running) { - gfx_run_events(state.gfx_cx); - } + while (keep_running) { + gfx_run_events(state.gfx_cx); + } } diff --git a/src/string.h b/src/string.h index 26b6d64..022c8a8 100644 --- a/src/string.h +++ b/src/string.h @@ -1,12 +1,13 @@ #ifndef ED_STRING_INCLUDED #define ED_STRING_INCLUDED -#include +#include #include #include -#include +#include -#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 { uint8_t *data; size_t len; @@ -14,13 +15,14 @@ typedef struct { bool owned; } string; - #ifdef ED_STRING_IMPLEMENTATION 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; iframe_elements.size > 0) { - cx->current_parent = cx->frame_elements.size - 1; - } + if (cx->frame_elements.size > 0) { + cx->current_parent = cx->frame_elements.size - 1; + } } void ui_pop_parent(ui_context *cx) { - if (_parent(cx->current_parent) < SIZE_MAX) { - cx->current_parent = _parent(cx->current_parent); - } + if (_parent(cx->current_parent) < SIZE_MAX) { + cx->current_parent = _parent(cx->current_parent); + } } size_t ui_element(ui_context *cx, string label, ui_axis axis, ui_semantic_size size[2], ui_flags flags) { - ui_element_frame_data frame_data = (ui_element_frame_data){ - .index = cx->frame_elements.size, - // TODO: don't just set this to label, because then elements - // with the same label can't be created together - .key = label, - .label = label, - .first = -1, - .last = -1, - .next = -1, - .prev = cx->frame_elements.data[cx->current_parent].last, - .parent = cx->current_parent, - .size.axis = axis, - .size.semantic_size[0] = size[0], - .size.semantic_size[1] = size[1], - .flags = flags, - }; + ui_element_frame_data frame_data = (ui_element_frame_data){ + .index = cx->frame_elements.size, + // TODO: don't just set this to label, because then elements + // with the same label can't be created together + .key = label, + .label = label, + .first = -1, + .last = -1, + .next = -1, + .prev = cx->frame_elements.data[cx->current_parent].last, + .parent = cx->current_parent, + .size.axis = axis, + .size.semantic_size[0] = size[0], + .size.semantic_size[1] = size[1], + .flags = flags, + }; - // Get cached element data - ui_element_cache_data *cache_data = ht_get(&cx->cached_elements, label); - if (cache_data) { - cache_data->last_instantiated_index = cx->frame_index; + // Get cached element data + ui_element_cache_data *cache_data = ht_get(&cx->cached_elements, label); + if (cache_data) { + cache_data->last_instantiated_index = cx->frame_index; - frame_data.size.computed_pos[0] = cache_data->size.computed_pos[0]; - frame_data.size.computed_pos[1] = cache_data->size.computed_pos[1]; + frame_data.size.computed_pos[0] = cache_data->size.computed_pos[0]; + frame_data.size.computed_pos[1] = cache_data->size.computed_pos[1]; - frame_data.size.computed_size[0] = cache_data->size.computed_size[0]; - frame_data.size.computed_size[1] = cache_data->size.computed_size[1]; - } else { - bool did_insert = ht_set(&cx->cached_elements, label, - &(ui_element_cache_data){ - .label = label, - .size = {0}, - .last_instantiated_index = cx->frame_index, - }); - assert("couldn't insert into ui element cache" && did_insert); - } + frame_data.size.computed_size[0] = cache_data->size.computed_size[0]; + frame_data.size.computed_size[1] = cache_data->size.computed_size[1]; + } else { + bool did_insert = ht_set(&cx->cached_elements, label, + &(ui_element_cache_data){ + .label = label, + .size = {0}, + .last_instantiated_index = cx->frame_index, + }); + assert("couldn't insert into ui element cache" && did_insert); + } - pushArray(ui_element_frame_data, &cx->frame_elements, frame_data); + pushArray(ui_element_frame_data, &cx->frame_elements, frame_data); - if (frame_data.prev < SIZE_MAX) { - _prev_ref(frame_data.index)->next = frame_data.index; - } - if (_elm(cx->current_parent)->first == SIZE_MAX) { - _elm(cx->current_parent)->first = frame_data.index; - } - _elm(cx->current_parent)->last = frame_data.index; + if (frame_data.prev < SIZE_MAX) { + _prev_ref(frame_data.index)->next = frame_data.index; + } + if (_elm(cx->current_parent)->first == SIZE_MAX) { + _elm(cx->current_parent)->first = frame_data.index; + } + _elm(cx->current_parent)->last = frame_data.index; - return frame_data.index; + return frame_data.index; } static uint32_t _ui_ancestor_size(ui_context *cx, size_t element_index, ui_axis axis) { - if (element_index == SIZE_MAX || _parent(element_index) == SIZE_MAX) { - return cx->frame_elements.data[0].size.computed_size[axis]; - } + if (element_index == SIZE_MAX || _parent(element_index) == SIZE_MAX) { + return cx->frame_elements.data[0].size.computed_size[axis]; + } - switch (_parent_ref(element_index)->size.semantic_size[axis].type) { - case UI_SEMANTIC_SIZE_FIT_TEXT: - case UI_SEMANTIC_SIZE_FILL: - case UI_SEMANTIC_SIZE_EXACT: - case UI_SEMANTIC_SIZE_PERCENT_OF_PARENT: - return _parent_ref(element_index)->size.computed_size[axis]; - break; + switch (_parent_ref(element_index)->size.semantic_size[axis].type) { + case UI_SEMANTIC_SIZE_FIT_TEXT: + case UI_SEMANTIC_SIZE_FILL: + case UI_SEMANTIC_SIZE_EXACT: + case UI_SEMANTIC_SIZE_PERCENT_OF_PARENT: + return _parent_ref(element_index)->size.computed_size[axis]; + break; - case UI_SEMANTIC_SIZE_CHILDREN_SUM: - return _ui_ancestor_size(cx, _parent(element_index), axis); - break; - } + case UI_SEMANTIC_SIZE_CHILDREN_SUM: + return _ui_ancestor_size(cx, _parent(element_index), axis); + break; + } } static void _ui_compute_simple_layout(ui_context *cx, ui_element_frame_data *elm, ui_axis axis, bool *post_compute) { - switch (elm->size.semantic_size[axis].type) { - case UI_SEMANTIC_SIZE_FIT_TEXT: - if (axis == UI_AXIS_HORIZONTAL) { - elm->size.computed_size[axis] = elm->label.len * _FONT_WIDTH; - } else if (axis == UI_AXIS_VERTICAL) { - elm->size.computed_size[axis] = _FONT_HEIGHT; + switch (elm->size.semantic_size[axis].type) { + case UI_SEMANTIC_SIZE_FIT_TEXT: + if (axis == UI_AXIS_HORIZONTAL) { + elm->size.computed_size[axis] = elm->label.len * _FONT_WIDTH; + } else if (axis == UI_AXIS_VERTICAL) { + elm->size.computed_size[axis] = _FONT_HEIGHT; + } + break; + + case UI_SEMANTIC_SIZE_CHILDREN_SUM: + post_compute[axis] = true; + break; + + case UI_SEMANTIC_SIZE_FILL: + // TODO: set to ancestor size for floating + break; + + case UI_SEMANTIC_SIZE_EXACT: + elm->size.computed_size[axis] = elm->size.semantic_size[axis].integer; + break; + + case UI_SEMANTIC_SIZE_PERCENT_OF_PARENT: { + float semantic_value = (float)elm->size.semantic_size[axis].integer; + elm->size.computed_size[axis] = + (uint32_t)((float)(_ui_ancestor_size(cx, elm->index, axis)) * + (semantic_value / 100.0)); + } break; } - break; - - case UI_SEMANTIC_SIZE_CHILDREN_SUM: - post_compute[axis] = true; - break; - - case UI_SEMANTIC_SIZE_FILL: - // TODO: set to ancestor size for floating - break; - - case UI_SEMANTIC_SIZE_EXACT: - elm->size.computed_size[axis] = elm->size.semantic_size[axis].integer; - break; - - case UI_SEMANTIC_SIZE_PERCENT_OF_PARENT: { - float semantic_value = (float)elm->size.semantic_size[axis].integer; - elm->size.computed_size[axis] = - (uint32_t)((float)(_ui_ancestor_size(cx, elm->index, axis)) * - (semantic_value / 100.0)); - } break; - } } static void _ui_compute_children_layout(ui_context *cx, ui_element_frame_data *elm) { - 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 - // because it will never get incremented in the loop below and cause a divide - // by zero and the number of fills for the axis of the box needs to start at - // zero or else it will be n+1 causing incorrect sizes - uint32_t num_fills[2] = {1, 1}; - num_fills[elm->size.axis] = 0; + // 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 by zero and the number of fills for the axis of the box needs to + // start at zero or else it will be n+1 causing incorrect sizes + uint32_t num_fills[2] = {1, 1}; + num_fills[elm->size.axis] = 0; - // TODO: maybe just use the actual data instead of copying? - uint32_t elm_size[2] = {elm->size.computed_size[0], - elm->size.computed_size[1]}; + // TODO: maybe just use the actual data instead of copying? + uint32_t elm_size[2] = {elm->size.computed_size[0], + elm->size.computed_size[1]}; - size_t child_index = elm->first; - if (child_index < SIZE_MAX) { - do { - ui_compute_layout(cx, child_index); + size_t child_index = elm->first; + if (child_index < SIZE_MAX) { + do { + ui_compute_layout(cx, child_index); - if (_elm(child_index)->size.semantic_size[elm->size.axis].type == - UI_SEMANTIC_SIZE_FILL) { - num_fills[elm->size.axis] += 1; - } else { - child_size[elm->size.axis] += - _elm(child_index)->size.computed_size[elm->size.axis]; - } - } while ((child_index = _next(child_index)) < SIZE_MAX); - } + if (_elm(child_index)->size.semantic_size[elm->size.axis].type == + UI_SEMANTIC_SIZE_FILL) { + num_fills[elm->size.axis] += 1; + } else { + child_size[elm->size.axis] += + _elm(child_index)->size.computed_size[elm->size.axis]; + } + } while ((child_index = _next(child_index)) < SIZE_MAX); + } - child_index = elm->first; - if (child_index < SIZE_MAX) { - do { - for (size_t axis = 0; axis < 2; ++axis) { - if (_elm(child_index)->size.semantic_size[axis].type == - UI_SEMANTIC_SIZE_FILL) { - _elm(child_index)->size.computed_size[axis] = - (elm_size[axis] - child_size[axis]) / num_fills[axis]; - } - } + child_index = elm->first; + if (child_index < SIZE_MAX) { + do { + for (size_t axis = 0; axis < 2; ++axis) { + if (_elm(child_index)->size.semantic_size[axis].type == + UI_SEMANTIC_SIZE_FILL) { + _elm(child_index)->size.computed_size[axis] = + (elm_size[axis] - child_size[axis]) / num_fills[axis]; + } + } - ui_compute_layout(cx, child_index); - } while ((child_index = _next(child_index)) < SIZE_MAX); - } + ui_compute_layout(cx, child_index); + } while ((child_index = _next(child_index)) < SIZE_MAX); + } } void ui_compute_layout(ui_context *cx, size_t element_index) { - if (element_index == SIZE_MAX) - return; + if (element_index == SIZE_MAX) + return; - ui_axis axis = UI_AXIS_HORIZONTAL; - __auto_type elm = _elm(element_index); + ui_axis axis = UI_AXIS_HORIZONTAL; + __auto_type elm = _elm(element_index); - if (_parent(element_index) < SIZE_MAX && - !_flags(element_index, UI_FLAG_FLOATING)) { - __auto_type parent = _parent_ref(element_index); - axis = parent->size.axis; - elm->size.computed_pos[0] = parent->size.computed_pos[0]; - elm->size.computed_pos[1] = parent->size.computed_pos[1]; + if (_parent(element_index) < SIZE_MAX && + !_flags(element_index, UI_FLAG_FLOATING)) { + __auto_type parent = _parent_ref(element_index); + axis = parent->size.axis; + elm->size.computed_pos[0] = parent->size.computed_pos[0]; + elm->size.computed_pos[1] = parent->size.computed_pos[1]; - // TODO: implement scrolling - // elm->size.computed_pos[axis] += parent.scroll_offset; - } - - if (!_flags(element_index, UI_FLAG_FLOATING) && - _prev(element_index) < SIZE_MAX) { - __auto_type prev = _prev_ref(element_index); - - if (prev >= 0) { - elm->size.computed_pos[axis] = - prev->size.computed_pos[axis] + prev->size.computed_size[axis]; + // TODO: implement scrolling + // elm->size.computed_pos[axis] += parent.scroll_offset; } - } - bool post_compute[2] = {false, false}; - // only compute layout for children of root - if (elm->index > 0) { - for (int i = 0; i < 2; ++i) { - _ui_compute_simple_layout(cx, elm, i, post_compute); - } - } - _ui_compute_children_layout(cx, elm); + if (!_flags(element_index, UI_FLAG_FLOATING) && + _prev(element_index) < SIZE_MAX) { + __auto_type prev = _prev_ref(element_index); - // NOTE(pcleavelin): the only difference between these two blocks is the - // ordering of the switch block they can probably be merged - if (post_compute[UI_AXIS_HORIZONTAL]) { - elm->size.computed_size[UI_AXIS_HORIZONTAL] = 0; - - size_t child_index = elm->first; - if (child_index < SIZE_MAX) { - do { - __auto_type child = _elm(child_index); - - switch (elm->size.axis) { - case UI_AXIS_HORIZONTAL: - elm->size.computed_size[UI_AXIS_HORIZONTAL] += - child->size.computed_size[UI_AXIS_HORIZONTAL]; - break; - - case UI_AXIS_VERTICAL: - if (child->size.computed_size[UI_AXIS_HORIZONTAL] > - elm->size.computed_size[UI_AXIS_HORIZONTAL]) { - elm->size.computed_size[UI_AXIS_HORIZONTAL] = - child->size.computed_size[UI_AXIS_HORIZONTAL]; - } - break; + if (prev >= 0) { + elm->size.computed_pos[axis] = + prev->size.computed_pos[axis] + prev->size.computed_size[axis]; } - } while ((child_index = _next(child_index)) < SIZE_MAX); } - } - if (post_compute[UI_AXIS_VERTICAL]) { - elm->size.computed_size[UI_AXIS_VERTICAL] = 0; - size_t child_index = elm->first; - if (child_index < SIZE_MAX) { - do { - __auto_type child = _elm(child_index); - - switch (elm->size.axis) { - case UI_AXIS_HORIZONTAL: - if (child->size.computed_size[UI_AXIS_VERTICAL] > - elm->size.computed_size[UI_AXIS_VERTICAL]) { - elm->size.computed_size[UI_AXIS_VERTICAL] = - child->size.computed_size[UI_AXIS_VERTICAL]; - } - break; - case UI_AXIS_VERTICAL: - elm->size.computed_size[UI_AXIS_VERTICAL] += - child->size.computed_size[UI_AXIS_VERTICAL]; - break; + bool post_compute[2] = {false, false}; + // only compute layout for children of root + if (elm->index > 0) { + for (int i = 0; i < 2; ++i) { + _ui_compute_simple_layout(cx, elm, i, post_compute); + } + } + _ui_compute_children_layout(cx, elm); + + // NOTE(pcleavelin): the only difference between these two blocks is the + // ordering of the switch block they can probably be merged + if (post_compute[UI_AXIS_HORIZONTAL]) { + elm->size.computed_size[UI_AXIS_HORIZONTAL] = 0; + + size_t child_index = elm->first; + if (child_index < SIZE_MAX) { + do { + __auto_type child = _elm(child_index); + + switch (elm->size.axis) { + case UI_AXIS_HORIZONTAL: + elm->size.computed_size[UI_AXIS_HORIZONTAL] += + child->size.computed_size[UI_AXIS_HORIZONTAL]; + break; + + case UI_AXIS_VERTICAL: + if (child->size.computed_size[UI_AXIS_HORIZONTAL] > + elm->size.computed_size[UI_AXIS_HORIZONTAL]) { + elm->size.computed_size[UI_AXIS_HORIZONTAL] = + child->size.computed_size[UI_AXIS_HORIZONTAL]; + } + break; + } + } while ((child_index = _next(child_index)) < SIZE_MAX); + } + } + if (post_compute[UI_AXIS_VERTICAL]) { + elm->size.computed_size[UI_AXIS_VERTICAL] = 0; + + size_t child_index = elm->first; + if (child_index < SIZE_MAX) { + do { + __auto_type child = _elm(child_index); + + switch (elm->size.axis) { + case UI_AXIS_HORIZONTAL: + if (child->size.computed_size[UI_AXIS_VERTICAL] > + elm->size.computed_size[UI_AXIS_VERTICAL]) { + elm->size.computed_size[UI_AXIS_VERTICAL] = + child->size.computed_size[UI_AXIS_VERTICAL]; + } + break; + case UI_AXIS_VERTICAL: + elm->size.computed_size[UI_AXIS_VERTICAL] += + child->size.computed_size[UI_AXIS_VERTICAL]; + break; + } + } while ((child_index = _next(child_index)) < SIZE_MAX); } - } while ((child_index = _next(child_index)) < SIZE_MAX); } - } } void ui_update_cache(ui_context *cx, size_t element_index) { - if (element_index == SIZE_MAX) - return; + if (element_index == SIZE_MAX) + return; - size_t child_index = _elm(element_index)->first; - if (child_index < SIZE_MAX) { - do { - __auto_type child = _elm(child_index); + size_t child_index = _elm(element_index)->first; + if (child_index < SIZE_MAX) { + do { + __auto_type child = _elm(child_index); - size_t last_instantiated_index = cx->frame_index; + size_t last_instantiated_index = cx->frame_index; - ui_element_cache_data *cache; - if ((cache = ht_get(&cx->cached_elements, child->key))) { - last_instantiated_index = cache->last_instantiated_index; - } + ui_element_cache_data *cache; + if ((cache = ht_get(&cx->cached_elements, child->key))) { + last_instantiated_index = cache->last_instantiated_index; + } - ht_set(&cx->cached_elements, child->key, - &(ui_element_cache_data){ - .label = child->label, - .size = - { - .axis = child->size.axis, - .semantic_size = {child->size.semantic_size[0], - child->size.semantic_size[1]}, - .computed_size = {child->size.computed_size[0], - child->size.computed_size[1]}, - .computed_pos = {child->size.computed_pos[0], - child->size.computed_pos[1]}, - }, - .last_instantiated_index = last_instantiated_index, - }); + ht_set(&cx->cached_elements, child->key, + &(ui_element_cache_data){ + .label = child->label, + .size = + { + .axis = child->size.axis, + .semantic_size = {child->size.semantic_size[0], + child->size.semantic_size[1]}, + .computed_size = {child->size.computed_size[0], + child->size.computed_size[1]}, + .computed_pos = {child->size.computed_pos[0], + child->size.computed_pos[1]}, + }, + .last_instantiated_index = last_instantiated_index, + }); - ui_update_cache(cx, child_index); - } while ((child_index = _next(child_index)) < SIZE_MAX); - } + ui_update_cache(cx, child_index); + } while ((child_index = _next(child_index)) < SIZE_MAX); + } } typedef void (*_ui_render_text_func)(string text, float position[2]); @@ -449,58 +466,65 @@ typedef void (*_ui_render_rect_func)(float position[2], float size[2], float color[4]); void ui_render(ui_context *cx, _ui_render_text_func text_func, _ui_render_rect_func rect_func) { - for (size_t i = 1; i < cx->frame_elements.size; ++i) { - string text = cx->frame_elements.data[i].key; - ui_element_frame_data *elm = &cx->frame_elements.data[i]; + for (size_t i = 1; i < cx->frame_elements.size; ++i) { + string text = cx->frame_elements.data[i].key; + ui_element_frame_data *elm = &cx->frame_elements.data[i]; - if (_flags(i, UI_FLAG_DRAW_TEXT)) { - text_func(text, (float[]){(float)elm->size.computed_pos[0], - (float)elm->size.computed_pos[1]}); - } + if (_flags(i, UI_FLAG_DRAW_TEXT)) { + text_func(text, (float[]){(float)elm->size.computed_pos[0], + (float)elm->size.computed_pos[1]}); + } - if (_flags(i, UI_FLAG_DRAW_BACKGROUND)) { - rect_func((float[]){(float)elm->size.computed_pos[0], - (float)elm->size.computed_pos[1]}, - (float[]){(float)elm->size.computed_size[0], - (float)elm->size.computed_size[1]}, - (float[]){1, 1, 1, 0.2}); + if (_flags(i, UI_FLAG_DRAW_BACKGROUND)) { + rect_func((float[]){(float)elm->size.computed_pos[0], + (float)elm->size.computed_pos[1]}, + (float[]){(float)elm->size.computed_size[0], + (float)elm->size.computed_size[1]}, + (float[]){1, 1, 1, 0.2}); + } } - } } void ui_prune(ui_context *cx) { - for (size_t i = 0; i < cx->cached_elements.capacity; ++i) { - if (cx->cached_elements.key_slots[i].key.data != NULL) { - string key = cx->cached_elements.key_slots[i].key; + for (size_t i = 0; i < cx->cached_elements.capacity; ++i) { + if (cx->cached_elements.key_slots[i].key.data != NULL) { + string key = cx->cached_elements.key_slots[i].key; - // 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); - if (cached && cached->last_instantiated_index < cx->frame_index - 5) { - // 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); + // 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); + if (cached && + cached->last_instantiated_index < cx->frame_index - 5) { + // 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); - ht_remove(&cx->cached_elements, key); - } + ht_remove(&cx->cached_elements, key); + } + } } - } - size_t child_index = _elm(0)->first; - do { - __auto_type elm = _elm(child_index); - if (elm->label.owned) { - free(elm->label.data); - } - } while ((child_index = _next(child_index)) < SIZE_MAX); + size_t child_index = _elm(0)->first; + do { + __auto_type elm = _elm(child_index); + if (elm->label.owned) { + free(elm->label.data); + } + } while ((child_index = _next(child_index)) < SIZE_MAX); - cx->frame_index += 1; - cx->frame_elements.size = 1; - cx->frame_elements.data[0].first = SIZE_MAX; - cx->frame_elements.data[0].prev = SIZE_MAX; - cx->frame_elements.data[0].next = SIZE_MAX; - cx->frame_elements.data[0].last = SIZE_MAX; - cx->frame_elements.data[0].parent = SIZE_MAX; - cx->current_parent = 0; + cx->frame_index += 1; + cx->frame_elements.size = 1; + cx->frame_elements.data[0].first = SIZE_MAX; + cx->frame_elements.data[0].prev = SIZE_MAX; + cx->frame_elements.data[0].next = SIZE_MAX; + cx->frame_elements.data[0].last = SIZE_MAX; + cx->frame_elements.data[0].parent = SIZE_MAX; + 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