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-07-18 19:00:14 +00:00
|
|
|
Drawgs_stringWithCursor (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;
|
|
|
|
}
|
|
|
|
|
|
|
|
struct interface_config
|
|
|
|
{
|
|
|
|
v4 PanelBGColors[4];
|
2020-03-20 07:55:13 +00:00
|
|
|
|
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-03-20 07:55:13 +00:00
|
|
|
struct ui_layout
|
2019-07-19 20:56:21 +00:00
|
|
|
{
|
2020-07-18 19:00:14 +00:00
|
|
|
rect2 Bounds;
|
2020-03-20 07:55:13 +00:00
|
|
|
v2 Margin;
|
|
|
|
r32 RowHeight;
|
|
|
|
r32 RowYAt;
|
|
|
|
|
|
|
|
b32 DrawHorizontal;
|
2020-03-22 05:44:44 +00:00
|
|
|
u32 ColumnsMax;
|
|
|
|
r32* ColumnWidths;
|
|
|
|
u32 ColumnsCount;
|
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;
|
|
|
|
};
|
|
|
|
|
|
|
|
static ui_layout
|
2020-07-18 19:00:14 +00:00
|
|
|
ui_CreateLayout(ui_interface Interface, rect2 Bounds)
|
2020-03-20 07:55:13 +00:00
|
|
|
{
|
|
|
|
ui_layout Result = {0};
|
|
|
|
Result.Bounds = Bounds;
|
|
|
|
Result.Margin = Interface.Style.Margin;
|
|
|
|
Result.RowHeight = Interface.Style.RowHeight;
|
|
|
|
Result.RowYAt = Bounds.Max.y - Result.RowHeight;
|
|
|
|
return Result;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2020-03-22 05:44:44 +00:00
|
|
|
ui_StartRow(ui_layout* Layout, u32 ColumnsMax)
|
2020-03-20 07:55:13 +00:00
|
|
|
{
|
|
|
|
Layout->DrawHorizontal = true;
|
2020-03-22 05:44:44 +00:00
|
|
|
Layout->ColumnsMax = ColumnsMax;
|
|
|
|
Layout->ColumnWidths = 0;
|
|
|
|
Layout->ColumnsCount = 0;
|
2020-03-20 07:55:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
ui_StartRow(ui_layout* Layout)
|
|
|
|
{
|
|
|
|
ui_StartRow(Layout, 0);
|
|
|
|
}
|
|
|
|
|
2020-03-22 05:44:44 +00:00
|
|
|
static void
|
|
|
|
ui_StartRow(ui_layout* Layout, u32 ColumnsMax, r32* ColumnWidths)
|
|
|
|
{
|
|
|
|
Layout->DrawHorizontal = true;
|
|
|
|
Layout->ColumnsMax = ColumnsMax;
|
|
|
|
Layout->ColumnWidths = ColumnWidths;
|
|
|
|
Layout->ColumnsCount = 0;
|
|
|
|
}
|
|
|
|
|
2020-03-20 07:55:13 +00:00
|
|
|
static void
|
|
|
|
ui_EndRow(ui_layout* Layout)
|
|
|
|
{
|
|
|
|
Layout->DrawHorizontal = false;
|
2020-03-22 05:44:44 +00:00
|
|
|
Layout->ColumnWidths = 0;
|
|
|
|
Layout->RowYAt -= Layout->RowHeight;
|
2020-03-20 07:55:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static b32
|
2020-07-18 19:00:14 +00:00
|
|
|
ui_TryReserveElementBounds(ui_layout* Layout, rect2* Bounds)
|
2020-03-20 07:55:13 +00:00
|
|
|
{
|
|
|
|
b32 Result = true;
|
|
|
|
if (!Layout->DrawHorizontal)
|
2019-07-19 20:56:21 +00:00
|
|
|
{
|
2020-03-20 07:55:13 +00:00
|
|
|
Bounds->Min = { Layout->Bounds.Min.x, Layout->RowYAt };
|
|
|
|
Bounds->Max = { Layout->Bounds.Max.x, Bounds->Min.y + Layout->RowHeight };
|
|
|
|
Layout->RowYAt -= Layout->RowHeight;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2020-03-22 05:44:44 +00:00
|
|
|
if (Layout->ColumnsMax > 0)
|
2019-07-19 20:56:21 +00:00
|
|
|
{
|
2020-03-22 05:44:44 +00:00
|
|
|
Assert(Layout->ColumnsCount < Layout->ColumnsMax);
|
|
|
|
if (Layout->ColumnWidths != 0)
|
|
|
|
{
|
|
|
|
v2 Min = { Layout->Bounds.Min.x, Layout->RowYAt };
|
|
|
|
for (u32 i = 0; i < Layout->ColumnsCount; i++)
|
|
|
|
{
|
|
|
|
Min.x += Layout->ColumnWidths[i];
|
|
|
|
}
|
|
|
|
Bounds->Min = Min;
|
|
|
|
Bounds->Max = Bounds->Min + v2{ Layout->ColumnWidths[Layout->ColumnsCount], Layout->RowHeight };
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2020-07-18 19:00:14 +00:00
|
|
|
r32 ElementWidth = Rect2Width(Layout->Bounds) / Layout->ColumnsMax;
|
2020-03-22 05:44:44 +00:00
|
|
|
Bounds->Min = {
|
|
|
|
Layout->Bounds.Min.x + (ElementWidth * Layout->ColumnsCount) + Layout->Margin.x,
|
|
|
|
Layout->RowYAt
|
|
|
|
};
|
|
|
|
Bounds->Max = {
|
|
|
|
Bounds->Min.x + ElementWidth - Layout->Margin.x,
|
|
|
|
Bounds->Min.y + Layout->RowHeight
|
|
|
|
};
|
|
|
|
}
|
|
|
|
Layout->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
|
|
|
|
ui_ReserveTextLineBounds(ui_interface Interface, gs_string Text, ui_layout* Layout)
|
2019-07-19 20:56:21 +00:00
|
|
|
{
|
2020-07-18 19:00:14 +00:00
|
|
|
rect2 Bounds = {0};
|
2020-03-20 07:55:13 +00:00
|
|
|
|
|
|
|
return Bounds;
|
|
|
|
}
|
|
|
|
|
2020-07-18 19:00:14 +00:00
|
|
|
static rect2
|
2020-03-22 04:13:35 +00:00
|
|
|
ui_ReserveElementBounds(ui_layout* Layout)
|
|
|
|
{
|
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-03-22 05:44:44 +00:00
|
|
|
ui_LayoutRemaining(ui_layout Layout)
|
|
|
|
{
|
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-06-22 04:59:42 +00:00
|
|
|
|
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-07-18 19:00:14 +00:00
|
|
|
ui_DrawString(ui_interface* Interface, gs_string String, rect2 Bounds, v4 Color, gs_string_alignment Alignment = Align_Left)
|
2020-03-20 07:55:13 +00:00
|
|
|
{
|
|
|
|
DEBUG_TRACK_FUNCTION;
|
2020-03-22 04:13:35 +00:00
|
|
|
render_quad_batch_constructor BatchConstructor = PushRenderTexture2DBatch(Interface->RenderBuffer,
|
2020-03-20 07:55:13 +00:00
|
|
|
String.Length,
|
|
|
|
Interface->Style.Font->BitmapMemory,
|
|
|
|
Interface->Style.Font->BitmapTextureHandle,
|
|
|
|
Interface->Style.Font->BitmapWidth,
|
|
|
|
Interface->Style.Font->BitmapHeight,
|
|
|
|
Interface->Style.Font->BitmapBytesPerPixel,
|
|
|
|
Interface->Style.Font->BitmapStride);
|
|
|
|
|
|
|
|
v2 RegisterPosition = Bounds.Min + Interface->Style.Margin;
|
|
|
|
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, Interface->Style.Font, Color);
|
|
|
|
}
|
|
|
|
else if (Alignment == Align_Right)
|
|
|
|
{
|
|
|
|
RegisterPosition = DrawStringRightAligned(&BatchConstructor, StringExpand(String), RegisterPosition, Interface->Style.Font, Color);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
InvalidCodePath;
|
2019-07-19 20:56:21 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-03-22 04:13:35 +00:00
|
|
|
static void
|
2020-07-18 19:00:14 +00:00
|
|
|
ui_LayoutDrawString(ui_interface* Interface, ui_layout* Layout, gs_string String, v4 Color, gs_string_alignment Alignment = Align_Left)
|
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))
|
|
|
|
{
|
|
|
|
// TODO(NAME): Not invalid, just haven't implemented yet.
|
|
|
|
// This is the case where Layout is in row mode without a fixed number of elements
|
|
|
|
InvalidCodePath;
|
|
|
|
}
|
|
|
|
ui_DrawString(Interface, String, Bounds, Color, Alignment);
|
|
|
|
}
|
|
|
|
|
2020-03-20 07:55:13 +00:00
|
|
|
static void
|
2020-07-18 19:00:14 +00:00
|
|
|
ui_TextBox(ui_interface* Interface, rect2 Bounds, gs_string Text, v4 BGColor, v4 TextColor)
|
2019-07-19 20:56:21 +00:00
|
|
|
{
|
2020-03-20 07:55:13 +00:00
|
|
|
ui_FillRect(Interface, Bounds, BGColor);
|
|
|
|
ui_DrawString(Interface, Text, Bounds, TextColor);
|
|
|
|
}
|
2019-07-19 20:56:21 +00:00
|
|
|
|
2020-03-20 07:55:13 +00:00
|
|
|
static b32
|
2020-07-18 19:00:14 +00:00
|
|
|
ui_Button(ui_interface* Interface, gs_string Text, rect2 Bounds, v4 InactiveColor, v4 HoverColor, v4 ClickedColor)
|
2019-07-19 20:56:21 +00:00
|
|
|
{
|
2020-03-20 07:55:13 +00:00
|
|
|
b32 Pressed = false;
|
|
|
|
v4 ButtonBG = InactiveColor;
|
2020-07-18 19:00:14 +00:00
|
|
|
if (PointIsInRect(Bounds, Interface->Mouse.Pos))
|
2019-07-19 20:56:21 +00:00
|
|
|
{
|
2020-03-20 07:55:13 +00:00
|
|
|
ButtonBG = HoverColor;
|
|
|
|
if (MouseButtonTransitionedDown(Interface->Mouse.LeftButtonState))
|
2019-07-19 20:56:21 +00:00
|
|
|
{
|
2020-03-20 07:55:13 +00:00
|
|
|
ButtonBG = ClickedColor;
|
|
|
|
Pressed = true;
|
2019-07-19 20:56:21 +00:00
|
|
|
}
|
|
|
|
}
|
2020-03-20 07:55:13 +00:00
|
|
|
ui_TextBox(Interface, Bounds, Text, ButtonBG, Interface->Style.TextColor);
|
|
|
|
return Pressed;
|
2019-07-19 20:56:21 +00:00
|
|
|
}
|
|
|
|
|
2020-03-20 07:55:13 +00:00
|
|
|
static 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-03-20 07:55:13 +00:00
|
|
|
v4 BGColor = Interface->Style.ButtonColor_Inactive;
|
|
|
|
v4 HoverColor = Interface->Style.ButtonColor_Active;
|
|
|
|
v4 SelectedColor = Interface->Style.ButtonColor_Selected;
|
|
|
|
return ui_Button(Interface, Text, Bounds, BGColor, HoverColor, SelectedColor);
|
|
|
|
}
|
|
|
|
|
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
|
|
|
{
|
|
|
|
list_item_colors Colors = ui_GetListItemColors(Interface, ListItemIndex);
|
|
|
|
return ui_Button(Interface, Text, Bounds, Colors.Hover, Colors.Selected, Colors.BGColor);
|
|
|
|
}
|
|
|
|
|
2020-03-20 07:55:13 +00:00
|
|
|
static b32
|
2020-07-18 19:00:14 +00:00
|
|
|
ui_LayoutButton(ui_interface* Interface, ui_layout* Layout, gs_string Text, v4 BGColor, v4 HoverColor, v4 SelectColor)
|
2020-03-20 07:55:13 +00:00
|
|
|
{
|
2020-07-18 19:00:14 +00:00
|
|
|
rect2 ButtonBounds = {0};
|
2020-03-20 07:55:13 +00:00
|
|
|
if (!ui_TryReserveElementBounds(Layout, &ButtonBounds))
|
2019-07-19 20:56:21 +00:00
|
|
|
{
|
2020-03-20 07:55:13 +00:00
|
|
|
ButtonBounds = ui_ReserveTextLineBounds(*Interface, Text, Layout);
|
2019-07-19 20:56:21 +00:00
|
|
|
}
|
2020-03-22 04:13:35 +00:00
|
|
|
return ui_Button(Interface, Text, ButtonBounds, BGColor, HoverColor, SelectColor);
|
|
|
|
}
|
|
|
|
|
|
|
|
static b32
|
2020-07-18 19:00:14 +00:00
|
|
|
ui_LayoutButton(ui_interface* Interface, ui_layout* Layout, gs_string Text)
|
2020-03-22 04:13:35 +00:00
|
|
|
{
|
2020-03-20 07:55:13 +00:00
|
|
|
v4 BGColor = Interface->Style.ButtonColor_Inactive;
|
|
|
|
v4 HoverColor = Interface->Style.ButtonColor_Active;
|
|
|
|
v4 SelectedColor = Interface->Style.ButtonColor_Selected;
|
2020-03-22 04:13:35 +00:00
|
|
|
return ui_LayoutButton(Interface, Layout, Text, BGColor, HoverColor, SelectedColor);
|
|
|
|
}
|
|
|
|
|
2020-06-22 04:59:42 +00:00
|
|
|
static b32
|
2020-07-18 19:00:14 +00:00
|
|
|
ui_LayoutListButton(ui_interface* Interface, ui_layout* Layout, gs_string Text, u32 ListItemIndex)
|
2020-03-22 04:13:35 +00:00
|
|
|
{
|
2020-06-22 04:59:42 +00:00
|
|
|
list_item_colors Colors = ui_GetListItemColors(Interface, ListItemIndex);
|
|
|
|
return ui_LayoutButton(Interface, Layout, Text, Colors.Hover, Colors.Selected, Colors.BGColor);
|
2020-03-20 07:55:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static b32
|
2020-07-18 19:00:14 +00:00
|
|
|
ui_LayoutListEntry(ui_interface* Interface, ui_layout* Layout, gs_string Text, u32 Index)
|
2020-03-20 07:55:13 +00:00
|
|
|
{
|
2020-07-18 19:00:14 +00:00
|
|
|
rect2 Bounds = {0};
|
2020-03-20 07:55:13 +00:00
|
|
|
if (!ui_TryReserveElementBounds(Layout, &Bounds))
|
2019-07-19 20:56:21 +00:00
|
|
|
{
|
2020-03-20 07:55:13 +00:00
|
|
|
// TODO(Peter): this isn't really invalid, but I don't have a concrete use case
|
|
|
|
// for it yet. This should only fire if the Layout component is drawing a row,
|
|
|
|
// but if you're in row mode during a list, what should happen?
|
|
|
|
// Punting this till I have a use case
|
|
|
|
InvalidCodePath;
|
2019-07-19 20:56:21 +00:00
|
|
|
}
|
2020-03-22 04:13:35 +00:00
|
|
|
v4 BGColor = ui_GetListItemBGColor(Interface->Style, Index);
|
2020-03-20 07:55:13 +00:00
|
|
|
v4 HoverColor = Interface->Style.ListBGHover;
|
|
|
|
v4 SelectedColor = Interface->Style.ListBGSelected;
|
|
|
|
return ui_Button(Interface, Text, Bounds, BGColor, HoverColor, SelectedColor);
|
2019-07-19 20:56:21 +00:00
|
|
|
}
|
|
|
|
|
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
|