diff --git a/meson.build b/meson.build index fcd14a8f..ec9be072 100644 --- a/meson.build +++ b/meson.build @@ -163,15 +163,12 @@ dll_sources = [ 'src/game/text.c', 'src/game/ui/common.c', 'src/game/ui/controllers/controls.c', - 'src/game/ui/events.c', - 'src/game/ui/widgets/console.c', 'src/game/ui/widgets/controls_column.c', 'src/game/ui/widgets/controls_dialog.c', 'src/game/ui/widgets/controls_input_selector.c', 'src/game/ui/widgets/controls_layout_selector.c', 'src/game/ui/widgets/label.c', 'src/game/ui/widgets/prompt.c', - 'src/game/ui/widgets/stack.c', 'src/game/ui/widgets/window.c', 'src/global/enum_str.c', 'src/global/vars.c', diff --git a/src/decomp/decomp.c b/src/decomp/decomp.c index 304fb9cc..5cb182e8 100644 --- a/src/decomp/decomp.c +++ b/src/decomp/decomp.c @@ -22,7 +22,6 @@ #include "game/shell.h" #include "game/sound.h" #include "game/text.h" -#include "game/ui/common.h" #include "global/const.h" #include "global/funcs.h" #include "global/vars.h" @@ -30,6 +29,7 @@ #include "lib/dinput.h" #include "specific/s_flagged_string.h" +#include #include #include @@ -757,7 +757,8 @@ bool __cdecl WinVidSpinMessageLoop(bool need_wait) UI_HandleKeyUp(msg.wParam); return 0; } else if (msg.message == WM_CHAR) { - UI_HandleChar(msg.wParam); + char insert_string[2] = { msg.wParam, '\0' }; + UI_HandleTextEdit(insert_string); return 0; } } diff --git a/src/game/console/common.c b/src/game/console/common.c index 5ebce06d..edb50a71 100644 --- a/src/game/console/common.c +++ b/src/game/console/common.c @@ -1,60 +1,13 @@ #include "game/console/common.h" #include "game/console/setup.h" -#include "game/input.h" #include "game/text.h" -#include "game/ui/widgets/console.h" - -#include - -static bool m_IsOpened = false; -static UI_WIDGET *m_Console; - -void Console_Init(void) -{ - m_Console = UI_Console_Create(); -} - -void Console_Shutdown(void) -{ - if (m_Console != NULL) { - m_Console->free(m_Console); - m_Console = NULL; - } - - m_IsOpened = false; -} - -void Console_Open(void) -{ - if (m_IsOpened) { - UI_Console_HandleClose(m_Console); - } - m_IsOpened = true; - UI_Console_HandleOpen(m_Console); -} - -void Console_Close(void) -{ - UI_Console_HandleClose(m_Console); - m_IsOpened = false; -} - -bool Console_IsOpened(void) -{ - return m_IsOpened; -} int32_t Console_GetMaxLineLength(void) { return TEXT_MAX_STRING_SIZE - 1; } -void Console_LogImpl(const char *const text) -{ - UI_Console_HandleLog(m_Console, text); -} - CONSOLE_COMMAND **Console_GetCommands(void) { return g_ConsoleCommands; @@ -62,7 +15,7 @@ CONSOLE_COMMAND **Console_GetCommands(void) void Console_Draw(void) { - UI_Console_ScrollLogs(m_Console); + Console_ScrollLogs(); #if 0 // TODO: draw screen quad diff --git a/src/game/console/common.h b/src/game/console/common.h index 5c76f62a..ff45ef5a 100644 --- a/src/game/console/common.h +++ b/src/game/console/common.h @@ -1,15 +1,5 @@ #pragma once -#include -#include +#include -void Console_Init(void); -void Console_Shutdown(void); void Console_Draw(void); - -void Console_Open(void); -void Console_Close(void); -bool Console_IsOpened(void); - -void Console_Log(const char *fmt, ...); -void Console_ScrollLogs(void); diff --git a/src/game/shell.c b/src/game/shell.c index 50db5102..c6b67f6b 100644 --- a/src/game/shell.c +++ b/src/game/shell.c @@ -10,10 +10,10 @@ #include "game/input.h" #include "game/music.h" #include "game/sound.h" -#include "game/ui/common.h" #include "global/funcs.h" #include "global/vars.h" +#include #include #include diff --git a/src/game/ui/common.c b/src/game/ui/common.c index c87fb17b..2e2b3a06 100644 --- a/src/game/ui/common.c +++ b/src/game/ui/common.c @@ -1,43 +1,19 @@ -#include "game/ui/common.h" +#include -#include +#include -void UI_Init(void) +UI_INPUT UI_TranslateInput(uint32_t system_keycode) { - UI_Events_Init(); -} - -void UI_Shutdown(void) -{ - UI_Events_Shutdown(); -} - -void UI_HandleKeyDown(const uint32_t key) -{ - const UI_EVENT event = { - .name = "key_down", - .sender = NULL, - .data = (void *)(uintptr_t)key, - }; - UI_Events_Fire(&event); -} - -void UI_HandleKeyUp(const uint32_t key) -{ - const UI_EVENT event = { - .name = "key_up", - .sender = NULL, - .data = (void *)(uintptr_t)key, - }; - UI_Events_Fire(&event); -} - -void UI_HandleChar(const uint32_t char_) -{ - const UI_EVENT event = { - .name = "char", - .sender = NULL, - .data = (void *)(uintptr_t)char_, - }; - UI_Events_Fire(&event); + // clang-format off + switch (system_keycode) { + case VK_LEFT: return UI_KEY_LEFT; + case VK_RIGHT: return UI_KEY_RIGHT; + case VK_HOME: return UI_KEY_HOME; + case VK_END: return UI_KEY_END; + case VK_BACK: return UI_KEY_BACK; + case VK_RETURN: return UI_KEY_RETURN; + case VK_ESCAPE: return UI_KEY_ESCAPE; + } + // clang-format on + return -1; } diff --git a/src/game/ui/common.h b/src/game/ui/common.h deleted file mode 100644 index d89d2504..00000000 --- a/src/game/ui/common.h +++ /dev/null @@ -1,10 +0,0 @@ -#pragma once - -#include "game/ui/events.h" - -void UI_Init(void); -void UI_Shutdown(void); - -void UI_HandleKeyDown(uint32_t key); -void UI_HandleKeyUp(uint32_t key); -void UI_HandleChar(uint32_t char_); diff --git a/src/game/ui/events.c b/src/game/ui/events.c deleted file mode 100644 index 7c5cb0ca..00000000 --- a/src/game/ui/events.c +++ /dev/null @@ -1,65 +0,0 @@ -#include "game/ui/events.h" - -#include -#include - -#include -#include - -typedef struct { - int32_t listener_id; - const char *event_name; - const UI_WIDGET *sender; - UI_EVENT_LISTENER listener; - void *user_data; -} M_LISTENER; - -static VECTOR *m_Listeners = NULL; -static int32_t m_ListenerID = 0; - -void UI_Events_Init(void) -{ - m_Listeners = Vector_Create(sizeof(M_LISTENER)); -} - -void UI_Events_Shutdown(void) -{ - Vector_Free(m_Listeners); -} - -int32_t UI_Events_Subscribe( - const char *const event_name, const UI_WIDGET *const sender, - const UI_EVENT_LISTENER listener, void *const user_data) -{ - M_LISTENER entry = { - .listener_id = m_ListenerID++, - .event_name = event_name, - .sender = sender, - .listener = listener, - .user_data = user_data, - }; - Vector_Add(m_Listeners, &entry); - return entry.listener_id; -} - -void UI_Events_Unsubscribe(const int32_t listener_id) -{ - for (int32_t i = 0; i < m_Listeners->count; i++) { - M_LISTENER entry = *(M_LISTENER *)Vector_Get(m_Listeners, i); - if (entry.listener_id == listener_id) { - Vector_RemoveAt(m_Listeners, i); - return; - } - } -} - -void UI_Events_Fire(const UI_EVENT *const event) -{ - for (int32_t i = 0; i < m_Listeners->count; i++) { - M_LISTENER entry = *(M_LISTENER *)Vector_Get(m_Listeners, i); - if (strcmp(entry.event_name, event->name) == 0 - && entry.sender == event->sender) { - entry.listener(event, entry.user_data); - } - } -} diff --git a/src/game/ui/events.h b/src/game/ui/events.h deleted file mode 100644 index 0686fa64..00000000 --- a/src/game/ui/events.h +++ /dev/null @@ -1,22 +0,0 @@ -#pragma once - -#include "game/ui/widgets/base.h" - -typedef struct { - const char *name; - const UI_WIDGET *sender; - void *data; -} UI_EVENT; - -typedef void (*UI_EVENT_LISTENER)(const UI_EVENT *, void *user_data); - -void UI_Events_Init(void); -void UI_Events_Shutdown(void); - -int32_t UI_Events_Subscribe( - const char *event_name, const UI_WIDGET *sender, UI_EVENT_LISTENER listener, - void *user_data); - -void UI_Events_Unsubscribe(int32_t listener_id); - -void UI_Events_Fire(const UI_EVENT *event); diff --git a/src/game/ui/widgets/base.h b/src/game/ui/widgets/base.h deleted file mode 100644 index 01c15117..00000000 --- a/src/game/ui/widgets/base.h +++ /dev/null @@ -1,23 +0,0 @@ -#pragma once - -#include -#include - -struct UI_WIDGET; - -typedef void (*UI_WIDGET_CONTROL)(struct UI_WIDGET *self); -typedef int32_t (*UI_WIDGET_GET_WIDTH)(const struct UI_WIDGET *self); -typedef int32_t (*UI_WIDGET_GET_HEIGHT)(const struct UI_WIDGET *self); -typedef void (*UI_WIDGET_SET_POSITION)( - struct UI_WIDGET *self, int32_t x, int32_t y); -typedef void (*UI_WIDGET_FREE)(struct UI_WIDGET *self); - -typedef struct UI_WIDGET { - UI_WIDGET_CONTROL control; - UI_WIDGET_GET_WIDTH get_width; - UI_WIDGET_GET_HEIGHT get_height; - UI_WIDGET_SET_POSITION set_position; - UI_WIDGET_FREE free; -} UI_WIDGET; - -typedef UI_WIDGET UI_WIDGET_VTABLE; diff --git a/src/game/ui/widgets/console.c b/src/game/ui/widgets/console.c deleted file mode 100644 index 271d83f2..00000000 --- a/src/game/ui/widgets/console.c +++ /dev/null @@ -1,183 +0,0 @@ -#include "game/ui/widgets/console.h" - -#include "game/clock.h" -#include "game/console/common.h" -#include "game/ui/events.h" -#include "game/ui/widgets/label.h" -#include "game/ui/widgets/prompt.h" -#include "game/ui/widgets/stack.h" - -#include -#include - -#include - -#define WINDOW_MARGIN 5 -#define LOG_MARGIN 3 -#define TEXT_HEIGHT 15 -#define MAX_LOG_LINES 20 -#define LOG_SCALE 0.8 -#define DELAY_PER_CHAR 0.2 - -typedef struct { - UI_WIDGET_VTABLE vtable; - UI_WIDGET *container; - UI_WIDGET *prompt; - char *log_lines; - bool any_logs_on_screen; - - int32_t listener1; - int32_t listener2; - - struct { - double expire_at; - UI_WIDGET *label; - } logs[MAX_LOG_LINES]; -} UI_CONSOLE; - -static void M_HandlePromptCancel(const UI_EVENT *event, void *data); -static void M_HandlePromptConfirm(const UI_EVENT *event, void *data); - -static int32_t M_GetWidth(const UI_CONSOLE *self); -static int32_t M_GetHeight(const UI_CONSOLE *self); -static void M_SetPosition(UI_CONSOLE *self, int32_t x, int32_t y); -static void M_Control(UI_CONSOLE *self); -static void M_Free(UI_CONSOLE *self); - -static int32_t M_GetWidth(const UI_CONSOLE *const self) -{ - return 640 - 2 * WINDOW_MARGIN; -} - -static int32_t M_GetHeight(const UI_CONSOLE *const self) -{ - return 480 - 2 * WINDOW_MARGIN; -} - -static void M_SetPosition(UI_CONSOLE *const self, int32_t x, int32_t y) -{ - return self->container->set_position(self->container, x, y); -} - -static void M_Control(UI_CONSOLE *const self) -{ - self->container->control(self->container); -} - -static void M_Free(UI_CONSOLE *const self) -{ - self->prompt->free(self->prompt); - self->container->free(self->container); - UI_Events_Unsubscribe(self->listener1); - UI_Events_Unsubscribe(self->listener2); - Memory_Free(self); -} - -static void M_HandlePromptCancel(const UI_EVENT *const event, void *const data) -{ - Console_Close(); -} - -static void M_HandlePromptConfirm(const UI_EVENT *const event, void *const data) -{ - const char *text = event->data; - Console_Eval(text); - Console_Close(); -} - -UI_WIDGET *UI_Console_Create(void) -{ - UI_CONSOLE *const self = Memory_Alloc(sizeof(UI_CONSOLE)); - self->vtable = (UI_WIDGET_VTABLE) { - .control = (UI_WIDGET_CONTROL)M_Control, - .get_width = (UI_WIDGET_GET_WIDTH)M_GetWidth, - .get_height = (UI_WIDGET_GET_HEIGHT)M_GetHeight, - .set_position = (UI_WIDGET_SET_POSITION)M_SetPosition, - .free = (UI_WIDGET_FREE)M_Free, - }; - - self->container = UI_Stack_Create( - UI_STACK_LAYOUT_VERTICAL_INVERSE, M_GetWidth(self), M_GetHeight(self)); - - self->prompt = UI_Prompt_Create(M_GetWidth(self), TEXT_HEIGHT + LOG_MARGIN); - UI_Stack_AddChild(self->container, self->prompt); - - for (int32_t i = 0; i < MAX_LOG_LINES; i++) { - self->logs[i].label = - UI_Label_Create("", M_GetWidth(self), UI_LABEL_AUTO_SIZE); - UI_Label_SetScale(self->logs[i].label, LOG_SCALE); - UI_Stack_AddChild(self->container, self->logs[i].label); - } - - M_SetPosition(self, WINDOW_MARGIN, WINDOW_MARGIN); - - self->listener1 = UI_Events_Subscribe( - "confirm", self->prompt, M_HandlePromptConfirm, NULL); - self->listener2 = - UI_Events_Subscribe("cancel", self->prompt, M_HandlePromptCancel, NULL); - - return (UI_WIDGET *)self; -} - -void UI_Console_HandleOpen(UI_WIDGET *const widget) -{ - UI_CONSOLE *const self = (UI_CONSOLE *)widget; - UI_Prompt_SetFocus(self->prompt, true); -} - -void UI_Console_HandleClose(UI_WIDGET *const widget) -{ - UI_CONSOLE *const self = (UI_CONSOLE *)widget; - UI_Prompt_SetFocus(self->prompt, false); - UI_Prompt_Clear(self->prompt); -} - -void UI_Console_HandleLog(UI_WIDGET *const widget, const char *const text) -{ - UI_CONSOLE *const self = (UI_CONSOLE *)widget; - - int32_t dst_idx = -1; - for (int32_t i = MAX_LOG_LINES - 1; i > 0; i--) { - if (self->logs[i].label == NULL) { - continue; - } - UI_Label_ChangeText( - self->logs[i].label, UI_Label_GetText(self->logs[i - 1].label)); - self->logs[i].expire_at = self->logs[i - 1].expire_at; - } - - if (self->logs[0].label == NULL) { - return; - } - - self->logs[0].expire_at = - Clock_GetHighPrecisionCounter() + 1000 * strlen(text) * DELAY_PER_CHAR; - UI_Label_ChangeText(self->logs[0].label, text); - - UI_Stack_DoLayout(self->container); - self->any_logs_on_screen = true; -} - -void UI_Console_ScrollLogs(UI_WIDGET *const widget) -{ - UI_CONSOLE *const self = (UI_CONSOLE *)widget; - - int32_t i = MAX_LOG_LINES - 1; - while (i >= 0 && !self->logs[i].expire_at) { - i--; - } - - bool need_layout = false; - while (i >= 0 && self->logs[i].expire_at - && Clock_GetHighPrecisionCounter() >= self->logs[i].expire_at) { - self->logs[i].expire_at = 0; - UI_Label_ChangeText(self->logs[i].label, ""); - need_layout = true; - i--; - } - - self->any_logs_on_screen = i >= 0; - if (need_layout) { - UI_Stack_DoLayout(self->container); - } -} diff --git a/src/game/ui/widgets/console.h b/src/game/ui/widgets/console.h deleted file mode 100644 index 9a73c128..00000000 --- a/src/game/ui/widgets/console.h +++ /dev/null @@ -1,10 +0,0 @@ -#pragma once - -#include "game/ui/widgets/base.h" - -UI_WIDGET *UI_Console_Create(void); - -void UI_Console_HandleOpen(UI_WIDGET *widget); -void UI_Console_HandleClose(UI_WIDGET *widget); -void UI_Console_HandleLog(UI_WIDGET *widget, const char *text); -void UI_Console_ScrollLogs(UI_WIDGET *widget); diff --git a/src/game/ui/widgets/controls_column.c b/src/game/ui/widgets/controls_column.c index acd49f73..8bd8b75f 100644 --- a/src/game/ui/widgets/controls_column.c +++ b/src/game/ui/widgets/controls_column.c @@ -1,8 +1,8 @@ #include "game/ui/widgets/controls_column.h" #include "game/ui/widgets/controls_input_selector.h" -#include "game/ui/widgets/stack.h" +#include #include typedef struct { diff --git a/src/game/ui/widgets/controls_column.h b/src/game/ui/widgets/controls_column.h index 34928f51..3589dd48 100644 --- a/src/game/ui/widgets/controls_column.h +++ b/src/game/ui/widgets/controls_column.h @@ -1,7 +1,8 @@ #pragma once #include "game/ui/controllers/controls.h" -#include "game/ui/widgets/base.h" + +#include UI_WIDGET *UI_ControlsColumn_Create( int32_t column, UI_CONTROLS_CONTROLLER *controller); diff --git a/src/game/ui/widgets/controls_dialog.c b/src/game/ui/widgets/controls_dialog.c index b2eaacbe..dcefefe5 100644 --- a/src/game/ui/widgets/controls_dialog.c +++ b/src/game/ui/widgets/controls_dialog.c @@ -2,9 +2,9 @@ #include "game/ui/widgets/controls_column.h" #include "game/ui/widgets/controls_layout_selector.h" -#include "game/ui/widgets/stack.h" -#include "game/ui/widgets/window.h" +#include +#include #include typedef struct { diff --git a/src/game/ui/widgets/controls_dialog.h b/src/game/ui/widgets/controls_dialog.h index c8f84eba..8cec14bb 100644 --- a/src/game/ui/widgets/controls_dialog.h +++ b/src/game/ui/widgets/controls_dialog.h @@ -1,6 +1,7 @@ #pragma once #include "game/ui/controllers/controls.h" -#include "game/ui/widgets/base.h" + +#include UI_WIDGET *UI_ControlsDialog_Create(UI_CONTROLS_CONTROLLER *controller); diff --git a/src/game/ui/widgets/controls_input_selector.c b/src/game/ui/widgets/controls_input_selector.c index 1bc8d631..5af70e63 100644 --- a/src/game/ui/widgets/controls_input_selector.c +++ b/src/game/ui/widgets/controls_input_selector.c @@ -1,8 +1,7 @@ #include "game/ui/widgets/controls_input_selector.h" -#include "game/ui/widgets/label.h" -#include "game/ui/widgets/stack.h" - +#include +#include #include typedef struct { diff --git a/src/game/ui/widgets/controls_input_selector.h b/src/game/ui/widgets/controls_input_selector.h index c5a12077..6c514c47 100644 --- a/src/game/ui/widgets/controls_input_selector.h +++ b/src/game/ui/widgets/controls_input_selector.h @@ -1,7 +1,8 @@ #pragma once #include "game/ui/controllers/controls.h" -#include "game/ui/widgets/base.h" + +#include UI_WIDGET *UI_ControlsInputSelector_Create( INPUT_ROLE input_role, UI_CONTROLS_CONTROLLER *controller); diff --git a/src/game/ui/widgets/controls_layout_selector.c b/src/game/ui/widgets/controls_layout_selector.c index 0a5866c3..1acc7314 100644 --- a/src/game/ui/widgets/controls_layout_selector.c +++ b/src/game/ui/widgets/controls_layout_selector.c @@ -1,7 +1,6 @@ #include "game/ui/widgets/controls_layout_selector.h" -#include "game/ui/widgets/label.h" - +#include #include typedef struct { diff --git a/src/game/ui/widgets/controls_layout_selector.h b/src/game/ui/widgets/controls_layout_selector.h index 99c663c8..a14cead0 100644 --- a/src/game/ui/widgets/controls_layout_selector.h +++ b/src/game/ui/widgets/controls_layout_selector.h @@ -1,6 +1,7 @@ #pragma once #include "game/ui/controllers/controls.h" -#include "game/ui/widgets/base.h" + +#include UI_WIDGET *UI_ControlsLayoutSelector_Create(UI_CONTROLS_CONTROLLER *controller); diff --git a/src/game/ui/widgets/label.c b/src/game/ui/widgets/label.c index b47515bb..2784894b 100644 --- a/src/game/ui/widgets/label.c +++ b/src/game/ui/widgets/label.c @@ -1,7 +1,6 @@ -#include "game/ui/widgets/label.h" - #include "game/text.h" +#include #include typedef struct { @@ -114,6 +113,12 @@ void UI_Label_SetScale(UI_WIDGET *const widget, const float scale) Text_SetScale(self->text, PHD_ONE * scale, PHD_ONE * scale); } +void UI_Label_SetZIndex(UI_WIDGET *const widget, const int32_t z_index) +{ + UI_LABEL *const self = (UI_LABEL *)widget; + self->text->pos.z = z_index; +} + int32_t UI_Label_MeasureTextWidth(UI_WIDGET *const widget) { UI_LABEL *const self = (UI_LABEL *)widget; diff --git a/src/game/ui/widgets/label.h b/src/game/ui/widgets/label.h deleted file mode 100644 index f224cb3b..00000000 --- a/src/game/ui/widgets/label.h +++ /dev/null @@ -1,16 +0,0 @@ -#pragma once - -#include "game/ui/widgets/base.h" - -#define UI_LABEL_AUTO_SIZE (-1) - -UI_WIDGET *UI_Label_Create(const char *text, int32_t width, int32_t height); - -void UI_Label_ChangeText(UI_WIDGET *widget, const char *text); -const char *UI_Label_GetText(UI_WIDGET *widget); - -void UI_Label_AddFrame(UI_WIDGET *widget); -void UI_Label_RemoveFrame(UI_WIDGET *widget); -void UI_Label_Flash(UI_WIDGET *widget, bool enable, int32_t rate); -void UI_Label_SetScale(UI_WIDGET *widget, float scale); -int32_t UI_Label_MeasureTextWidth(UI_WIDGET *widget); diff --git a/src/game/ui/widgets/prompt.c b/src/game/ui/widgets/prompt.c index bc125e4c..d1802363 100644 --- a/src/game/ui/widgets/prompt.c +++ b/src/game/ui/widgets/prompt.c @@ -1,286 +1,11 @@ -#include "game/ui/widgets/prompt.h" +#include -#include "game/input.h" -#include "game/text.h" -#include "game/ui/common.h" -#include "game/ui/events.h" -#include "game/ui/widgets/label.h" - -#include -#include - -#include -#include - -static const char m_ValidPromptChars[] = - "abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789.- "; - -typedef struct { - UI_WIDGET_VTABLE vtable; - UI_WIDGET *label; - UI_WIDGET *caret; - - int32_t listener1; - int32_t listener2; - - struct { - int32_t x; - int32_t y; - } pos; - bool is_focused; - int32_t current_text_capacity; - char *current_text; - int32_t caret_pos; -} UI_PROMPT; - -static void M_UpdatePromptLabel(UI_PROMPT *self); -static void M_UpdateCaretLabel(UI_PROMPT *self); -static void M_MoveCaretLeft(UI_PROMPT *self); -static void M_MoveCaretRight(UI_PROMPT *self); -static void M_MoveCaretStart(UI_PROMPT *self); -static void M_MoveCaretEnd(UI_PROMPT *self); -static void M_DeleteCharBack(UI_PROMPT *self); -static void M_Confirm(UI_PROMPT *self); -static void M_Cancel(UI_PROMPT *self); -static void M_Clear(UI_PROMPT *self); - -static int32_t M_GetWidth(const UI_PROMPT *self); -static int32_t M_GetHeight(const UI_PROMPT *self); -static void M_SetPosition(UI_PROMPT *self, int32_t x, int32_t y); -static void M_Control(UI_PROMPT *self); -static void M_Free(UI_PROMPT *self); -static void M_HandleKeyDown(const UI_EVENT *event, void *user_data); -static void M_HandleChar(const UI_EVENT *event, void *user_data); - -static void M_UpdatePromptLabel(UI_PROMPT *const self) -{ - UI_Label_ChangeText(self->label, self->current_text); -} - -static void M_UpdateCaretLabel(UI_PROMPT *const self) -{ - const char old = self->current_text[self->caret_pos]; - self->current_text[self->caret_pos] = '\0'; - UI_Label_ChangeText(self->label, self->current_text); - const int32_t width = UI_Label_MeasureTextWidth(self->label); - self->current_text[self->caret_pos] = old; - UI_Label_ChangeText(self->label, self->current_text); - - self->caret->set_position(self->caret, self->pos.x + width, self->pos.y); -} - -static int32_t M_GetWidth(const UI_PROMPT *const self) -{ - return self->label->get_width(self->label); -} - -static int32_t M_GetHeight(const UI_PROMPT *const self) -{ - return self->label->get_height(self->label); -} - -static void M_SetPosition( - UI_PROMPT *const self, const int32_t x, const int32_t y) -{ - self->pos.x = x; - self->pos.y = y; - self->label->set_position(self->label, x, y); - M_UpdateCaretLabel(self); -} - -static void M_Control(UI_PROMPT *const self) -{ - self->label->control(self->label); - self->caret->control(self->caret); -} - -static void M_Free(UI_PROMPT *const self) -{ - self->label->free(self->label); - self->caret->free(self->caret); - UI_Events_Unsubscribe(self->listener1); - UI_Events_Unsubscribe(self->listener2); - Memory_FreePointer(&self->current_text); - Memory_Free(self); -} - -static void M_MoveCaretLeft(UI_PROMPT *const self) -{ - if (self->caret_pos > 0) { - self->caret_pos--; - M_UpdateCaretLabel(self); - } -} - -static void M_MoveCaretRight(UI_PROMPT *const self) -{ - if (self->caret_pos < (int32_t)strlen(self->current_text)) { - self->caret_pos++; - M_UpdateCaretLabel(self); - } -} - -static void M_MoveCaretStart(UI_PROMPT *const self) -{ - self->caret_pos = 0; - M_UpdateCaretLabel(self); -} - -static void M_MoveCaretEnd(UI_PROMPT *const self) -{ - self->caret_pos = strlen(self->current_text); - M_UpdateCaretLabel(self); -} - -static void M_DeleteCharBack(UI_PROMPT *const self) -{ - if (self->caret_pos <= 0) { - return; - } - - memmove( - self->current_text + self->caret_pos - 1, - self->current_text + self->caret_pos, - strlen(self->current_text) + 1 - self->caret_pos); - - self->caret_pos--; - M_UpdatePromptLabel(self); - M_UpdateCaretLabel(self); -} - -static void M_Confirm(UI_PROMPT *const self) -{ - if (String_IsEmpty(self->current_text)) { - M_Cancel(self); - return; - } - const UI_EVENT event = { - .name = "confirm", - .sender = (const UI_WIDGET *)self, - .data = self->current_text, - }; - UI_Events_Fire(&event); - M_Clear(self); - M_UpdateCaretLabel(self); -} - -static void M_Cancel(UI_PROMPT *const self) -{ - const UI_EVENT event = { - .name = "cancel", - .sender = (const UI_WIDGET *)self, - .data = self->current_text, - }; - UI_Events_Fire(&event); - M_Clear(self); -} - -static void M_Clear(UI_PROMPT *const self) -{ - strcpy(self->current_text, ""); - self->caret_pos = 0; - M_UpdatePromptLabel(self); - M_UpdateCaretLabel(self); -} - -static void M_HandleKeyDown(const UI_EVENT *const event, void *const user_data) -{ - const uint32_t key = (uint32_t)(uintptr_t)event->data; - UI_PROMPT *const self = user_data; - - if (!self->is_focused) { - return; - } - - // clang-format off - switch (key) { - case VK_LEFT: M_MoveCaretLeft(self); break; - case VK_RIGHT: M_MoveCaretRight(self); break; - case VK_HOME: M_MoveCaretStart(self); break; - case VK_END: M_MoveCaretEnd(self); break; - case VK_BACK: M_DeleteCharBack(self); break; - case VK_RETURN: M_Confirm(self); break; - case VK_ESCAPE: M_Cancel(self); break; - } - // clang-format on -} - -static void M_HandleChar(const UI_EVENT *const event, void *const user_data) -{ - const uint32_t char_ = (uint32_t)(uintptr_t)event->data; - UI_PROMPT *const self = user_data; - - if (!self->is_focused) { - return; - } - - char insert_string[2] = { char_, '\0' }; - const size_t insert_length = strlen(insert_string); - - if (strlen(insert_string) != 1 - || !strstr(m_ValidPromptChars, insert_string)) { - return; - } - - const size_t available_space = - self->current_text_capacity - strlen(self->current_text); - if (insert_length >= available_space) { - self->current_text_capacity *= 2; - self->current_text = - Memory_Realloc(self->current_text, self->current_text_capacity); - } - - memmove( - self->current_text + self->caret_pos + insert_length, - self->current_text + self->caret_pos, - strlen(self->current_text) + 1 - self->caret_pos); - memcpy(self->current_text + self->caret_pos, insert_string, insert_length); - - self->caret_pos += insert_length; - M_UpdatePromptLabel(self); - M_UpdateCaretLabel(self); -} - -UI_WIDGET *UI_Prompt_Create(const int32_t width, const int32_t height) -{ - UI_PROMPT *const self = Memory_Alloc(sizeof(UI_PROMPT)); - self->vtable = (UI_WIDGET_VTABLE) { - .control = (UI_WIDGET_CONTROL)M_Control, - .get_width = (UI_WIDGET_GET_WIDTH)M_GetWidth, - .get_height = (UI_WIDGET_GET_HEIGHT)M_GetHeight, - .set_position = (UI_WIDGET_SET_POSITION)M_SetPosition, - .free = (UI_WIDGET_FREE)M_Free, - }; - - self->current_text_capacity = 1; - self->current_text = Memory_Alloc(self->current_text_capacity); - self->caret = UI_Label_Create("", width, height); - self->label = UI_Label_Create(self->current_text, width, height); - self->is_focused = false; - - self->listener1 = - UI_Events_Subscribe("key_down", NULL, M_HandleKeyDown, self); - self->listener2 = UI_Events_Subscribe("char", NULL, M_HandleChar, self); - - return (UI_WIDGET *)self; -} - -void UI_Prompt_SetFocus(UI_WIDGET *const widget, const bool is_focused) +const char *UI_Prompt_GetPromptChar(void) { - UI_PROMPT *const self = (UI_PROMPT *)widget; - self->is_focused = is_focused; - if (is_focused) { - Input_EnterListenMode(); - UI_Label_ChangeText(self->caret, "\x11"); - UI_Label_Flash(self->caret, 1, 20); - } else { - Input_ExitListenMode(); - UI_Label_ChangeText(self->caret, ""); - } + return "\x11"; } -void UI_Prompt_Clear(UI_WIDGET *const widget) +int32_t UI_Prompt_GetCaretFlashRate(void) { - UI_PROMPT *const self = (UI_PROMPT *)widget; - M_Clear(self); + return 20; } diff --git a/src/game/ui/widgets/prompt.h b/src/game/ui/widgets/prompt.h deleted file mode 100644 index 4657e6c8..00000000 --- a/src/game/ui/widgets/prompt.h +++ /dev/null @@ -1,8 +0,0 @@ -#pragma once - -#include "game/ui/widgets/base.h" - -UI_WIDGET *UI_Prompt_Create(int32_t width, int32_t height); - -void UI_Prompt_SetFocus(UI_WIDGET *widget, bool is_focused); -void UI_Prompt_Clear(UI_WIDGET *widget); diff --git a/src/game/ui/widgets/stack.c b/src/game/ui/widgets/stack.c deleted file mode 100644 index b6f5b79c..00000000 --- a/src/game/ui/widgets/stack.c +++ /dev/null @@ -1,176 +0,0 @@ -#include "game/ui/widgets/stack.h" - -#include -#include -#include - -typedef struct { - UI_WIDGET_VTABLE vtable; - - int32_t width; - int32_t height; - int32_t x; - int32_t y; - UI_STACK_LAYOUT layout; - VECTOR *children; -} UI_STACK; - -static int32_t M_GetHeight(const UI_STACK *self); -static int32_t M_GetWidth(const UI_STACK *self); -static void M_SetPosition(UI_STACK *self, int32_t x, int32_t y); -static void M_Control(UI_STACK *self); -static void M_Free(UI_STACK *self); - -static int32_t M_GetWidth(const UI_STACK *const self) -{ - if (self->width != UI_STACK_AUTO_SIZE) { - return self->width; - } - int32_t result = 0; - for (int32_t i = 0; i < self->children->count; i++) { - const UI_WIDGET *const child = - *(UI_WIDGET **)Vector_Get(self->children, i); - switch (self->layout) { - case UI_STACK_LAYOUT_HORIZONTAL: - case UI_STACK_LAYOUT_HORIZONTAL_INVERSE: - result += child->get_width(child); - break; - case UI_STACK_LAYOUT_VERTICAL: - case UI_STACK_LAYOUT_VERTICAL_INVERSE: - result = MAX(result, child->get_width(child)); - break; - } - } - return result; -} - -static int32_t M_GetHeight(const UI_STACK *const self) -{ - if (self->height != UI_STACK_AUTO_SIZE) { - return self->height; - } - int32_t result = 0; - for (int32_t i = 0; i < self->children->count; i++) { - const UI_WIDGET *const child = - *(UI_WIDGET **)Vector_Get(self->children, i); - switch (self->layout) { - case UI_STACK_LAYOUT_HORIZONTAL: - case UI_STACK_LAYOUT_HORIZONTAL_INVERSE: - result = MAX(result, child->get_height(child)); - break; - case UI_STACK_LAYOUT_VERTICAL: - case UI_STACK_LAYOUT_VERTICAL_INVERSE: - result += child->get_height(child); - break; - } - } - return result; -} - -static void M_SetPosition( - UI_STACK *const self, const int32_t x, const int32_t y) -{ - self->x = x; - self->y = y; - UI_Stack_DoLayout((UI_WIDGET *)self); -} - -static void M_Control(UI_STACK *const self) -{ - for (int32_t i = 0; i < self->children->count; i++) { - UI_WIDGET *const child = *(UI_WIDGET **)Vector_Get(self->children, i); - if (child->control != NULL) { - child->control(child); - } - } -} - -static void M_Free(UI_STACK *const self) -{ - Vector_Free(self->children); - Memory_Free(self); -} - -void UI_Stack_AddChild(UI_WIDGET *const widget, UI_WIDGET *const child) -{ - UI_STACK *const self = (UI_STACK *)widget; - Vector_Add(self->children, (void *)&child); -} - -UI_WIDGET *UI_Stack_Create( - const UI_STACK_LAYOUT layout, const int32_t width, const int32_t height) -{ - UI_STACK *const self = Memory_Alloc(sizeof(UI_STACK)); - self->vtable = (UI_WIDGET_VTABLE) { - .control = (UI_WIDGET_CONTROL)M_Control, - .get_width = (UI_WIDGET_GET_WIDTH)M_GetWidth, - .get_height = (UI_WIDGET_GET_HEIGHT)M_GetHeight, - .set_position = (UI_WIDGET_SET_POSITION)M_SetPosition, - .free = (UI_WIDGET_FREE)M_Free, - }; - - self->width = width; - self->height = height; - self->layout = layout; - self->children = Vector_Create(sizeof(UI_WIDGET *)); - return (UI_WIDGET *)self; -} - -void UI_Stack_DoLayout(UI_WIDGET *const widget) -{ - UI_STACK *const self = (UI_STACK *)widget; - const int32_t self_width = M_GetWidth(self); - const int32_t self_height = M_GetHeight(self); - const int32_t start_x = self->x; - const int32_t start_y = self->y; - int32_t x; - int32_t y; - - switch (self->layout) { - case UI_STACK_LAYOUT_HORIZONTAL: - case UI_STACK_LAYOUT_VERTICAL: - x = start_x; - y = start_y; - break; - case UI_STACK_LAYOUT_HORIZONTAL_INVERSE: - case UI_STACK_LAYOUT_VERTICAL_INVERSE: - x = start_x + self_width; - y = start_y + self_height; - break; - } - - for (int32_t i = 0; i < self->children->count; i++) { - UI_WIDGET *const child = *(UI_WIDGET **)Vector_Get(self->children, i); - const int32_t child_width = child->get_width(child); - const int32_t child_height = child->get_height(child); - - // centre in the other axis - switch (self->layout) { - case UI_STACK_LAYOUT_HORIZONTAL: - case UI_STACK_LAYOUT_HORIZONTAL_INVERSE: - y = start_y + (self_height - child_height) / 2; - break; - case UI_STACK_LAYOUT_VERTICAL: - case UI_STACK_LAYOUT_VERTICAL_INVERSE: - x = start_x + (self_width - child_width) / 2; - break; - } - - child->set_position(child, x, y); - - switch (self->layout) { - case UI_STACK_LAYOUT_HORIZONTAL_INVERSE: - x -= child_width; - break; - case UI_STACK_LAYOUT_VERTICAL_INVERSE: - y -= child_height; - break; - case UI_STACK_LAYOUT_HORIZONTAL: - x += child_width; - break; - case UI_STACK_LAYOUT_VERTICAL: - y += child_height; - break; - } - } -} diff --git a/src/game/ui/widgets/stack.h b/src/game/ui/widgets/stack.h deleted file mode 100644 index c1968616..00000000 --- a/src/game/ui/widgets/stack.h +++ /dev/null @@ -1,17 +0,0 @@ -#pragma once - -#include "game/ui/widgets/base.h" - -#define UI_STACK_AUTO_SIZE (-1) -typedef enum { - UI_STACK_LAYOUT_HORIZONTAL, - UI_STACK_LAYOUT_VERTICAL, - UI_STACK_LAYOUT_HORIZONTAL_INVERSE, - UI_STACK_LAYOUT_VERTICAL_INVERSE, -} UI_STACK_LAYOUT; - -UI_WIDGET *UI_Stack_Create( - UI_STACK_LAYOUT layout, int32_t width, int32_t height); -void UI_Stack_AddChild(UI_WIDGET *self, UI_WIDGET *child); -void UI_Stack_DoLayout(UI_WIDGET *self); -void UI_Stack_SetInverse(UI_WIDGET *self, bool inverse); diff --git a/src/game/ui/widgets/window.c b/src/game/ui/widgets/window.c index e4ff34d1..ce9555fc 100644 --- a/src/game/ui/widgets/window.c +++ b/src/game/ui/widgets/window.c @@ -1,7 +1,6 @@ -#include "game/ui/widgets/window.h" - #include "game/text.h" +#include #include typedef struct { diff --git a/src/game/ui/widgets/window.h b/src/game/ui/widgets/window.h deleted file mode 100644 index 3dff3bde..00000000 --- a/src/game/ui/widgets/window.h +++ /dev/null @@ -1,7 +0,0 @@ -#pragma once - -#include "game/ui/widgets/base.h" - -UI_WIDGET *UI_Window_Create( - UI_WIDGET *root, int32_t border_top, int32_t border_right, - int32_t border_bottom, int32_t border_left); diff --git a/subprojects/libtrx b/subprojects/libtrx index 0776388f..7fda1131 160000 --- a/subprojects/libtrx +++ b/subprojects/libtrx @@ -1 +1 @@ -Subproject commit 0776388ff52e6dade3ffc8d469806ec43a0dce18 +Subproject commit 7fda11313b93326fbb74c833fb42e5e31edec2f2