ui code -> extend 'active' system to all left/middle/right mouse buttons, rather than just min/max (left/right)

This commit is contained in:
Ryan Fleury
2024-02-07 14:20:21 -08:00
parent fb9e890653
commit b9cec99cd4
4 changed files with 118 additions and 78 deletions
+3
View File
@@ -170,6 +170,9 @@
#define DeferLoop(begin, end) for(int _i_ = ((begin), 0); !_i_; _i_ += 1, (end))
#define DeferLoopChecked(begin, end) for(int _i_ = 2 * !(begin); (_i_ == 2 ? ((end), 0) : !_i_); _i_ += 1, (end))
#define EachEnumVal(type, it) type it = (type)0; it < type##_COUNT; it = (type)(it+1)
#define EachNonZeroEnumVal(type, it) type it = (type)1; it < type##_COUNT; it = (type)(it+1)
#if LANG_CPP
# define zero_struct {}
#else
+4 -1
View File
@@ -7537,7 +7537,10 @@ df_eval_viz_windowed_row_list_from_viz_block_list(Arena *arena, DBGI_Scope *scop
internal void
df_set_hover_eval(DF_Window *ws, Vec2F32 pos, DF_CtrlCtx ctrl_ctx, DF_Entity *file, TxtPt pt, U64 vaddr, String8 string)
{
if(ws->hover_eval_last_frame_idx+1 < df_frame_index() && ui_key_match(ui_active_key(Side_Min), ui_key_zero()) && ui_key_match(ui_active_key(Side_Max), ui_key_zero()))
if(ws->hover_eval_last_frame_idx+1 < df_frame_index() &&
ui_key_match(ui_active_key(UI_MouseButtonKind_Left), ui_key_zero()) &&
ui_key_match(ui_active_key(UI_MouseButtonKind_Middle), ui_key_zero()) &&
ui_key_match(ui_active_key(UI_MouseButtonKind_Right), ui_key_zero()))
{
B32 is_new_string = !str8_match(ws->hover_eval_string, string, 0);
if(is_new_string)
+95 -73
View File
@@ -591,9 +591,9 @@ ui_hot_key(void)
}
internal UI_Key
ui_active_key(Side side)
ui_active_key(UI_MouseButtonKind button_kind)
{
return ui_state->active_box_key[side];
return ui_state->active_box_key[button_kind];
}
//- rjf: controls over interaction
@@ -601,7 +601,10 @@ ui_active_key(Side side)
internal void
ui_kill_action(void)
{
ui_state->active_box_key[Side_Min] = ui_state->active_box_key[Side_Max] = ui_key_zero();
for(EachEnumVal(UI_MouseButtonKind, k))
{
ui_state->active_box_key[k] = ui_key_zero();
}
}
//- rjf: box cache lookup
@@ -958,32 +961,41 @@ ui_begin_build(OS_EventList *events, OS_Handle window, UI_NavActionList *nav_act
}
//- rjf: reset hot if we don't have an active widget
if(ui_key_match(ui_state->active_box_key[Side_Min], ui_key_zero()) &&
ui_key_match(ui_state->active_box_key[Side_Max], ui_key_zero()))
{
ui_state->hot_box_key = ui_key_zero();
B32 has_active = 0;
for(EachEnumVal(UI_MouseButtonKind, k))
{
if(!ui_key_match(ui_state->active_box_key[k], ui_key_zero()))
{
has_active = 1;
}
}
if(!has_active)
{
ui_state->hot_box_key = ui_key_zero();
}
}
//- rjf: reset active if our active box is disabled
for(Side side = (Side)0; side < Side_COUNT; side = (Side)(side+1))
for(EachEnumVal(UI_MouseButtonKind, k))
{
if(!ui_key_match(ui_state->active_box_key[side], ui_key_zero()))
if(!ui_key_match(ui_state->active_box_key[k], ui_key_zero()))
{
UI_Box *box = ui_box_from_key(ui_state->active_box_key[side]);
UI_Box *box = ui_box_from_key(ui_state->active_box_key[k]);
if(!ui_box_is_nil(box) && box->flags & UI_BoxFlag_Disabled)
{
ui_state->active_box_key[side] = ui_key_zero();
ui_state->active_box_key[k] = ui_key_zero();
}
}
}
//- rjf: reset active keys if they have been pruned
for(Side side = Side_Min; side < Side_COUNT; side = (Side)(side + 1))
for(EachEnumVal(UI_MouseButtonKind, k))
{
UI_Box *box = ui_box_from_key(ui_state->active_box_key[side]);
UI_Box *box = ui_box_from_key(ui_state->active_box_key[k]);
if(ui_box_is_nil(box))
{
ui_state->active_box_key[side] = ui_key_zero();
ui_state->active_box_key[k] = ui_key_zero();
}
}
@@ -993,9 +1005,9 @@ ui_begin_build(OS_EventList *events, OS_Handle window, UI_NavActionList *nav_act
if((event->kind == OS_EventKind_Press || event->kind == OS_EventKind_Release) &&
!os_handle_match(event->window, window))
{
for(Side side = Side_Min; side < Side_COUNT; side = (Side)(side + 1))
for(EachEnumVal(UI_MouseButtonKind, k))
{
ui_state->active_box_key[side] = ui_key_zero();
ui_state->active_box_key[k] = ui_key_zero();
}
break;
}
@@ -1154,7 +1166,7 @@ ui_end_build(void)
{
// rjf: grab states informing animation
B32 is_hot = ui_key_match(box->key, ui_state->hot_box_key);
B32 is_active = ui_key_match(box->key, ui_state->active_box_key[Side_Min]);
B32 is_active = ui_key_match(box->key, ui_state->active_box_key[UI_MouseButtonKind_Left]);
B32 is_disabled = !!(box->flags & UI_BoxFlag_Disabled) && (box->first_disabled_build_index+10 < ui_state->build_index ||
box->first_touched_build_index == box->first_disabled_build_index);
B32 is_focus_hot = !!(box->flags & UI_BoxFlag_FocusHot) && !(box->flags & UI_BoxFlag_FocusHotDisabled);
@@ -1273,7 +1285,7 @@ ui_end_build(void)
//- rjf: hover cursor
{
UI_Box *hot = ui_box_from_key(ui_state->hot_box_key);
UI_Box *active = ui_box_from_key(ui_state->active_box_key[Side_Min]);
UI_Box *active = ui_box_from_key(ui_state->active_box_key[UI_MouseButtonKind_Left]);
UI_Box *box = ui_box_is_nil(active) ? hot : active;
OS_Cursor cursor = box->hover_cursor;
if(box->flags & UI_BoxFlag_Disabled && box->flags & UI_BoxFlag_Clickable)
@@ -1313,58 +1325,68 @@ ui_end_build(void)
}
//- rjf: hovering possibly-truncated drawn text -> store text
if(ui_key_match(ui_key_zero(), ui_state->active_box_key[Side_Min]) &&
ui_key_match(ui_key_zero(), ui_state->active_box_key[Side_Max]))
{
B32 found = 0;
for(UI_Box *box = ui_state->root, *next = 0; !ui_box_is_nil(box); box = next)
B32 inactive = 1;
for(EachEnumVal(UI_MouseButtonKind, k))
{
UI_BoxRec rec = ui_box_rec_df_pre(box, ui_state->root);
next = rec.next;
S32 pop_idx = 0;
for(UI_Box *b = box; !ui_box_is_nil(b) && pop_idx <= rec.pop_count; b = b->parent, pop_idx += 1)
if(!ui_key_match(ui_key_zero(), ui_state->active_box_key[k]))
{
if(b->flags & UI_BoxFlag_DrawText && !(b->flags & UI_BoxFlag_DisableTextTrunc))
inactive = 0;
break;
}
}
if(inactive)
{
B32 found = 0;
for(UI_Box *box = ui_state->root, *next = 0; !ui_box_is_nil(box); box = next)
{
UI_BoxRec rec = ui_box_rec_df_pre(box, ui_state->root);
next = rec.next;
S32 pop_idx = 0;
for(UI_Box *b = box; !ui_box_is_nil(b) && pop_idx <= rec.pop_count; b = b->parent, pop_idx += 1)
{
String8 box_display_string = ui_box_display_string(b);
Vec2F32 text_pos = ui_box_text_position(b);
Vec2F32 drawn_text_dim = b->display_string_runs.dim;
B32 text_is_truncated = (drawn_text_dim.x + text_pos.x > b->rect.x1);
B32 mouse_is_hovering = contains_2f32(r2f32p(text_pos.x,
b->rect.y0,
Min(text_pos.x+drawn_text_dim.x, b->rect.x1),
b->rect.y1),
ui_state->mouse);
if(text_is_truncated && mouse_is_hovering)
if(b->flags & UI_BoxFlag_DrawText && !(b->flags & UI_BoxFlag_DisableTextTrunc))
{
if(!str8_match(box_display_string, ui_state->string_hover_string, 0))
String8 box_display_string = ui_box_display_string(b);
Vec2F32 text_pos = ui_box_text_position(b);
Vec2F32 drawn_text_dim = b->display_string_runs.dim;
B32 text_is_truncated = (drawn_text_dim.x + text_pos.x > b->rect.x1);
B32 mouse_is_hovering = contains_2f32(r2f32p(text_pos.x,
b->rect.y0,
Min(text_pos.x+drawn_text_dim.x, b->rect.x1),
b->rect.y1),
ui_state->mouse);
if(text_is_truncated && mouse_is_hovering)
{
arena_clear(ui_state->string_hover_arena);
ui_state->string_hover_string = push_str8_copy(ui_state->string_hover_arena, box_display_string);
ui_state->string_hover_fancy_runs = d_fancy_run_list_copy(ui_state->string_hover_arena, &b->display_string_runs);
ui_state->string_hover_begin_us = os_now_microseconds();
if(!str8_match(box_display_string, ui_state->string_hover_string, 0))
{
arena_clear(ui_state->string_hover_arena);
ui_state->string_hover_string = push_str8_copy(ui_state->string_hover_arena, box_display_string);
ui_state->string_hover_fancy_runs = d_fancy_run_list_copy(ui_state->string_hover_arena, &b->display_string_runs);
ui_state->string_hover_begin_us = os_now_microseconds();
}
ui_state->string_hover_build_index = ui_state->build_index;
found = 1;
goto break_all_hover_string;
}
ui_state->string_hover_build_index = ui_state->build_index;
found = 1;
}
if(b != box && contains_2f32(b->rect, ui_state->mouse) && b->flags & UI_BoxFlag_DrawText)
{
goto break_all_hover_string;
}
}
if(b != box && contains_2f32(b->rect, ui_state->mouse) && b->flags & UI_BoxFlag_DrawText)
{
goto break_all_hover_string;
}
}
}
break_all_hover_string:;
if(!found)
{
arena_clear(ui_state->string_hover_arena);
ui_state->string_hover_build_index = 0;
MemoryZeroStruct(&ui_state->string_hover_string);
}
if(found && !ui_string_hover_active())
{
ui_state->is_animating = 1;
break_all_hover_string:;
if(!found)
{
arena_clear(ui_state->string_hover_arena);
ui_state->string_hover_build_index = 0;
MemoryZeroStruct(&ui_state->string_hover_string);
}
if(found && !ui_string_hover_active())
{
ui_state->is_animating = 1;
}
}
}
@@ -2449,8 +2471,8 @@ ui_signal_from_box(UI_Box *box)
if(box->flags & UI_BoxFlag_MouseClickable && !ui_key_match(ui_key_zero(), box->key)) ProfScope("clickability")
{
// rjf: hot management
if((ui_key_match(ui_key_zero(), ui_state->active_box_key[Side_Min]) &&
ui_key_match(ui_key_zero(), ui_state->active_box_key[Side_Max])) &&
if((ui_key_match(ui_key_zero(), ui_state->active_box_key[UI_MouseButtonKind_Left]) &&
ui_key_match(ui_key_zero(), ui_state->active_box_key[UI_MouseButtonKind_Right])) &&
ui_key_match(ui_key_zero(), ui_state->hot_box_key) &&
mouse_is_over)
{
@@ -2458,8 +2480,8 @@ ui_signal_from_box(UI_Box *box)
}
else if(ui_key_match(ui_state->hot_box_key, box->key) &&
!mouse_is_over &&
!ui_key_match(ui_state->active_box_key[Side_Min], box->key) &&
!ui_key_match(ui_state->active_box_key[Side_Max], box->key))
!ui_key_match(ui_state->active_box_key[UI_MouseButtonKind_Left], box->key) &&
!ui_key_match(ui_state->active_box_key[UI_MouseButtonKind_Right], box->key))
{
ui_state->hot_box_key = ui_key_zero();
}
@@ -2467,48 +2489,48 @@ ui_signal_from_box(UI_Box *box)
// rjf: active management (left click)
if(!disabled &&
ui_key_match(ui_state->hot_box_key, box->key) &&
ui_key_match(ui_state->active_box_key[Side_Min], ui_key_zero()) &&
ui_key_match(ui_state->active_box_key[UI_MouseButtonKind_Left], ui_key_zero()) &&
left_press != 0)
{
os_eat_event(ui_state->events, left_press);
result.pressed = 1;
ui_state->active_box_key[Side_Min] = box->key;
ui_state->active_box_key[UI_MouseButtonKind_Left] = box->key;
}
else if(!disabled &&
ui_key_match(ui_state->active_box_key[Side_Min], box->key) &&
ui_key_match(ui_state->active_box_key[UI_MouseButtonKind_Left], box->key) &&
left_release != 0)
{
os_eat_event(ui_state->events, left_release);
result.released = 1;
result.clicked = mouse_is_over;
ui_state->hot_box_key = mouse_is_over ? box->key : ui_key_zero();
ui_state->active_box_key[Side_Min] = ui_key_zero();
ui_state->active_box_key[UI_MouseButtonKind_Left] = ui_key_zero();
}
// rjf: active management (right click)
if(!disabled &&
ui_key_match(ui_state->hot_box_key, box->key) &&
ui_key_match(ui_state->active_box_key[Side_Max], ui_key_zero()) &&
ui_key_match(ui_state->active_box_key[UI_MouseButtonKind_Right], ui_key_zero()) &&
right_press != 0)
{
os_eat_event(ui_state->events, right_press);
// NOTE(rjf): Add this in if it ever needs to exist:
// result.right_pressed = 1;
ui_state->active_box_key[Side_Max] = box->key;
ui_state->active_box_key[UI_MouseButtonKind_Right] = box->key;
}
else if(!disabled &&
ui_key_match(ui_state->active_box_key[Side_Max], box->key) &&
ui_key_match(ui_state->active_box_key[UI_MouseButtonKind_Right], box->key) &&
right_release != 0)
{
os_eat_event(ui_state->events, right_release);
// NOTE(rjf): Add this in if it ever needs to exist:
// result.right_released = 1;
result.right_clicked = mouse_is_over;
ui_state->active_box_key[Side_Max] = ui_key_zero();
ui_state->active_box_key[UI_MouseButtonKind_Right] = ui_key_zero();
}
// rjf: dragging
if(ui_key_match(ui_state->active_box_key[Side_Min], box->key))
if(ui_key_match(ui_state->active_box_key[UI_MouseButtonKind_Left], box->key))
{
result.dragging = 1;
if(result.pressed)
@@ -2700,8 +2722,8 @@ ui_signal_from_box(UI_Box *box)
//- rjf: set hovering status
result.hovering = mouse_is_over && ((ui_key_match(ui_state->hot_box_key, ui_key_zero()) ||
ui_key_match(ui_state->hot_box_key, box->key)) &&
(ui_key_match(ui_state->active_box_key[Side_Min], ui_key_zero()) ||
ui_key_match(ui_state->active_box_key[Side_Min], box->key)));
(ui_key_match(ui_state->active_box_key[UI_MouseButtonKind_Left], ui_key_zero()) ||
ui_key_match(ui_state->active_box_key[UI_MouseButtonKind_Left], box->key)));
result.mouse_over = mouse_is_over;
//- rjf: clicking in default nav -> set navigation state to this box
+16 -4
View File
@@ -31,6 +31,18 @@ struct UI_IconInfo
String8 icon_kind_text_map[UI_IconKind_COUNT];
};
////////////////////////////////
//~ rjf: Mouse Button Kinds
typedef enum UI_MouseButtonKind
{
UI_MouseButtonKind_Left,
UI_MouseButtonKind_Middle,
UI_MouseButtonKind_Right,
UI_MouseButtonKind_COUNT
}
UI_MouseButtonKind;
////////////////////////////////
//~ rjf: Focus Types
@@ -415,10 +427,10 @@ struct UI_State
//- rjf: user interaction state
UI_Key hot_box_key;
UI_Key active_box_key[Side_COUNT];
UI_Key active_box_key[UI_MouseButtonKind_COUNT];
UI_Key clipboard_copy_key;
F32 time_since_last_click[Side_COUNT];
UI_Key last_click_key[Side_COUNT];
F32 time_since_last_click[UI_MouseButtonKind_COUNT];
UI_Key last_click_key[UI_MouseButtonKind_COUNT];
Vec2F32 drag_start_mouse;
Arena *drag_state_arena;
String8 drag_state_data;
@@ -551,7 +563,7 @@ internal D_FancyRunList ui_string_hover_runs(Arena *arena);
//- rjf: interaction keys
internal UI_Key ui_hot_key(void);
internal UI_Key ui_active_key(Side side);
internal UI_Key ui_active_key(UI_MouseButtonKind button_kind);
//- rjf: controls over interaction
internal void ui_kill_action(void);