2020-01-02 02:41:43 +00:00
|
|
|
//
|
|
|
|
// File: interface.h
|
|
|
|
// Author: Peter Slattery
|
|
|
|
// Creation Date: 2020-01-01
|
|
|
|
//
|
|
|
|
#ifndef INTERFACE_H
|
|
|
|
|
2020-07-18 19:00:14 +00:00
|
|
|
enum gs_string_alignment
|
2019-07-19 20:56:21 +00:00
|
|
|
{
|
2019-11-02 21:09:57 +00:00
|
|
|
Align_Left,
|
|
|
|
Align_Center,
|
|
|
|
Align_Right,
|
|
|
|
};
|
|
|
|
|
|
|
|
internal void
|
|
|
|
DrawCharacter_ (render_quad_batch_constructor* BatchConstructor, r32 MinX, r32 MinY, codepoint_bitmap CodepointInfo, v4 Color)
|
|
|
|
{
|
2019-12-27 00:23:43 +00:00
|
|
|
s32 AlignedMinX = (s32)(MinX);
|
|
|
|
s32 AlignedMinY = (s32)(MinY);
|
2019-12-29 00:01:34 +00:00
|
|
|
s32 AlignedMaxX = AlignedMinX + (CodepointInfo.Width);
|
2019-12-27 00:23:43 +00:00
|
|
|
s32 AlignedMaxY = AlignedMinY + (CodepointInfo.Height);
|
2019-08-18 12:56:18 +00:00
|
|
|
|
2020-03-22 04:13:35 +00:00
|
|
|
PushQuad2DOnBatch(BatchConstructor,
|
|
|
|
v2{(r32)AlignedMinX, (r32)AlignedMinY}, v2{(r32)AlignedMaxX, (r32)AlignedMinY},
|
|
|
|
v2{(r32)AlignedMaxX, (r32)AlignedMaxY}, v2{(r32)AlignedMinX, (r32)AlignedMaxY},
|
|
|
|
CodepointInfo.UVMin, CodepointInfo.UVMax,
|
2019-08-18 12:56:18 +00:00
|
|
|
Color);
|
2019-11-02 21:09:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
internal v2
|
|
|
|
DrawCharacterLeftAligned (render_quad_batch_constructor* BatchConstructor, char C, bitmap_font Font, v2 Position, v4 Color)
|
|
|
|
{
|
|
|
|
s32 GlyphDataIndex = GetIndexForCodepoint(Font, C);
|
|
|
|
codepoint_bitmap CodepointInfo = Font.CodepointValues[GlyphDataIndex];
|
2019-08-18 12:56:18 +00:00
|
|
|
|
2020-03-22 04:13:35 +00:00
|
|
|
// NOTE(Peter):
|
2019-11-02 21:09:57 +00:00
|
|
|
r32 MinX = Position.x + CodepointInfo.XOffset;
|
|
|
|
r32 MinY = Position.y + CodepointInfo.YOffset;
|
|
|
|
DrawCharacter_(BatchConstructor, MinX, MinY, CodepointInfo, Color);
|
|
|
|
|
2020-03-22 04:13:35 +00:00
|
|
|
// NOTE(Peter):
|
2019-11-02 21:09:57 +00:00
|
|
|
v2 PointAfterCharacter = v2{Position.x + CodepointInfo.Width, Position.y};
|
|
|
|
return PointAfterCharacter;
|
|
|
|
}
|
|
|
|
|
|
|
|
internal v2
|
|
|
|
DrawCharacterRightAligned (render_quad_batch_constructor* BatchConstructor, char C, bitmap_font Font, v2 Position, v4 Color)
|
|
|
|
{
|
|
|
|
s32 GlyphDataIndex = GetIndexForCodepoint(Font, C);
|
|
|
|
codepoint_bitmap CodepointInfo = Font.CodepointValues[GlyphDataIndex];
|
|
|
|
|
2020-03-22 04:13:35 +00:00
|
|
|
// NOTE(Peter):
|
2019-11-02 21:09:57 +00:00
|
|
|
r32 MinX = Position.x - (CodepointInfo.XOffset + CodepointInfo.Width);
|
|
|
|
r32 MinY = Position.y + CodepointInfo.YOffset + CodepointInfo.YOffset;
|
|
|
|
DrawCharacter_(BatchConstructor, MinX, MinY, CodepointInfo, Color);
|
|
|
|
|
2020-03-22 04:13:35 +00:00
|
|
|
// NOTE(Peter):
|
2019-11-02 21:09:57 +00:00
|
|
|
v2 PointAfterCharacter = v2{Position.x-(CodepointInfo.Width + CodepointInfo.XOffset), Position.y};
|
|
|
|
return PointAfterCharacter;
|
|
|
|
}
|
|
|
|
|
|
|
|
internal v2
|
2020-07-18 19:00:14 +00:00
|
|
|
DrawStringLeftAligned (render_quad_batch_constructor* BatchConstructor, s32 Length, char* gs_string, v2 InitialRegisterPosition, bitmap_font* Font, v4 Color)
|
2019-11-02 21:09:57 +00:00
|
|
|
{
|
|
|
|
v2 RegisterPosition = InitialRegisterPosition;
|
2020-07-18 19:00:14 +00:00
|
|
|
char* C = gs_string;
|
2019-11-02 21:09:57 +00:00
|
|
|
for (s32 i = 0; i < Length; i++)
|
|
|
|
{
|
|
|
|
v2 PositionAfterCharacter = DrawCharacterLeftAligned(BatchConstructor, *C, *Font, RegisterPosition, Color);
|
|
|
|
RegisterPosition.x = PositionAfterCharacter.x;
|
|
|
|
C++;
|
|
|
|
}
|
|
|
|
return RegisterPosition;
|
|
|
|
}
|
|
|
|
|
|
|
|
internal v2
|
2020-07-18 19:00:14 +00:00
|
|
|
DrawStringRightAligned (render_quad_batch_constructor* BatchConstructor, s32 Length, char* gs_string, v2 InitialRegisterPosition, bitmap_font* Font, v4 Color)
|
2019-11-02 21:09:57 +00:00
|
|
|
{
|
|
|
|
v2 RegisterPosition = InitialRegisterPosition;
|
2020-07-18 19:00:14 +00:00
|
|
|
char* C = gs_string + Length - 1;
|
2019-11-02 21:09:57 +00:00
|
|
|
for (s32 i = Length - 1; i >= 0; i--)
|
|
|
|
{
|
|
|
|
v2 PositionAfterCharacter = DrawCharacterRightAligned(BatchConstructor, *C, *Font, RegisterPosition, Color);
|
|
|
|
RegisterPosition.x = PositionAfterCharacter.x;
|
|
|
|
C--;
|
|
|
|
}
|
|
|
|
return RegisterPosition;
|
2019-07-19 20:56:21 +00:00
|
|
|
}
|
|
|
|
|
2019-08-18 12:56:18 +00:00
|
|
|
internal v2
|
2020-07-18 19:00:14 +00:00
|
|
|
DrawString(render_command_buffer* RenderBuffer, gs_string String, bitmap_font* Font, v2 Position, v4 Color, gs_string_alignment Alignment = Align_Left)
|
2019-07-19 20:56:21 +00:00
|
|
|
{
|
2019-08-18 12:56:18 +00:00
|
|
|
DEBUG_TRACK_FUNCTION;
|
|
|
|
v2 LowerRight = Position;
|
2019-07-19 20:56:21 +00:00
|
|
|
|
2019-08-18 12:56:18 +00:00
|
|
|
render_quad_batch_constructor BatchConstructor = PushRenderTexture2DBatch(RenderBuffer, String.Length,
|
|
|
|
Font->BitmapMemory,
|
|
|
|
Font->BitmapTextureHandle,
|
|
|
|
Font->BitmapWidth,
|
|
|
|
Font->BitmapHeight,
|
|
|
|
Font->BitmapBytesPerPixel,
|
|
|
|
Font->BitmapStride);
|
2019-07-19 20:56:21 +00:00
|
|
|
|
2019-08-18 12:56:18 +00:00
|
|
|
v2 RegisterPosition = Position;
|
2019-11-02 21:09:57 +00:00
|
|
|
if (Alignment == Align_Left)
|
2019-07-19 20:56:21 +00:00
|
|
|
{
|
2020-03-20 07:55:13 +00:00
|
|
|
RegisterPosition = DrawStringLeftAligned(&BatchConstructor, StringExpand(String), RegisterPosition, Font, Color);
|
2019-11-02 21:09:57 +00:00
|
|
|
}
|
|
|
|
else if (Alignment == Align_Right)
|
|
|
|
{
|
2020-03-20 07:55:13 +00:00
|
|
|
RegisterPosition = DrawStringRightAligned(&BatchConstructor, StringExpand(String), RegisterPosition, Font, Color);
|
2019-11-02 21:09:57 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
InvalidCodePath;
|
2019-07-19 20:56:21 +00:00
|
|
|
}
|
2019-08-18 12:56:18 +00:00
|
|
|
|
|
|
|
LowerRight.x = RegisterPosition.x;
|
|
|
|
|
|
|
|
return LowerRight;
|
2019-07-19 20:56:21 +00:00
|
|
|
}
|
|
|
|
|
2019-08-18 12:56:18 +00:00
|
|
|
internal void
|
2019-11-02 21:09:57 +00:00
|
|
|
DrawCursor (render_quad_batch_constructor* BatchConstructor, v2 Position, v4 Color, bitmap_font Font)
|
2019-07-19 20:56:21 +00:00
|
|
|
{
|
2019-08-18 12:56:18 +00:00
|
|
|
v2 Min = Position;
|
2019-11-02 21:09:57 +00:00
|
|
|
v2 Max = Position + v2{(r32)Font.MaxCharWidth, (r32)(Font.Ascent + Font.Descent)};
|
2019-08-18 12:56:18 +00:00
|
|
|
PushQuad2DOnBatch(BatchConstructor, Min, Max, Color);
|
2019-07-19 20:56:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
internal v2
|
2020-11-08 06:54:59 +00:00
|
|
|
DrawStringWithCursor (render_command_buffer* RenderBuffer, gs_string String, s32 CursorPosition, bitmap_font* Font, v2 Position, v4 Color, v4 CursorColor, gs_string_alignment Alignment = Align_Left)
|
2019-07-19 20:56:21 +00:00
|
|
|
{
|
|
|
|
DEBUG_TRACK_FUNCTION;
|
|
|
|
v2 LowerRight = Position;
|
|
|
|
|
2019-08-18 12:56:18 +00:00
|
|
|
// NOTE(Peter): We push this on first so that the cursor will be drawn underneath any character it may overlap with
|
|
|
|
render_quad_batch_constructor CursorBatch = PushRenderQuad2DBatch(RenderBuffer, 1);
|
2019-07-19 20:56:21 +00:00
|
|
|
render_quad_batch_constructor BatchConstructor = PushRenderTexture2DBatch(RenderBuffer, String.Length,
|
2019-07-22 06:30:53 +00:00
|
|
|
Font->BitmapMemory,
|
|
|
|
Font->BitmapTextureHandle,
|
|
|
|
Font->BitmapWidth,
|
|
|
|
Font->BitmapHeight,
|
|
|
|
Font->BitmapBytesPerPixel,
|
|
|
|
Font->BitmapStride);
|
2019-07-19 20:56:21 +00:00
|
|
|
|
|
|
|
v2 RegisterPosition = Position;
|
2019-11-02 21:09:57 +00:00
|
|
|
if (Alignment == Align_Left)
|
2019-07-19 20:56:21 +00:00
|
|
|
{
|
2020-03-20 07:55:13 +00:00
|
|
|
RegisterPosition = DrawStringLeftAligned(&BatchConstructor, StringExpand(String), RegisterPosition, Font, Color);
|
2019-11-02 21:09:57 +00:00
|
|
|
DrawCursor(&CursorBatch, RegisterPosition, GreenV4, *Font);
|
|
|
|
if (String.Length - CursorPosition > 0)
|
2019-08-18 12:56:18 +00:00
|
|
|
{
|
2019-11-02 21:09:57 +00:00
|
|
|
RegisterPosition = DrawStringLeftAligned(&BatchConstructor,
|
|
|
|
String.Length - CursorPosition,
|
2020-07-18 19:00:14 +00:00
|
|
|
String.Str + CursorPosition,
|
2019-11-02 21:09:57 +00:00
|
|
|
RegisterPosition, Font, Color);
|
2019-08-18 12:56:18 +00:00
|
|
|
}
|
2019-07-19 20:56:21 +00:00
|
|
|
}
|
2019-11-02 21:09:57 +00:00
|
|
|
else if (Alignment == Align_Right)
|
|
|
|
{
|
|
|
|
RegisterPosition = DrawStringRightAligned(&BatchConstructor,
|
2020-07-18 19:00:14 +00:00
|
|
|
CursorPosition, String.Str,
|
2019-11-02 21:09:57 +00:00
|
|
|
RegisterPosition, Font, Color);
|
|
|
|
DrawCursor(&CursorBatch, RegisterPosition, GreenV4, *Font);
|
|
|
|
if (String.Length - CursorPosition > 0)
|
|
|
|
{
|
|
|
|
RegisterPosition = DrawStringRightAligned(&BatchConstructor,
|
|
|
|
String.Length - CursorPosition,
|
2020-07-18 19:00:14 +00:00
|
|
|
String.Str + CursorPosition,
|
2019-11-02 21:09:57 +00:00
|
|
|
RegisterPosition, Font, Color);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
2019-08-18 12:56:18 +00:00
|
|
|
{
|
2019-11-02 21:09:57 +00:00
|
|
|
InvalidCodePath;
|
2019-08-18 12:56:18 +00:00
|
|
|
}
|
2019-07-19 20:56:21 +00:00
|
|
|
|
2019-08-18 12:56:18 +00:00
|
|
|
LowerRight.x = RegisterPosition.x;
|
2019-07-19 20:56:21 +00:00
|
|
|
return LowerRight;
|
|
|
|
}
|
|
|
|
|
2020-11-08 06:54:59 +00:00
|
|
|
enum ui_widget_flag
|
|
|
|
{
|
|
|
|
UIWidgetFlag_DrawBackground,
|
2020-11-09 03:42:14 +00:00
|
|
|
UIWidgetFlag_DrawString,
|
2020-11-08 06:54:59 +00:00
|
|
|
UIWidgetFlag_DrawOutline,
|
|
|
|
UIWidgetFlag_Clickable,
|
2020-11-15 07:30:24 +00:00
|
|
|
UIWidgetFlag_DrawHorizontalFill,
|
2020-11-08 06:54:59 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
struct ui_widget_id
|
|
|
|
{
|
2020-11-09 03:42:14 +00:00
|
|
|
u64 Id;
|
|
|
|
u64 ParentId;
|
|
|
|
};
|
|
|
|
|
|
|
|
enum ui_layout_direction
|
|
|
|
{
|
|
|
|
LayoutDirection_TopDown,
|
|
|
|
LayoutDirection_BottomUp,
|
2020-11-08 06:54:59 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
struct ui_widget
|
|
|
|
{
|
|
|
|
ui_widget_id Id;
|
|
|
|
|
|
|
|
gs_string String;
|
|
|
|
gs_string_alignment Alignment;
|
|
|
|
|
|
|
|
rect2 Bounds;
|
|
|
|
u64 Flags;
|
2020-11-09 03:42:14 +00:00
|
|
|
|
|
|
|
ui_widget* Next;
|
|
|
|
|
2020-11-15 07:30:24 +00:00
|
|
|
// Slider
|
|
|
|
r32 HorizontalFillPercent;
|
|
|
|
|
2020-11-09 03:42:14 +00:00
|
|
|
// Layout
|
|
|
|
ui_widget* Parent;
|
|
|
|
|
|
|
|
v2 Margin;
|
|
|
|
r32 RowHeight;
|
|
|
|
r32 RowYAt;
|
|
|
|
|
|
|
|
ui_layout_direction FillDirection;
|
|
|
|
|
|
|
|
b32 DrawHorizontal;
|
|
|
|
u32 ColumnsMax;
|
|
|
|
r32* ColumnWidths;
|
|
|
|
u32 ColumnsCount;
|
|
|
|
|
|
|
|
// NOTE(pjs): I'm not sure this will stay but
|
|
|
|
// its here so that when we end things like a dropdown,
|
|
|
|
// we can check the retained state of that dropdown
|
|
|
|
ui_widget_id WidgetReference;
|
|
|
|
|
|
|
|
ui_widget* ChildrenRoot;
|
|
|
|
ui_widget* ChildrenHead;
|
|
|
|
u32 ChildCount;
|
2020-11-08 06:54:59 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
struct ui_eval_result
|
|
|
|
{
|
|
|
|
bool Clicked;
|
2020-11-15 07:30:24 +00:00
|
|
|
bool Held;
|
|
|
|
v2 DragDelta;
|
2020-11-08 06:54:59 +00:00
|
|
|
};
|
|
|
|
|
2019-07-19 20:56:21 +00:00
|
|
|
struct interface_config
|
|
|
|
{
|
|
|
|
v4 PanelBGColors[4];
|
2020-03-20 07:55:13 +00:00
|
|
|
|
2020-11-08 06:54:59 +00:00
|
|
|
// TODO(pjs): Turn these into _Default, _Hot, _Active
|
2019-07-19 20:56:21 +00:00
|
|
|
v4 ButtonColor_Inactive, ButtonColor_Active, ButtonColor_Selected;
|
2020-03-20 07:55:13 +00:00
|
|
|
|
2019-07-19 20:56:21 +00:00
|
|
|
v4 TextColor;
|
2020-03-20 07:55:13 +00:00
|
|
|
|
|
|
|
#define LIST_BG_COLORS_COUNT 2
|
|
|
|
v4 ListBGColors[LIST_BG_COLORS_COUNT];
|
|
|
|
v4 ListBGHover;
|
|
|
|
v4 ListBGSelected;
|
|
|
|
|
2019-07-19 20:56:21 +00:00
|
|
|
bitmap_font* Font;
|
|
|
|
r32 FontSize;
|
|
|
|
v2 Margin;
|
2020-03-20 07:55:13 +00:00
|
|
|
r32 RowHeight;
|
2019-07-19 20:56:21 +00:00
|
|
|
};
|
|
|
|
|
2020-11-08 06:54:59 +00:00
|
|
|
struct ui_widget_retained_state
|
|
|
|
{
|
|
|
|
ui_widget_id Id;
|
|
|
|
bool Value;
|
2020-11-15 07:30:24 +00:00
|
|
|
r32 InitialValueR32;
|
2020-11-15 01:18:38 +00:00
|
|
|
u32 FramesSinceAccess;
|
2019-07-19 20:56:21 +00:00
|
|
|
};
|
|
|
|
|
2020-03-20 07:55:13 +00:00
|
|
|
struct ui_interface
|
2019-07-19 20:56:21 +00:00
|
|
|
{
|
2020-03-20 07:55:13 +00:00
|
|
|
interface_config Style;
|
|
|
|
mouse_state Mouse;
|
|
|
|
render_command_buffer* RenderBuffer;
|
2020-11-08 06:54:59 +00:00
|
|
|
|
|
|
|
ui_widget* Widgets;
|
|
|
|
u64 WidgetsCount;
|
|
|
|
u64 WidgetsCountMax;
|
|
|
|
|
2020-11-09 03:42:14 +00:00
|
|
|
ui_widget* DrawOrderHead;
|
|
|
|
ui_widget* DrawOrderRoot;
|
|
|
|
|
2020-11-08 06:54:59 +00:00
|
|
|
ui_widget_id HotWidget;
|
|
|
|
ui_widget_id ActiveWidget;
|
|
|
|
|
2020-11-09 03:42:14 +00:00
|
|
|
ui_widget* ActiveLayout;
|
2020-11-08 06:54:59 +00:00
|
|
|
|
|
|
|
#define RETAINED_STATE_MAX 128
|
|
|
|
ui_widget_retained_state RetainedState[RETAINED_STATE_MAX];
|
|
|
|
u64 RetainedStateCount;
|
2020-11-15 01:18:38 +00:00
|
|
|
|
|
|
|
gs_memory_arena* PerFrameMemory;
|
2020-03-20 07:55:13 +00:00
|
|
|
};
|
|
|
|
|
2020-11-08 06:54:59 +00:00
|
|
|
internal void
|
|
|
|
ui_InterfaceReset(ui_interface* Interface)
|
|
|
|
{
|
|
|
|
Interface->WidgetsCount = 0;
|
2020-11-09 03:42:14 +00:00
|
|
|
Interface->DrawOrderHead = 0;
|
|
|
|
Interface->DrawOrderRoot = 0;
|
2020-11-15 01:18:38 +00:00
|
|
|
ClearArena(Interface->PerFrameMemory);
|
|
|
|
|
|
|
|
for (u32 i = 0; i < Interface->RetainedStateCount; i++)
|
|
|
|
{
|
|
|
|
Interface->RetainedState[i].FramesSinceAccess += 1;
|
|
|
|
if (Interface->RetainedState[i].FramesSinceAccess > 1)
|
|
|
|
{
|
|
|
|
Interface->RetainedState[i] = {0};
|
|
|
|
}
|
|
|
|
}
|
2020-11-08 06:54:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
internal bool
|
|
|
|
ui_WidgetIdsEqual(ui_widget_id A, ui_widget_id B)
|
|
|
|
{
|
2020-11-09 03:42:14 +00:00
|
|
|
bool Result = (A.Id == B.Id) && (A.ParentId == B.ParentId);
|
2020-11-08 06:54:59 +00:00
|
|
|
return Result;
|
|
|
|
}
|
|
|
|
|
|
|
|
internal ui_widget_retained_state*
|
|
|
|
ui_GetRetainedState(ui_interface* Interface, ui_widget_id Id)
|
|
|
|
{
|
|
|
|
ui_widget_retained_state* Result = 0;
|
|
|
|
for (u64 i = 0; i < Interface->RetainedStateCount; i++)
|
|
|
|
{
|
|
|
|
if (ui_WidgetIdsEqual(Interface->RetainedState[i].Id, Id))
|
|
|
|
{
|
2020-11-15 01:18:38 +00:00
|
|
|
Interface->RetainedState[i].FramesSinceAccess = 0;
|
2020-11-08 06:54:59 +00:00
|
|
|
Result = Interface->RetainedState + i;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return Result;
|
|
|
|
}
|
|
|
|
|
|
|
|
internal ui_widget_retained_state*
|
2020-11-15 01:18:38 +00:00
|
|
|
ui_CreateRetainedState(ui_interface* Interface, ui_widget* Widget)
|
2020-11-08 06:54:59 +00:00
|
|
|
{
|
|
|
|
u64 Index = Interface->RetainedStateCount++;
|
|
|
|
ui_widget_retained_state* Result = Interface->RetainedState + Index;
|
2020-11-15 01:18:38 +00:00
|
|
|
Result->Id = Widget->Id;
|
2020-11-08 06:54:59 +00:00
|
|
|
return Result;
|
|
|
|
}
|
|
|
|
|
2020-11-09 03:42:14 +00:00
|
|
|
internal ui_widget*
|
|
|
|
ui_CreateWidget(ui_interface* Interface, gs_string String)
|
|
|
|
{
|
|
|
|
Assert(Interface->WidgetsCount < Interface->WidgetsCountMax);
|
|
|
|
ui_widget* Result = Interface->Widgets + Interface->WidgetsCount++;
|
2020-11-15 01:18:38 +00:00
|
|
|
ZeroStruct(Result);
|
|
|
|
|
2020-11-09 03:42:14 +00:00
|
|
|
Result->Parent = Interface->ActiveLayout;
|
|
|
|
|
|
|
|
u64 Id = HashDJB2ToU64(StringExpand(String));
|
|
|
|
if (Result->Parent)
|
|
|
|
{
|
|
|
|
Id = HashAppendDJB2ToU32(Id, Result->Parent->Id.Id);
|
|
|
|
Id = HashAppendDJB2ToU32(Id, Result->Parent->ChildCount);
|
2020-11-15 01:18:38 +00:00
|
|
|
Result->Id.ParentId = Result->Parent->Id.Id;
|
2020-11-09 03:42:14 +00:00
|
|
|
}
|
|
|
|
Result->Id.Id = Id;
|
|
|
|
|
2020-11-15 01:18:38 +00:00
|
|
|
Result->String = PushStringCopy(Interface->PerFrameMemory, String.ConstString);
|
2020-11-09 03:42:14 +00:00
|
|
|
Result->Alignment = Align_Left;
|
|
|
|
Result->Next = 0;
|
|
|
|
Result->ChildrenRoot = 0;
|
|
|
|
Result->ChildrenHead = 0;
|
|
|
|
Result->Flags = 0;
|
|
|
|
return Result;
|
|
|
|
}
|
|
|
|
|
2020-11-15 01:18:38 +00:00
|
|
|
internal void
|
|
|
|
ui_WidgetSetFlag(ui_widget* Widget, u64 Flag)
|
|
|
|
{
|
|
|
|
u64 Value = ((u64)1 << Flag);
|
|
|
|
Widget->Flags = Widget->Flags | Value;
|
|
|
|
}
|
|
|
|
|
|
|
|
internal bool
|
|
|
|
ui_WidgetIsFlagSet(ui_widget Widget, u64 Flag)
|
|
|
|
{
|
|
|
|
u64 Value = ((u64)1 << Flag);
|
|
|
|
bool Result = (Widget.Flags & Value);
|
|
|
|
return Result;
|
|
|
|
}
|
|
|
|
|
2020-11-08 06:54:59 +00:00
|
|
|
//
|
|
|
|
// Interaction
|
|
|
|
//
|
|
|
|
|
|
|
|
internal b32
|
|
|
|
ui_MouseClickedRect(ui_interface Interface, rect2 Rect)
|
|
|
|
{
|
|
|
|
b32 Result = MouseButtonTransitionedDown(Interface.Mouse.LeftButtonState);
|
|
|
|
Result &= PointIsInRect(Rect, Interface.Mouse.Pos);
|
|
|
|
return Result;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Layout
|
|
|
|
|
2020-11-15 07:38:56 +00:00
|
|
|
internal ui_widget*
|
|
|
|
ui_CreateLayoutWidget(ui_interface* Interface, rect2 Bounds, ui_layout_direction FillDir, gs_string Name)
|
2020-03-20 07:55:13 +00:00
|
|
|
{
|
2020-11-15 01:18:38 +00:00
|
|
|
ui_widget* Result = ui_CreateWidget(Interface, Name);
|
2020-11-15 07:30:24 +00:00
|
|
|
//ui_WidgetSetFlag(Result, UIWidgetFlag_DrawOutline);
|
2020-11-15 01:18:38 +00:00
|
|
|
|
2020-11-09 03:42:14 +00:00
|
|
|
Result->Bounds = Bounds;
|
|
|
|
Result->Margin = Interface->Style.Margin;
|
|
|
|
Result->RowHeight = Interface->Style.RowHeight;
|
|
|
|
Result->FillDirection = FillDir;
|
2020-11-15 01:18:38 +00:00
|
|
|
|
2020-11-09 03:42:14 +00:00
|
|
|
switch(FillDir)
|
2020-11-08 06:54:59 +00:00
|
|
|
{
|
|
|
|
case LayoutDirection_BottomUp:
|
|
|
|
{
|
2020-11-15 07:30:24 +00:00
|
|
|
Result->RowYAt = Bounds.Min.y + Result->Margin.y;
|
2020-11-08 06:54:59 +00:00
|
|
|
}break;
|
|
|
|
|
|
|
|
case LayoutDirection_TopDown:
|
|
|
|
{
|
2020-11-15 07:30:24 +00:00
|
|
|
Result->RowYAt = Bounds.Max.y - (Result->RowHeight + Result->Margin.y);
|
2020-11-08 06:54:59 +00:00
|
|
|
}break;
|
|
|
|
}
|
|
|
|
|
2020-11-15 07:38:56 +00:00
|
|
|
return Result;
|
|
|
|
}
|
|
|
|
|
|
|
|
static ui_widget*
|
|
|
|
ui_PushLayout(ui_interface* Interface, rect2 Bounds, ui_layout_direction FillDir, gs_string Name)
|
|
|
|
{
|
|
|
|
ui_widget* Result = ui_CreateLayoutWidget(Interface, Bounds, FillDir, Name);
|
|
|
|
|
2020-11-09 03:42:14 +00:00
|
|
|
if (Interface->DrawOrderRoot)
|
|
|
|
{
|
|
|
|
SLLPushOrInit(Interface->ActiveLayout->ChildrenRoot, Interface->ActiveLayout->ChildrenHead, Result);
|
|
|
|
Interface->ActiveLayout->ChildCount++;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
SLLPushOrInit(Interface->DrawOrderRoot, Interface->DrawOrderHead, Result);
|
|
|
|
}
|
2020-11-15 07:38:56 +00:00
|
|
|
|
|
|
|
Interface->ActiveLayout = Result;
|
|
|
|
return Result;
|
|
|
|
}
|
|
|
|
|
|
|
|
static ui_widget*
|
|
|
|
ui_PushOverlayLayout(ui_interface* Interface, rect2 Bounds, ui_layout_direction FillDir, gs_string Name)
|
|
|
|
{
|
|
|
|
ui_widget* Result = ui_CreateLayoutWidget(Interface, Bounds, FillDir, Name);
|
|
|
|
SLLPushOrInit(Interface->DrawOrderRoot, Interface->DrawOrderHead, Result);
|
2020-11-09 03:42:14 +00:00
|
|
|
Interface->ActiveLayout = Result;
|
2020-03-20 07:55:13 +00:00
|
|
|
return Result;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2020-11-08 06:54:59 +00:00
|
|
|
ui_PopLayout(ui_interface* Interface)
|
2020-03-20 07:55:13 +00:00
|
|
|
{
|
2020-11-09 03:42:14 +00:00
|
|
|
Assert(Interface->ActiveLayout != 0);
|
|
|
|
Interface->ActiveLayout = Interface->ActiveLayout->Parent;
|
2020-03-20 07:55:13 +00:00
|
|
|
}
|
|
|
|
|
2020-03-22 05:44:44 +00:00
|
|
|
static void
|
2020-11-08 06:54:59 +00:00
|
|
|
ui_StartRow(ui_interface* Interface, u32 ColumnsMax = 0)
|
2020-03-22 05:44:44 +00:00
|
|
|
{
|
2020-11-09 03:42:14 +00:00
|
|
|
Interface->ActiveLayout->DrawHorizontal = true;
|
|
|
|
Interface->ActiveLayout->ColumnsMax = ColumnsMax;
|
|
|
|
Interface->ActiveLayout->ColumnWidths = 0;
|
|
|
|
Interface->ActiveLayout->ColumnsCount = 0;
|
2020-03-22 05:44:44 +00:00
|
|
|
}
|
|
|
|
|
2020-03-20 07:55:13 +00:00
|
|
|
static void
|
2020-11-08 06:54:59 +00:00
|
|
|
ui_StartRow(ui_interface* Interface, u32 ColumnsMax, r32* ColumnWidths)
|
2020-03-20 07:55:13 +00:00
|
|
|
{
|
2020-11-09 03:42:14 +00:00
|
|
|
Interface->ActiveLayout->DrawHorizontal = true;
|
|
|
|
Interface->ActiveLayout->ColumnsMax = ColumnsMax;
|
|
|
|
Interface->ActiveLayout->ColumnWidths = ColumnWidths;
|
|
|
|
Interface->ActiveLayout->ColumnsCount = 0;
|
2020-11-08 06:54:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
ui_EndRow(ui_interface* Interface)
|
|
|
|
{
|
2020-11-09 03:42:14 +00:00
|
|
|
Interface->ActiveLayout->DrawHorizontal = false;
|
|
|
|
Interface->ActiveLayout->ColumnWidths = 0;
|
2020-11-15 07:30:24 +00:00
|
|
|
Interface->ActiveLayout->RowYAt -= (Interface->ActiveLayout->RowHeight + Interface->ActiveLayout->Margin.y);
|
2020-03-20 07:55:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static b32
|
2020-11-09 03:42:14 +00:00
|
|
|
ui_TryReserveElementBounds(ui_widget* Widget, rect2* Bounds)
|
2020-03-20 07:55:13 +00:00
|
|
|
{
|
|
|
|
b32 Result = true;
|
2020-11-09 03:42:14 +00:00
|
|
|
if (!Widget->DrawHorizontal)
|
2019-07-19 20:56:21 +00:00
|
|
|
{
|
2020-11-15 07:30:24 +00:00
|
|
|
Bounds->Min = { Widget->Bounds.Min.x + Widget->Margin.x, Widget->RowYAt };
|
|
|
|
Bounds->Max = { Widget->Bounds.Max.x - Widget->Margin.x, Bounds->Min.y + Widget->RowHeight };
|
2020-11-08 06:54:59 +00:00
|
|
|
|
2020-11-15 07:30:24 +00:00
|
|
|
r32 RowOffset = Widget->RowHeight + Widget->Margin.y;
|
2020-11-09 03:42:14 +00:00
|
|
|
switch (Widget->FillDirection)
|
2020-11-08 06:54:59 +00:00
|
|
|
{
|
|
|
|
case LayoutDirection_BottomUp:
|
|
|
|
{
|
2020-11-15 07:30:24 +00:00
|
|
|
Widget->RowYAt += RowOffset;
|
2020-11-08 06:54:59 +00:00
|
|
|
}break;
|
|
|
|
|
|
|
|
case LayoutDirection_TopDown:
|
|
|
|
{
|
2020-11-15 07:30:24 +00:00
|
|
|
Widget->RowYAt -= RowOffset;
|
2020-11-08 06:54:59 +00:00
|
|
|
}break;
|
|
|
|
|
|
|
|
InvalidDefaultCase;
|
|
|
|
}
|
2020-03-20 07:55:13 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2020-11-09 03:42:14 +00:00
|
|
|
if (Widget->ColumnsMax > 0)
|
2019-07-19 20:56:21 +00:00
|
|
|
{
|
2020-11-09 03:42:14 +00:00
|
|
|
Assert(Widget->ColumnsCount < Widget->ColumnsMax);
|
|
|
|
if (Widget->ColumnWidths != 0)
|
2020-03-22 05:44:44 +00:00
|
|
|
{
|
2020-11-15 07:30:24 +00:00
|
|
|
v2 Min = {
|
|
|
|
Widget->Bounds.Min.x + Widget->Margin.x,
|
|
|
|
Widget->RowYAt
|
|
|
|
};
|
2020-11-09 03:42:14 +00:00
|
|
|
for (u32 i = 0; i < Widget->ColumnsCount; i++)
|
2020-03-22 05:44:44 +00:00
|
|
|
{
|
2020-11-09 03:42:14 +00:00
|
|
|
Min.x += Widget->ColumnWidths[i];
|
2020-03-22 05:44:44 +00:00
|
|
|
}
|
|
|
|
Bounds->Min = Min;
|
2020-11-15 07:30:24 +00:00
|
|
|
Bounds->Max = Bounds->Min + v2{
|
|
|
|
Widget->ColumnWidths[Widget->ColumnsCount] - Widget->Margin.x,
|
|
|
|
Widget->RowHeight
|
|
|
|
};
|
2020-03-22 05:44:44 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2020-11-09 03:42:14 +00:00
|
|
|
r32 ElementWidth = Rect2Width(Widget->Bounds) / Widget->ColumnsMax;
|
2020-03-22 05:44:44 +00:00
|
|
|
Bounds->Min = {
|
2020-11-09 03:42:14 +00:00
|
|
|
Widget->Bounds.Min.x + (ElementWidth * Widget->ColumnsCount) + Widget->Margin.x,
|
|
|
|
Widget->RowYAt
|
2020-03-22 05:44:44 +00:00
|
|
|
};
|
|
|
|
Bounds->Max = {
|
2020-11-15 07:30:24 +00:00
|
|
|
Bounds->Min.x + ElementWidth - (Widget->Margin.x * 2),
|
2020-11-09 03:42:14 +00:00
|
|
|
Bounds->Min.y + Widget->RowHeight
|
2020-03-22 05:44:44 +00:00
|
|
|
};
|
|
|
|
}
|
2020-11-09 03:42:14 +00:00
|
|
|
Widget->ColumnsCount++;
|
2019-07-19 20:56:21 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2020-03-20 07:55:13 +00:00
|
|
|
Result = false;
|
2019-07-19 20:56:21 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return Result;
|
|
|
|
}
|
|
|
|
|
2020-07-18 19:00:14 +00:00
|
|
|
static rect2
|
2020-11-09 03:42:14 +00:00
|
|
|
ui_ReserveElementBounds(ui_widget* Layout)
|
2020-03-22 04:13:35 +00:00
|
|
|
{
|
2020-07-18 19:00:14 +00:00
|
|
|
rect2 Bounds = {0};
|
2020-03-22 04:13:35 +00:00
|
|
|
if (!ui_TryReserveElementBounds(Layout, &Bounds))
|
|
|
|
{
|
|
|
|
InvalidCodePath;
|
|
|
|
}
|
|
|
|
return Bounds;
|
|
|
|
}
|
|
|
|
|
2020-07-18 19:00:14 +00:00
|
|
|
static rect2
|
2020-11-09 03:42:14 +00:00
|
|
|
ui_LayoutRemaining(ui_widget Layout)
|
2020-03-22 05:44:44 +00:00
|
|
|
{
|
2020-07-18 19:00:14 +00:00
|
|
|
rect2 Result = Layout.Bounds;
|
2020-03-22 05:44:44 +00:00
|
|
|
Result.Max.y = Layout.RowYAt;
|
|
|
|
if (Layout.DrawHorizontal)
|
|
|
|
{
|
|
|
|
Result.Max.y -= Layout.RowHeight;
|
|
|
|
}
|
|
|
|
return Result;
|
2020-11-03 20:49:16 +00:00
|
|
|
}
|
|
|
|
|
2020-11-08 06:54:59 +00:00
|
|
|
// Widgets
|
2020-11-03 20:49:16 +00:00
|
|
|
|
2020-11-08 06:54:59 +00:00
|
|
|
internal ui_eval_result
|
|
|
|
ui_EvaluateWidget(ui_interface* Interface, ui_widget* Widget, rect2 Bounds)
|
|
|
|
{
|
|
|
|
ui_eval_result Result = {};
|
|
|
|
|
|
|
|
Widget->Bounds = Bounds;
|
2020-11-09 03:42:14 +00:00
|
|
|
SLLPushOrInit(Interface->ActiveLayout->ChildrenRoot, Interface->ActiveLayout->ChildrenHead, Widget);
|
2020-11-15 01:18:38 +00:00
|
|
|
Interface->ActiveLayout->ChildCount += 1;
|
2020-11-08 06:54:59 +00:00
|
|
|
|
|
|
|
if (ui_WidgetIsFlagSet(*Widget, UIWidgetFlag_Clickable))
|
|
|
|
{
|
|
|
|
if (PointIsInRect(Widget->Bounds, Interface->Mouse.Pos))
|
|
|
|
{
|
|
|
|
if (ui_WidgetIdsEqual(Interface->HotWidget, Widget->Id) && MouseButtonTransitionedDown(Interface->Mouse.LeftButtonState))
|
|
|
|
{
|
2020-11-15 07:30:24 +00:00
|
|
|
Assert(!ui_WidgetIdsEqual(Interface->ActiveWidget, Widget->Id));
|
2020-11-08 06:54:59 +00:00
|
|
|
Result.Clicked = true;
|
|
|
|
Interface->ActiveWidget = Widget->Id;
|
|
|
|
}
|
|
|
|
Interface->HotWidget = Widget->Id;
|
|
|
|
}
|
2020-11-15 07:30:24 +00:00
|
|
|
|
|
|
|
|
|
|
|
if (MouseButtonHeldDown(Interface->Mouse.LeftButtonState) &&
|
|
|
|
PointIsInRect(Widget->Bounds, Interface->Mouse.DownPos))
|
|
|
|
{
|
|
|
|
Result.Held = true;
|
|
|
|
Result.DragDelta = Interface->Mouse.Pos - Interface->Mouse.DownPos;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ui_WidgetIdsEqual(Interface->ActiveWidget, Widget->Id) &&
|
|
|
|
MouseButtonTransitionedUp(Interface->Mouse.LeftButtonState))
|
|
|
|
{
|
|
|
|
Interface->ActiveWidget = {};
|
|
|
|
}
|
|
|
|
|
2020-11-08 06:54:59 +00:00
|
|
|
}
|
|
|
|
|
2020-11-03 20:49:16 +00:00
|
|
|
return Result;
|
2020-03-22 05:44:44 +00:00
|
|
|
}
|
2020-06-22 04:59:42 +00:00
|
|
|
|
2020-11-08 06:54:59 +00:00
|
|
|
internal ui_eval_result
|
|
|
|
ui_EvaluateWidget(ui_interface* Interface, ui_widget* Widget)
|
|
|
|
{
|
|
|
|
rect2 Bounds = {0};
|
2020-11-09 03:42:14 +00:00
|
|
|
ui_widget* Layout = Interface->ActiveLayout;
|
2020-11-08 06:54:59 +00:00
|
|
|
if (!ui_TryReserveElementBounds(Layout, &Bounds))
|
|
|
|
{
|
|
|
|
// TODO(pjs): This isn't invalid, but Idk when we'd hit this case yet
|
|
|
|
InvalidCodePath;
|
|
|
|
}
|
|
|
|
|
|
|
|
return ui_EvaluateWidget(Interface, Widget, Bounds);
|
|
|
|
}
|
|
|
|
|
2020-03-20 07:55:13 +00:00
|
|
|
//
|
|
|
|
// Drawing Functions
|
|
|
|
//
|
|
|
|
|
2020-06-22 04:59:42 +00:00
|
|
|
static r32
|
|
|
|
ui_GetTextLineHeight(ui_interface Interface)
|
|
|
|
{
|
|
|
|
r32 Result = Interface.Style.Font->PixelHeight + (2 * Interface.Style.Margin.y);
|
|
|
|
return Result;
|
|
|
|
}
|
|
|
|
|
2020-03-20 07:55:13 +00:00
|
|
|
static void
|
2020-07-18 19:00:14 +00:00
|
|
|
ui_FillRect(ui_interface* Interface, rect2 Bounds, v4 Color)
|
2020-03-20 07:55:13 +00:00
|
|
|
{
|
2020-07-18 19:00:14 +00:00
|
|
|
PushRenderQuad2D(Interface->RenderBuffer, Bounds.Min, Bounds.Max, Color);
|
2019-07-19 20:56:21 +00:00
|
|
|
}
|
|
|
|
|
2020-03-20 07:55:13 +00:00
|
|
|
static void
|
2020-07-18 19:00:14 +00:00
|
|
|
ui_OutlineRect(ui_interface* Interface, rect2 Bounds, r32 Thickness, v4 Color)
|
2019-07-19 20:56:21 +00:00
|
|
|
{
|
2020-03-20 07:55:13 +00:00
|
|
|
PushRenderBoundingBox2D(Interface->RenderBuffer, Bounds.Min, Bounds.Max, Thickness, Color);
|
|
|
|
}
|
|
|
|
|
|
|
|
internal void
|
2020-11-15 07:44:06 +00:00
|
|
|
ui_Label(ui_interface* Interface, gs_string String, rect2 Bounds, gs_string_alignment Alignment = Align_Left)
|
2020-03-20 07:55:13 +00:00
|
|
|
{
|
|
|
|
DEBUG_TRACK_FUNCTION;
|
2020-11-09 03:42:14 +00:00
|
|
|
ui_widget* Widget = ui_CreateWidget(Interface, String);
|
2020-11-15 07:44:06 +00:00
|
|
|
ui_WidgetSetFlag(Widget, UIWidgetFlag_DrawString);
|
|
|
|
ui_EvaluateWidget(Interface, Widget, Bounds);
|
2020-03-22 04:13:35 +00:00
|
|
|
}
|
|
|
|
|
2020-11-08 06:54:59 +00:00
|
|
|
internal void
|
2020-11-15 07:44:06 +00:00
|
|
|
ui_Label(ui_interface* Interface, gs_string String, gs_string_alignment Alignment = Align_Left)
|
2019-07-19 20:56:21 +00:00
|
|
|
{
|
2020-11-08 06:54:59 +00:00
|
|
|
DEBUG_TRACK_FUNCTION;
|
2020-11-09 03:42:14 +00:00
|
|
|
ui_widget* Widget = ui_CreateWidget(Interface, String);
|
2020-11-15 07:44:06 +00:00
|
|
|
ui_WidgetSetFlag(Widget, UIWidgetFlag_DrawString);
|
2020-11-09 03:42:14 +00:00
|
|
|
ui_EvaluateWidget(Interface, Widget);
|
2020-03-20 07:55:13 +00:00
|
|
|
}
|
2019-07-19 20:56:21 +00:00
|
|
|
|
2020-11-15 07:30:24 +00:00
|
|
|
internal ui_widget*
|
|
|
|
ui_CreateButtonWidget(ui_interface* Interface, gs_string Text)
|
2019-07-19 20:56:21 +00:00
|
|
|
{
|
2020-11-09 03:42:14 +00:00
|
|
|
ui_widget* Widget = ui_CreateWidget(Interface, Text);
|
|
|
|
ui_WidgetSetFlag(Widget, UIWidgetFlag_Clickable);
|
|
|
|
ui_WidgetSetFlag(Widget, UIWidgetFlag_DrawBackground);
|
|
|
|
ui_WidgetSetFlag(Widget, UIWidgetFlag_DrawString);
|
2020-11-15 07:30:24 +00:00
|
|
|
ui_WidgetSetFlag(Widget, UIWidgetFlag_DrawOutline);
|
|
|
|
return Widget;
|
|
|
|
}
|
|
|
|
|
|
|
|
internal b32
|
|
|
|
ui_Button(ui_interface* Interface, gs_string Text)
|
|
|
|
{
|
|
|
|
ui_widget* Widget = ui_CreateButtonWidget(Interface, Text);
|
2020-11-09 03:42:14 +00:00
|
|
|
ui_eval_result Result = ui_EvaluateWidget(Interface, Widget);
|
2020-11-08 06:54:59 +00:00
|
|
|
return Result.Clicked;
|
2019-07-19 20:56:21 +00:00
|
|
|
}
|
|
|
|
|
2020-11-15 07:30:24 +00:00
|
|
|
internal b32
|
2020-07-18 19:00:14 +00:00
|
|
|
ui_Button(ui_interface* Interface, gs_string Text, rect2 Bounds)
|
2019-07-19 20:56:21 +00:00
|
|
|
{
|
2020-11-15 07:30:24 +00:00
|
|
|
ui_widget* Widget = ui_CreateButtonWidget(Interface, Text);
|
2020-11-09 03:42:14 +00:00
|
|
|
ui_eval_result Result = ui_EvaluateWidget(Interface, Widget, Bounds);
|
2020-11-08 06:54:59 +00:00
|
|
|
return Result.Clicked;
|
2020-03-20 07:55:13 +00:00
|
|
|
}
|
|
|
|
|
2020-06-22 04:59:42 +00:00
|
|
|
struct list_item_colors
|
|
|
|
{
|
|
|
|
v4 Hover;
|
|
|
|
v4 Selected;
|
|
|
|
v4 BGColor;
|
|
|
|
};
|
|
|
|
|
|
|
|
inline v4
|
|
|
|
ui_GetListItemBGColor(interface_config Style, u32 ElementIndex)
|
|
|
|
{
|
|
|
|
v4 Result = Style.ListBGColors[ElementIndex % LIST_BG_COLORS_COUNT];
|
|
|
|
return Result;
|
|
|
|
}
|
|
|
|
|
|
|
|
static list_item_colors
|
|
|
|
ui_GetListItemColors(ui_interface* Interface, u32 ListItemIndex)
|
|
|
|
{
|
|
|
|
list_item_colors Result = {};
|
|
|
|
Result.Hover = Interface->Style.ListBGHover;
|
|
|
|
Result.Selected = Interface->Style.ListBGSelected;
|
|
|
|
Result.BGColor = ui_GetListItemBGColor(Interface->Style, ListItemIndex);
|
|
|
|
return Result;
|
|
|
|
}
|
|
|
|
|
|
|
|
static b32
|
2020-07-18 19:00:14 +00:00
|
|
|
ui_ListButton(ui_interface* Interface, gs_string Text, rect2 Bounds, u32 ListItemIndex)
|
2020-06-22 04:59:42 +00:00
|
|
|
{
|
2020-11-09 03:42:14 +00:00
|
|
|
ui_widget* Widget = ui_CreateWidget(Interface, Text);
|
|
|
|
ui_WidgetSetFlag(Widget, UIWidgetFlag_DrawBackground);
|
|
|
|
ui_WidgetSetFlag(Widget, UIWidgetFlag_Clickable);
|
2020-11-08 06:54:59 +00:00
|
|
|
// TODO(pjs): Reimplement alternating color backgrounds
|
2020-11-09 03:42:14 +00:00
|
|
|
Widget->Bounds = Bounds;
|
|
|
|
ui_eval_result Result = ui_EvaluateWidget(Interface, Widget);
|
2020-11-08 06:54:59 +00:00
|
|
|
return Result.Clicked;
|
2020-03-22 04:13:35 +00:00
|
|
|
}
|
|
|
|
|
2020-06-22 04:59:42 +00:00
|
|
|
static b32
|
2020-11-09 03:42:14 +00:00
|
|
|
ui_LayoutListButton(ui_interface* Interface, gs_string Text, u32 ListItemIndex)
|
2020-03-22 04:13:35 +00:00
|
|
|
{
|
2020-11-09 03:42:14 +00:00
|
|
|
// TODO(pjs): Reimplement alternating colors
|
2020-11-08 06:54:59 +00:00
|
|
|
return ui_Button(Interface, Text);
|
2020-03-20 07:55:13 +00:00
|
|
|
}
|
|
|
|
|
2020-11-08 06:54:59 +00:00
|
|
|
internal bool
|
2020-11-09 03:42:14 +00:00
|
|
|
ui_EvaluateDropdown(ui_interface* Interface, ui_widget* Widget, ui_eval_result EvalResult)
|
2020-11-08 06:54:59 +00:00
|
|
|
{
|
2020-11-09 03:42:14 +00:00
|
|
|
ui_widget_retained_state* State = ui_GetRetainedState(Interface, Widget->Id);
|
2020-11-15 07:44:06 +00:00
|
|
|
if (!State)
|
|
|
|
{
|
2020-11-15 01:18:38 +00:00
|
|
|
State = ui_CreateRetainedState(Interface, Widget);
|
2020-11-08 06:54:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (EvalResult.Clicked)
|
|
|
|
{
|
|
|
|
State->Value = !State->Value;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (State->Value)
|
|
|
|
{
|
2020-11-09 03:42:14 +00:00
|
|
|
ui_widget ParentLayout = *Interface->ActiveLayout;
|
2020-11-08 06:54:59 +00:00
|
|
|
|
2020-11-09 03:42:14 +00:00
|
|
|
r32 SpaceAbove = ParentLayout.Bounds.Max.y - Widget->Bounds.Max.y;
|
|
|
|
r32 SpaceBelow = Widget->Bounds.Min.y - ParentLayout.Bounds.Min.y;
|
2020-11-08 06:54:59 +00:00
|
|
|
ui_layout_direction Direction = LayoutDirection_TopDown;
|
|
|
|
rect2 MenuBounds = {};
|
|
|
|
|
|
|
|
if (SpaceAbove > SpaceBelow)
|
|
|
|
{
|
|
|
|
r32 ParentLayoutMaxY = ParentLayout.Bounds.Max.y;
|
|
|
|
Direction = LayoutDirection_BottomUp;
|
|
|
|
MenuBounds = rect2{
|
2020-11-09 03:42:14 +00:00
|
|
|
v2{ Widget->Bounds.Min.x, Widget->Bounds.Max.y },
|
|
|
|
v2{ Widget->Bounds.Max.x, ParentLayoutMaxY }
|
2020-11-08 06:54:59 +00:00
|
|
|
};
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
r32 ParentLayoutMinY = ParentLayout.Bounds.Min.y;
|
|
|
|
Direction = LayoutDirection_TopDown;
|
|
|
|
MenuBounds = rect2{
|
2020-11-09 03:42:14 +00:00
|
|
|
v2{ Widget->Bounds.Min.x, ParentLayoutMinY },
|
|
|
|
v2{ Widget->Bounds.Max.x, Widget->Bounds.Min.y }
|
2020-11-08 06:54:59 +00:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2020-11-15 07:38:56 +00:00
|
|
|
ui_widget* Layout = ui_PushOverlayLayout(Interface, MenuBounds, Direction, MakeString("WidgetLayout"));
|
2020-11-15 07:30:24 +00:00
|
|
|
Layout->Margin.y = 0;
|
2020-11-09 03:42:14 +00:00
|
|
|
Layout->WidgetReference = Widget->Id;
|
2020-11-08 06:54:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return State->Value;
|
|
|
|
}
|
|
|
|
|
|
|
|
internal bool
|
|
|
|
ui_BeginDropdown(ui_interface* Interface, gs_string Text, rect2 Bounds)
|
|
|
|
{
|
2020-11-09 03:42:14 +00:00
|
|
|
ui_widget* Widget = ui_CreateWidget(Interface, Text);
|
|
|
|
ui_WidgetSetFlag(Widget, UIWidgetFlag_Clickable);
|
|
|
|
ui_WidgetSetFlag(Widget, UIWidgetFlag_DrawBackground);
|
2020-11-15 01:18:38 +00:00
|
|
|
ui_WidgetSetFlag(Widget, UIWidgetFlag_DrawString);
|
2020-11-15 07:30:24 +00:00
|
|
|
ui_WidgetSetFlag(Widget, UIWidgetFlag_DrawOutline);
|
2020-11-09 03:42:14 +00:00
|
|
|
ui_eval_result Result = ui_EvaluateWidget(Interface, Widget, Bounds);
|
2020-11-08 06:54:59 +00:00
|
|
|
return ui_EvaluateDropdown(Interface, Widget, Result);
|
|
|
|
}
|
|
|
|
|
|
|
|
internal bool
|
|
|
|
ui_BeginDropdown(ui_interface* Interface, gs_string Text)
|
|
|
|
{
|
2020-11-09 03:42:14 +00:00
|
|
|
ui_widget* Widget = ui_CreateWidget(Interface, Text);
|
|
|
|
ui_WidgetSetFlag(Widget, UIWidgetFlag_Clickable);
|
|
|
|
ui_WidgetSetFlag(Widget, UIWidgetFlag_DrawBackground);
|
2020-11-15 01:18:38 +00:00
|
|
|
ui_WidgetSetFlag(Widget, UIWidgetFlag_DrawString);
|
2020-11-15 07:30:24 +00:00
|
|
|
ui_WidgetSetFlag(Widget, UIWidgetFlag_DrawOutline);
|
2020-11-09 03:42:14 +00:00
|
|
|
ui_eval_result Result = ui_EvaluateWidget(Interface, Widget);
|
2020-11-08 06:54:59 +00:00
|
|
|
return ui_EvaluateDropdown(Interface, Widget, Result);
|
|
|
|
}
|
|
|
|
|
|
|
|
internal void
|
|
|
|
ui_EndDropdown(ui_interface* Interface)
|
|
|
|
{
|
2020-11-09 03:42:14 +00:00
|
|
|
ui_widget* Layout = Interface->ActiveLayout;
|
|
|
|
ui_widget_retained_state* State = ui_GetRetainedState(Interface, Layout->WidgetReference);
|
2020-11-08 06:54:59 +00:00
|
|
|
if (State)
|
|
|
|
{
|
|
|
|
if (State->Value)
|
|
|
|
{
|
|
|
|
ui_PopLayout(Interface);
|
|
|
|
}
|
|
|
|
}
|
2019-07-19 20:56:21 +00:00
|
|
|
}
|
|
|
|
|
2020-11-15 07:30:24 +00:00
|
|
|
internal r32
|
|
|
|
ui_EvaluateRangeSlider(ui_interface* Interface, ui_widget* Widget, ui_eval_result EvalResult, r32 Value, r32 MinValue, r32 MaxValue)
|
|
|
|
{
|
|
|
|
r32 NewValue = Value;
|
|
|
|
|
|
|
|
ui_widget_retained_state* State = ui_GetRetainedState(Interface, Widget->Id);
|
|
|
|
if (!State)
|
|
|
|
{
|
|
|
|
State = ui_CreateRetainedState(Interface, Widget);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (EvalResult.Clicked)
|
|
|
|
{
|
|
|
|
State->InitialValueR32 = Value;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (EvalResult.Held)
|
|
|
|
{
|
|
|
|
r32 Percent = (Interface->Mouse.Pos.x - Widget->Bounds.Min.x) / Rect2Width(Widget->Bounds);
|
|
|
|
NewValue = LerpR32(Percent, MinValue, MaxValue);
|
|
|
|
}
|
|
|
|
|
|
|
|
NewValue = Clamp(MinValue, NewValue, MaxValue);
|
|
|
|
Widget->HorizontalFillPercent = RemapR32(NewValue, MinValue, MaxValue, 0, 1);
|
|
|
|
return NewValue;
|
|
|
|
}
|
|
|
|
|
|
|
|
internal ui_widget*
|
|
|
|
ui_CreateRangeSliderWidget(ui_interface* Interface, gs_string Text, r32 Value)
|
|
|
|
{
|
|
|
|
ui_widget* Widget = ui_CreateWidget(Interface, Text);
|
|
|
|
ui_WidgetSetFlag(Widget, UIWidgetFlag_Clickable);
|
|
|
|
ui_WidgetSetFlag(Widget, UIWidgetFlag_DrawBackground);
|
|
|
|
ui_WidgetSetFlag(Widget, UIWidgetFlag_DrawString);
|
|
|
|
ui_WidgetSetFlag(Widget, UIWidgetFlag_DrawHorizontalFill);
|
|
|
|
ui_WidgetSetFlag(Widget, UIWidgetFlag_DrawOutline);
|
|
|
|
Widget->String = PushStringF(Interface->PerFrameMemory, 128, "%f", Value);
|
|
|
|
return Widget;
|
|
|
|
}
|
|
|
|
|
|
|
|
internal r32
|
|
|
|
ui_RangeSlider(ui_interface* Interface, gs_string Text, r32 Value, r32 ValueMin, r32 ValueMax)
|
|
|
|
{
|
|
|
|
ui_widget* Widget = ui_CreateRangeSliderWidget(Interface, Text, Value);
|
|
|
|
ui_eval_result Result = ui_EvaluateWidget(Interface, Widget);
|
|
|
|
return ui_EvaluateRangeSlider(Interface, Widget, Result, Value, ValueMin, ValueMax);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
internal r32
|
|
|
|
ui_RangeSlider(ui_interface* Interface, gs_string Text, rect2 Bounds, r32 Value, r32 ValueMin, r32 ValueMax)
|
|
|
|
{
|
|
|
|
ui_widget* Widget = ui_CreateRangeSliderWidget(Interface, Text, Value);
|
|
|
|
ui_eval_result Result = ui_EvaluateWidget(Interface, Widget, Bounds);
|
|
|
|
return ui_EvaluateRangeSlider(Interface, Widget, Result, Value, ValueMin, ValueMax);
|
|
|
|
}
|
|
|
|
|
|
|
|
internal bool
|
|
|
|
ui_Toggle(ui_interface* Interface, gs_string Text, bool Value)
|
|
|
|
{
|
|
|
|
ui_widget* Widget = ui_CreateWidget(Interface, Text);
|
|
|
|
ui_WidgetSetFlag(Widget, UIWidgetFlag_Clickable);
|
|
|
|
ui_WidgetSetFlag(Widget, UIWidgetFlag_DrawBackground);
|
|
|
|
ui_WidgetSetFlag(Widget, UIWidgetFlag_DrawHorizontalFill);
|
|
|
|
ui_WidgetSetFlag(Widget, UIWidgetFlag_DrawOutline);
|
|
|
|
ui_eval_result Eval = ui_EvaluateWidget(Interface, Widget);
|
|
|
|
|
|
|
|
bool Result = Eval.Clicked ? !Value : Value;
|
|
|
|
Widget->HorizontalFillPercent = Result ? 1.0f : 0.0f;
|
|
|
|
return Result;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-03-20 07:55:13 +00:00
|
|
|
//
|
|
|
|
// OLD
|
|
|
|
//
|
|
|
|
|
2019-07-19 20:56:21 +00:00
|
|
|
enum selection_state
|
|
|
|
{
|
|
|
|
Selection_None,
|
|
|
|
Selection_Selected,
|
|
|
|
Selection_Deselected,
|
|
|
|
};
|
|
|
|
|
2019-12-29 00:01:34 +00:00
|
|
|
struct interface_list
|
2019-07-19 20:56:21 +00:00
|
|
|
{
|
2020-07-18 19:00:14 +00:00
|
|
|
rect2 ListBounds;
|
2019-07-19 20:56:21 +00:00
|
|
|
|
2019-12-29 00:01:34 +00:00
|
|
|
v2 ListElementDimensions;
|
|
|
|
v2 ElementLabelIndent;
|
2019-07-19 20:56:21 +00:00
|
|
|
|
2019-12-29 00:01:34 +00:00
|
|
|
v4 TextColor;
|
|
|
|
v4* LineBGColors;
|
|
|
|
s32 LineBGColorsCount;
|
|
|
|
v4 LineBGHoverColor;
|
2019-07-19 20:56:21 +00:00
|
|
|
|
2019-12-29 00:01:34 +00:00
|
|
|
s32 ListElementsCount;
|
|
|
|
};
|
|
|
|
|
2020-07-18 19:00:14 +00:00
|
|
|
internal rect2
|
2019-12-29 00:01:34 +00:00
|
|
|
DrawListElementBackground(interface_list* List, mouse_state Mouse, render_command_buffer* RenderBuffer)
|
|
|
|
{
|
2020-07-18 19:00:14 +00:00
|
|
|
rect2 LineBounds = {};
|
2019-12-29 00:01:34 +00:00
|
|
|
LineBounds.Min = v2{
|
|
|
|
List->ListBounds.Min.x,
|
|
|
|
List->ListBounds.Max.y - (List->ListElementDimensions.y * (List->ListElementsCount + 1))
|
|
|
|
};
|
|
|
|
LineBounds.Max = LineBounds.Min + List->ListElementDimensions;
|
2019-07-19 20:56:21 +00:00
|
|
|
|
2019-12-29 00:01:34 +00:00
|
|
|
v4 Color = List->LineBGColors[List->ListElementsCount % List->LineBGColorsCount];
|
2020-07-18 19:00:14 +00:00
|
|
|
if (PointIsInRect(LineBounds, Mouse.Pos))
|
2019-07-19 20:56:21 +00:00
|
|
|
{
|
2019-12-29 00:01:34 +00:00
|
|
|
Color = List->LineBGHoverColor;
|
2019-07-19 20:56:21 +00:00
|
|
|
}
|
|
|
|
|
2019-12-29 00:01:34 +00:00
|
|
|
PushRenderQuad2D(RenderBuffer, LineBounds.Min, LineBounds.Max, Color);
|
|
|
|
return LineBounds;
|
2019-07-19 20:56:21 +00:00
|
|
|
}
|
|
|
|
|
2020-07-18 19:00:14 +00:00
|
|
|
internal rect2
|
|
|
|
DrawListElement(gs_string Label, interface_list* List, mouse_state Mouse, render_command_buffer* RenderBuffer, interface_config Interface)
|
2019-07-19 20:56:21 +00:00
|
|
|
{
|
2020-07-18 19:00:14 +00:00
|
|
|
rect2 Bounds = DrawListElementBackground(List, Mouse, RenderBuffer);
|
2019-07-19 20:56:21 +00:00
|
|
|
|
2019-12-29 00:01:34 +00:00
|
|
|
v2 LabelPosition = Bounds.Min + List->ElementLabelIndent;
|
|
|
|
DrawString(RenderBuffer, Label, Interface.Font, LabelPosition, List->TextColor);
|
2019-07-19 20:56:21 +00:00
|
|
|
|
2019-12-29 00:01:34 +00:00
|
|
|
List->ListElementsCount++;
|
|
|
|
return Bounds;
|
2019-07-19 20:56:21 +00:00
|
|
|
}
|
|
|
|
|
2019-12-29 00:01:34 +00:00
|
|
|
|
2019-07-19 20:56:21 +00:00
|
|
|
internal r32
|
2019-11-01 14:38:44 +00:00
|
|
|
EvaluateColorChannelSlider (render_command_buffer* RenderBuffer, v4 ChannelMask, v2 Min, v2 Max, r32 Current, mouse_state Mouse)
|
2019-07-19 20:56:21 +00:00
|
|
|
{
|
|
|
|
r32 Result = Current;
|
|
|
|
|
2020-07-18 19:00:14 +00:00
|
|
|
// TODO(Peter): Can this come from outside the function? Would rather pass rect around than min/max
|
|
|
|
rect2 Rect = rect2{ Min, Max };
|
|
|
|
|
2019-07-19 20:56:21 +00:00
|
|
|
render_quad_batch_constructor Batch = PushRenderQuad2DBatch(RenderBuffer, 2);
|
|
|
|
|
|
|
|
v4 LeftColor = ChannelMask * 0;
|
|
|
|
LeftColor.a = 1.f;
|
|
|
|
v4 RightColor = ChannelMask;
|
2020-03-22 04:13:35 +00:00
|
|
|
PushQuad2DOnBatch(&Batch,
|
2020-07-18 19:00:14 +00:00
|
|
|
RectBottomLeft(Rect), RectBottomRight(Rect),
|
|
|
|
RectTopRight(Rect), RectTopLeft(Rect),
|
2019-07-19 20:56:21 +00:00
|
|
|
v2{0, 0}, v2{1, 0}, v2{1, 1}, v2{0, 1},
|
|
|
|
LeftColor, RightColor, RightColor, LeftColor);
|
|
|
|
|
2019-11-11 20:02:24 +00:00
|
|
|
if (MouseButtonTransitionedDown(Mouse.LeftButtonState))
|
2019-07-19 20:56:21 +00:00
|
|
|
{
|
2020-07-18 19:00:14 +00:00
|
|
|
if (PointIsInRect(Rect, Mouse.DownPos))
|
2019-07-19 20:56:21 +00:00
|
|
|
{
|
2019-10-30 14:28:02 +00:00
|
|
|
Result = ((r32)Mouse.Pos.x - Min.x) / (Max.x - Min.x);
|
2020-07-18 19:00:14 +00:00
|
|
|
Result = Clamp01(Result);
|
2019-07-19 20:56:21 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
r32 DragBarWidth = 8;
|
2020-07-18 19:00:14 +00:00
|
|
|
v2 DragBarMin = v2{
|
|
|
|
LerpR32(Result, Min.x, Max.x) - (DragBarWidth / 2),
|
|
|
|
Min.y - 2
|
|
|
|
};
|
2019-07-19 20:56:21 +00:00
|
|
|
v2 DragBarMax = DragBarMin + v2{DragBarWidth, (Max.y - Min.y) + 4};
|
|
|
|
|
|
|
|
PushQuad2DOnBatch(&Batch, DragBarMin, DragBarMax, v4{.3f, .3f, .3f, 1.f});
|
|
|
|
|
|
|
|
return Result;
|
|
|
|
}
|
|
|
|
|
|
|
|
internal b32
|
2019-11-01 14:38:44 +00:00
|
|
|
EvaluateColorPicker (render_command_buffer* RenderBuffer, v4* Value, v2 PanelMin, interface_config Config, mouse_state Mouse)
|
2019-07-19 20:56:21 +00:00
|
|
|
{
|
|
|
|
b32 ShouldClose = false;
|
|
|
|
|
|
|
|
v2 PanelMax = v2{400, 500};
|
2020-07-18 19:00:14 +00:00
|
|
|
// TODO(Peter): Can this get passed from outside? rather pass rect2 than min/max pairs
|
|
|
|
rect2 PanelRect = rect2{PanelMin, PanelMax};
|
|
|
|
if (MouseButtonTransitionedDown(Mouse.LeftButtonState) && !PointIsInRect(PanelRect, Mouse.Pos))
|
2019-07-19 20:56:21 +00:00
|
|
|
{
|
|
|
|
ShouldClose = true;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2020-07-18 19:00:14 +00:00
|
|
|
PushRenderQuad2D(RenderBuffer, PanelRect.Min, PanelRect.Max, v4{.5f, .5f, .5f, 1.f});
|
2019-07-19 20:56:21 +00:00
|
|
|
|
2020-07-18 19:00:14 +00:00
|
|
|
v2 TitleMin = v2{PanelRect.Min.x + 5, PanelRect.Max.y - (Config.Font->PixelHeight + 5)};
|
|
|
|
DrawString(RenderBuffer, MakeString("Color Picker"), Config.Font,
|
2019-07-19 20:56:21 +00:00
|
|
|
TitleMin, WhiteV4);
|
|
|
|
|
|
|
|
v2 SliderDim = v2{(PanelMax.x - PanelMin.x) - 20, 32};
|
|
|
|
// channel sliders
|
|
|
|
v2 SliderMin = TitleMin - v2{0, SliderDim.y + 10};
|
2019-10-30 14:28:02 +00:00
|
|
|
Value->r = EvaluateColorChannelSlider(RenderBuffer, RedV4, SliderMin, SliderMin + SliderDim, Value->r, Mouse);
|
2019-07-19 20:56:21 +00:00
|
|
|
SliderMin.y -= SliderDim.y + 10;
|
2019-10-30 14:28:02 +00:00
|
|
|
Value->g = EvaluateColorChannelSlider(RenderBuffer, GreenV4, SliderMin, SliderMin + SliderDim, Value->g, Mouse);
|
2019-07-19 20:56:21 +00:00
|
|
|
SliderMin.y -= SliderDim.y + 10;
|
2019-10-30 14:28:02 +00:00
|
|
|
Value->b = EvaluateColorChannelSlider(RenderBuffer, BlueV4, SliderMin, SliderMin + SliderDim, Value->b, Mouse);
|
2019-07-19 20:56:21 +00:00
|
|
|
SliderMin.y -= SliderDim.y + 10;
|
2019-10-30 14:28:02 +00:00
|
|
|
Value->a = EvaluateColorChannelSlider(RenderBuffer, WhiteV4, SliderMin, SliderMin + SliderDim, Value->a, Mouse);
|
2019-07-19 20:56:21 +00:00
|
|
|
|
|
|
|
// Output Color Display
|
|
|
|
SliderMin.y -= 100;
|
|
|
|
PushRenderQuad2D(RenderBuffer, SliderMin, SliderMin + v2{75, 75}, *Value);
|
|
|
|
}
|
|
|
|
|
|
|
|
return ShouldClose;
|
2019-08-18 12:56:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
struct search_lister_result
|
|
|
|
{
|
|
|
|
s32 HotItem;
|
2019-09-02 06:03:38 +00:00
|
|
|
s32 SelectedItem;
|
2019-08-18 12:56:18 +00:00
|
|
|
b32 ShouldRemainOpen;
|
|
|
|
};
|
|
|
|
|
2020-07-18 19:00:14 +00:00
|
|
|
typedef gs_string search_lister_get_list_item_at_offset(u8* ListMemory, s32 ListLength, gs_string Searchgs_string, s32 Offset);
|
2019-08-18 12:56:18 +00:00
|
|
|
|
|
|
|
internal search_lister_result
|
2020-07-18 19:00:14 +00:00
|
|
|
EvaluateSearchLister (ui_interface* Interface, v2 TopLeft, v2 Dimension, gs_string Title,
|
|
|
|
gs_string* ItemList, s32* ListLUT, s32 ListLength,
|
2019-11-01 11:17:46 +00:00
|
|
|
s32 HotItem,
|
2020-07-18 19:00:14 +00:00
|
|
|
gs_string* Searchgs_string, s32 Searchgs_stringCursorPosition)
|
2019-08-18 12:56:18 +00:00
|
|
|
{
|
|
|
|
search_lister_result Result = {};
|
|
|
|
Result.ShouldRemainOpen = true;
|
|
|
|
Result.HotItem = HotItem;
|
|
|
|
|
2020-03-20 07:55:13 +00:00
|
|
|
// TODO(Peter): Was tired. Nothing wrong with the code below
|
|
|
|
InvalidCodePath;
|
|
|
|
#if 0
|
2019-08-18 12:56:18 +00:00
|
|
|
// Title Bar
|
2020-07-18 19:00:14 +00:00
|
|
|
rect2 TitleBarBounds = rect2{v2{TopLeft.x, TopLeft.y - 30}, v2{TopLeft.x + 300, TopLeft.y}};
|
2020-03-20 07:55:13 +00:00
|
|
|
ui_FillRect(Interface, TitleBarBounds, v4{.3f, .3f, .3f, 1.f});
|
2020-07-18 19:00:14 +00:00
|
|
|
ui_Drawgs_string(Interface, Title, TitleBarBounds, Interface->Style.TextColor);
|
2019-11-01 11:17:46 +00:00
|
|
|
|
2020-07-18 19:00:14 +00:00
|
|
|
MakeStringBuffer(Debuggs_string, 256);
|
|
|
|
PrintF(&Debuggs_string, "Hot Item: %d | Filtered Items: %d", HotItem, ListLength);
|
|
|
|
rect2 DebugBounds = MakeRectMinWidth(v2{ TopLeft.x + 256, TopLeft.y - 25}, v2{256, Interface->Style.LineHeight});
|
|
|
|
ui_Drawgs_string(Interface, Debuggs_string, DebugBounds, Interface->Style.TextColor);
|
2019-08-18 12:56:18 +00:00
|
|
|
|
|
|
|
// Search Bar
|
|
|
|
PushRenderQuad2D(RenderBuffer, v2{TopLeft.x, TopLeft.y - 30}, v2{TopLeft.x + 300, TopLeft.y}, v4{.3f, .3f, .3f, 1.f});
|
2020-07-18 19:00:14 +00:00
|
|
|
Drawgs_stringWithCursor(RenderBuffer, *Searchgs_string, Searchgs_stringCursorPosition, Font, v2{TopLeft.x, TopLeft.y - 25}, WhiteV4, GreenV4);
|
2019-08-18 12:56:18 +00:00
|
|
|
TopLeft.y -= 30;
|
|
|
|
|
2019-11-01 11:17:46 +00:00
|
|
|
for (s32 i = 0; i < ListLength; i++)
|
2019-08-18 12:56:18 +00:00
|
|
|
{
|
2019-11-01 11:17:46 +00:00
|
|
|
s32 FilteredIndex = ListLUT[i];
|
2020-07-18 19:00:14 +00:00
|
|
|
gs_string ListItemgs_string = ItemList[FilteredIndex];
|
2019-11-01 11:17:46 +00:00
|
|
|
|
2019-10-30 16:10:15 +00:00
|
|
|
v2 Min = v2{TopLeft.x, TopLeft.y - 30};
|
|
|
|
v2 Max = Min + Dimension - v2{0, Config.Margin.y};
|
2019-08-18 12:56:18 +00:00
|
|
|
|
2019-10-30 16:10:15 +00:00
|
|
|
v4 ButtonColor = Config.ButtonColor_Inactive;
|
2019-11-01 11:17:46 +00:00
|
|
|
if (i == HotItem)
|
2019-08-18 12:56:18 +00:00
|
|
|
{
|
2019-10-30 16:10:15 +00:00
|
|
|
ButtonColor = Config.ButtonColor_Active;
|
|
|
|
}
|
|
|
|
|
2020-07-18 19:00:14 +00:00
|
|
|
if (ui_Button(Interface, ListItemgs_string, rect2{Min, Max}))
|
2019-10-30 16:10:15 +00:00
|
|
|
{
|
2019-11-01 11:17:46 +00:00
|
|
|
Result.SelectedItem = i;
|
2019-08-18 12:56:18 +00:00
|
|
|
}
|
2019-10-30 16:10:15 +00:00
|
|
|
|
|
|
|
TopLeft.y -= 30;
|
2019-08-18 12:56:18 +00:00
|
|
|
}
|
2020-03-20 07:55:13 +00:00
|
|
|
#endif
|
2019-08-18 12:56:18 +00:00
|
|
|
|
|
|
|
return Result;
|
2020-01-02 02:41:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#define INTERFACE_H
|
|
|
|
#endif // INTERFACE_H
|