-
Notifications
You must be signed in to change notification settings - Fork 5
/
Copy pathwidgets.button.go
580 lines (502 loc) · 23.9 KB
/
widgets.button.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
package imgui
// in 'repeat' mode, Button*() functions return repeated true in a typematic manner (using io.KeyRepeatDelay/io.KeyRepeatRate setting). Note that you can call IsItemActive() after any Button() to tell if the button is held in the current frame.
func PushButtonRepeat(repeat bool) {
PushItemFlag(ImGuiItemFlags_ButtonRepeat, repeat)
}
func PopButtonRepeat() { PopItemFlag() }
// button
func Button(label string) bool {
return ButtonEx(label, &ImVec2{}, ImGuiButtonFlags_None)
}
// Small buttons fits within text without additional vertical spacing.
// button with FramePadding=(0,0) to easily embed within text
func SmallButton(label string) bool {
var g = GImGui
var backup_padding_y = g.Style.FramePadding.y
g.Style.FramePadding.y = 0.0
var pressed = ButtonEx(label, &ImVec2{}, ImGuiButtonFlags_AlignTextBaseLine)
g.Style.FramePadding.y = backup_padding_y
return pressed
}
func ButtonEx(label string, size_arg *ImVec2, flags ImGuiButtonFlags) bool {
var window = GetCurrentWindow()
if window.SkipItems {
return false
}
var g = GImGui
var style = g.Style
var id = window.GetIDs(label)
var label_size = CalcTextSize(label, true, 0)
var pos = window.DC.CursorPos
if (flags&ImGuiButtonFlags_AlignTextBaseLine) != 0 && style.FramePadding.y < window.DC.CurrLineTextBaseOffset { // Try to vertically align buttons that are smaller/have no padding so that text baseline matches (bit hacky, since it shouldn't be a flag)
pos.y += window.DC.CurrLineTextBaseOffset - style.FramePadding.y
}
var size = CalcItemSize(*size_arg, label_size.x+style.FramePadding.x*2.0, label_size.y+style.FramePadding.y*2.0)
var bb = ImRect{pos, pos.Add(size)}
ItemSizeVec(&size, style.FramePadding.y)
if !ItemAdd(&bb, id, nil, 0) {
return false
}
if g.LastItemData.InFlags&ImGuiItemFlags_ButtonRepeat != 0 {
flags |= ImGuiButtonFlags_Repeat
}
var hovered, held bool
var pressed = ButtonBehavior(&bb, id, &hovered, &held, flags)
// Render
var col ImU32
if held && hovered {
col = GetColorU32FromID(ImGuiCol_ButtonActive, 1)
} else if hovered {
col = GetColorU32FromID(ImGuiCol_ButtonHovered, 1)
} else {
col = GetColorU32FromID(ImGuiCol_Button, 1)
}
RenderNavHighlight(&bb, id, 0)
RenderFrame(bb.Min, bb.Max, col, true, style.FrameRounding)
if g.LogEnabled {
LogSetNextTextDecoration("[", "]")
}
min, max := bb.Min.Add(style.FramePadding), bb.Max.Sub(style.FramePadding)
RenderTextClipped(&min, &max, label, &label_size, &style.ButtonTextAlign, &bb)
return pressed
}
// The ButtonBehavior() function is key to many interactions and used by many/most widgets.
// Because we handle so many cases (keyboard/gamepad navigation, drag and drop) and many specific behavior (via ImGuiButtonFlags_),
// this code is a little complex.
// By far the most common path is interacting with the Mouse using the default ImGuiButtonFlags_PressedOnClickRelease button behavior.
// See the series of events below and the corresponding state reported by dear imgui:
// ------------------------------------------------------------------------------------------------------------------------------------------------
// with PressedOnClickRelease: return-value IsItemHovered() IsItemActive() IsItemActivated() IsItemDeactivated() IsItemClicked()
//
// Frame N+0 (mouse is outside bb) - - - - - -
// Frame N+1 (mouse moves inside bb) - true - - - -
// Frame N+2 (mouse button is down) - true true true - true
// Frame N+3 (mouse button is down) - true true - - -
// Frame N+4 (mouse moves outside bb) - - true - - -
// Frame N+5 (mouse moves inside bb) - true true - - -
// Frame N+6 (mouse button is released) true true - - true -
// Frame N+7 (mouse button is released) - true - - - -
// Frame N+8 (mouse moves outside bb) - - - - - -
//
// ------------------------------------------------------------------------------------------------------------------------------------------------
// with PressedOnClick: return-value IsItemHovered() IsItemActive() IsItemActivated() IsItemDeactivated() IsItemClicked()
//
// Frame N+2 (mouse button is down) true true true true - true
// Frame N+3 (mouse button is down) - true true - - -
// Frame N+6 (mouse button is released) - true - - true -
// Frame N+7 (mouse button is released) - true - - - -
//
// ------------------------------------------------------------------------------------------------------------------------------------------------
// with PressedOnRelease: return-value IsItemHovered() IsItemActive() IsItemActivated() IsItemDeactivated() IsItemClicked()
//
// Frame N+2 (mouse button is down) - true - - - true
// Frame N+3 (mouse button is down) - true - - - -
// Frame N+6 (mouse button is released) true true - - - -
// Frame N+7 (mouse button is released) - true - - - -
//
// ------------------------------------------------------------------------------------------------------------------------------------------------
// with PressedOnDoubleClick: return-value IsItemHovered() IsItemActive() IsItemActivated() IsItemDeactivated() IsItemClicked()
//
// Frame N+0 (mouse button is down) - true - - - true
// Frame N+1 (mouse button is down) - true - - - -
// Frame N+2 (mouse button is released) - true - - - -
// Frame N+3 (mouse button is released) - true - - - -
// Frame N+4 (mouse button is down) true true true true - true
// Frame N+5 (mouse button is down) - true true - - -
// Frame N+6 (mouse button is released) - true - - true -
// Frame N+7 (mouse button is released) - true - - - -
//
// ------------------------------------------------------------------------------------------------------------------------------------------------
// Note that some combinations are supported,
// - PressedOnDragDropHold can generally be associated with any flag.
// - PressedOnDoubleClick can be associated by PressedOnClickRelease/PressedOnRelease, in which case the second release event won't be reported.
// ------------------------------------------------------------------------------------------------------------------------------------------------
// The behavior of the return-value changes when ImGuiButtonFlags_Repeat is set:
//
// Repeat+ Repeat+ Repeat+ Repeat+
// PressedOnClickRelease PressedOnClick PressedOnRelease PressedOnDoubleClick
//
// -------------------------------------------------------------------------------------------------------------------------------------------------
//
// Frame N+0 (mouse button is down) - true - true
// ... - - - -
// Frame N + RepeatDelay true true - true
// ... - - - -
// Frame N + RepeatDelay + RepeatRate*N true true - true
//
// -------------------------------------------------------------------------------------------------------------------------------------------------
func ButtonBehavior(bb *ImRect, id ImGuiID, out_hovered *bool, out_held *bool, flags ImGuiButtonFlags) bool {
var g = GImGui
var window = GetCurrentWindow()
// Default only reacts to left mouse button
if (flags & ImGuiButtonFlags_MouseButtonMask_) == 0 {
flags |= ImGuiButtonFlags_MouseButtonDefault_
}
// Default behavior requires click + release inside bounding box
if (flags & ImGuiButtonFlags_PressedOnMask_) == 0 {
flags |= ImGuiButtonFlags_PressedOnDefault_
}
var backup_hovered_window = g.HoveredWindow
var flatten_hovered_children = (flags&ImGuiButtonFlags_FlattenChildren != 0) && g.HoveredWindow != nil && g.HoveredWindow.RootWindow == window
if flatten_hovered_children {
g.HoveredWindow = window
}
var pressed = false
var hovered = ItemHoverable(bb, id)
// Drag source doesn't report as hovered
if hovered && g.DragDropActive && g.DragDropPayload.SourceId == id && g.DragDropSourceFlags&ImGuiDragDropFlags_SourceNoDisableHover == 0 {
hovered = false
}
// Special mode for Drag and Drop where holding button pressed for a long time while dragging another item triggers the button
if g.DragDropActive && (flags&ImGuiButtonFlags_PressedOnDragDropHold != 0) && g.DragDropSourceFlags&ImGuiDragDropFlags_SourceNoHoldToOpenOthers == 0 {
if IsItemHovered(ImGuiHoveredFlags_AllowWhenBlockedByActiveItem) {
hovered = true
SetHoveredID(id)
if g.HoveredIdTimer-g.IO.DeltaTime <= DRAGDROP_HOLD_TO_OPEN_TIMER && g.HoveredIdTimer >= DRAGDROP_HOLD_TO_OPEN_TIMER {
pressed = true
g.DragDropHoldJustPressedId = id
FocusWindow(window)
}
}
}
if flatten_hovered_children {
g.HoveredWindow = backup_hovered_window
}
// AllowOverlap mode (rarely used) requires previous frame HoveredId to be null or to match. This allows using patterns where a later submitted widget overlaps a previous one.
if hovered && (flags&ImGuiButtonFlags_AllowItemOverlap != 0) && (g.HoveredIdPreviousFrame != id && g.HoveredIdPreviousFrame != 0) {
hovered = false
}
// Mouse handling
if hovered {
if flags&ImGuiButtonFlags_NoKeyModifiers == 0 || (!g.IO.KeyCtrl && !g.IO.KeyShift && !g.IO.KeyAlt) {
// Poll buttons
var mouse_button_clicked ImGuiMouseButton = -1
var mouse_button_released ImGuiMouseButton = -1
if (flags&ImGuiButtonFlags_MouseButtonLeft != 0) && g.IO.MouseClicked[0] {
mouse_button_clicked = 0
} else if (flags&ImGuiButtonFlags_MouseButtonRight != 0) && g.IO.MouseClicked[1] {
mouse_button_clicked = 1
} else if (flags&ImGuiButtonFlags_MouseButtonMiddle != 0) && g.IO.MouseClicked[2] {
mouse_button_clicked = 2
}
if (flags&ImGuiButtonFlags_MouseButtonLeft != 0) && g.IO.MouseReleased[0] {
mouse_button_released = 0
} else if (flags&ImGuiButtonFlags_MouseButtonRight != 0) && g.IO.MouseReleased[1] {
mouse_button_released = 1
} else if (flags&ImGuiButtonFlags_MouseButtonMiddle != 0) && g.IO.MouseReleased[2] {
mouse_button_released = 2
}
if mouse_button_clicked != -1 && g.ActiveId != id {
if flags&(ImGuiButtonFlags_PressedOnClickRelease|ImGuiButtonFlags_PressedOnClickReleaseAnywhere) != 0 {
SetActiveID(id, window)
g.ActiveIdMouseButton = mouse_button_clicked
if flags&ImGuiButtonFlags_NoNavFocus == 0 {
SetFocusID(id, window)
}
FocusWindow(window)
}
if (flags&ImGuiButtonFlags_PressedOnClick != 0) || ((flags&ImGuiButtonFlags_PressedOnDoubleClick != 0) && g.IO.MouseDoubleClicked[mouse_button_clicked]) {
pressed = true
if flags&ImGuiButtonFlags_NoHoldingActiveId != 0 {
ClearActiveID()
} else {
SetActiveID(id, window) // Hold on ID
}
g.ActiveIdMouseButton = mouse_button_clicked
FocusWindow(window)
}
}
if (flags&ImGuiButtonFlags_PressedOnRelease != 0) && mouse_button_released != -1 {
// Repeat mode trumps on release behavior
var has_repeated_at_least_once = (flags&ImGuiButtonFlags_Repeat != 0) && g.IO.MouseDownDurationPrev[mouse_button_released] >= g.IO.KeyRepeatDelay
if !has_repeated_at_least_once {
pressed = true
}
ClearActiveID()
}
// 'Repeat' mode acts when held regardless of _PressedOn flags (see table above).
// Relies on repeat logic of IsMouseClicked() but we may as well do it ourselves if we end up exposing finer RepeatDelay/RepeatRate settings.
if g.ActiveId == id && (flags&ImGuiButtonFlags_Repeat != 0) {
if g.IO.MouseDownDuration[g.ActiveIdMouseButton] > 0.0 && IsMouseClicked(g.ActiveIdMouseButton, true) {
pressed = true
}
}
}
if pressed {
g.NavDisableHighlight = true
}
}
// Gamepad/Keyboard navigation
// We report navigated item as hovered but we don't set g.HoveredId to not interfere with mouse.
if g.NavId == id && !g.NavDisableHighlight && g.NavDisableMouseHover && (g.ActiveId == 0 || g.ActiveId == id || g.ActiveId == window.MoveId) {
if flags&ImGuiButtonFlags_NoHoveredOnFocus == 0 {
hovered = true
}
}
if g.NavActivateDownId == id {
var nav_activated_by_code = (g.NavActivateId == id)
var nav_activated_by_inputs bool
if (flags & ImGuiButtonFlags_Repeat) != 0 {
nav_activated_by_inputs = IsNavInputTest(ImGuiNavInput_Activate, ImGuiInputReadMode_Repeat)
} else {
nav_activated_by_inputs = IsNavInputTest(ImGuiNavInput_Activate, ImGuiInputReadMode_Pressed)
}
if nav_activated_by_code || nav_activated_by_inputs {
pressed = true
}
if nav_activated_by_code || nav_activated_by_inputs || g.ActiveId == id {
// Set active id so it can be queried by user via IsItemActive(), equivalent of holding the mouse button.
g.NavActivateId = id // This is so SetActiveId assign a Nav source
SetActiveID(id, window)
if (nav_activated_by_code || nav_activated_by_inputs) && flags&ImGuiButtonFlags_NoNavFocus == 0 {
SetFocusID(id, window)
}
}
}
// Process while held
var held = false
if g.ActiveId == id {
if g.ActiveIdSource == ImGuiInputSource_Mouse {
if g.ActiveIdIsJustActivated {
g.ActiveIdClickOffset = g.IO.MousePos.Sub(bb.Min)
}
var mouse_button = g.ActiveIdMouseButton
IM_ASSERT(mouse_button >= 0 && mouse_button < ImGuiMouseButton_COUNT)
if g.IO.MouseDown[mouse_button] {
held = true
} else {
var release_in = hovered && (flags&ImGuiButtonFlags_PressedOnClickRelease) != 0
var release_anywhere = (flags & ImGuiButtonFlags_PressedOnClickReleaseAnywhere) != 0
if (release_in || release_anywhere) && !g.DragDropActive {
// Report as pressed when releasing the mouse (this is the most common path)
var is_double_click_release = (flags&ImGuiButtonFlags_PressedOnDoubleClick != 0) && g.IO.MouseDownWasDoubleClick[mouse_button]
var is_repeating_already = (flags&ImGuiButtonFlags_Repeat != 0) && g.IO.MouseDownDurationPrev[mouse_button] >= g.IO.KeyRepeatDelay // Repeat mode trumps <on release>
if !is_double_click_release && !is_repeating_already {
pressed = true
}
}
ClearActiveID()
}
if flags&ImGuiButtonFlags_NoNavFocus == 0 {
g.NavDisableHighlight = true
}
} else if g.ActiveIdSource == ImGuiInputSource_Nav {
// When activated using Nav, we hold on the ActiveID until activation button is released
if g.NavActivateDownId != id {
ClearActiveID()
}
}
if pressed {
g.ActiveIdHasBeenPressedBefore = true
}
}
if out_hovered != nil {
*out_hovered = hovered
}
if out_held != nil {
*out_held = held
}
return pressed
}
func CollapseButton(id ImGuiID, pos *ImVec2) bool {
var g = GImGui
var window = g.CurrentWindow
var bb = ImRect{*pos, pos.Add(ImVec2{g.FontSize, g.FontSize}).Add(g.Style.FramePadding.Scale(2.0))}
ItemAdd(&bb, id, nil, 0)
var hovered, held bool
var pressed = ButtonBehavior(&bb, id, &hovered, &held, ImGuiButtonFlags_None)
// Render
var bg_col = GetColorU32FromID(ImGuiCol_Button, 1)
if held && hovered {
bg_col = GetColorU32FromID(ImGuiCol_ButtonActive, 1)
} else if hovered {
bg_col = GetColorU32FromID(ImGuiCol_ButtonHovered, 1)
}
var text_col = GetColorU32FromID(ImGuiCol_Text, 1)
var center = bb.GetCenter()
if hovered || held {
window.DrawList.AddCircleFilled(center /*+ ImVec2(0.0f, -0.5f)*/, g.FontSize*0.5+1.0, bg_col, 12)
}
var arrow = ImGuiDir_Down
if window.Collapsed {
arrow = ImGuiDir_Right
}
RenderArrow(window.DrawList, bb.Min.Add(g.Style.FramePadding), text_col, arrow, 1.0)
// Switch to moving the window after mouse is moved beyond the initial drag threshold
if IsItemActive() && IsMouseDragging(0, -1) {
StartMouseMovingWindow(window)
}
return pressed
}
// flexible button behavior without the visuals, frequently useful to build custom behaviors using the public api (along with IsItemActive, IsItemHovered, etc.)
// Tip: use ImGui::PushID()/PopID() to push indices or pointers in the ID stack.
// Then you can keep 'str_id' empty or the same for all your buttons (instead of creating a string based on a non-string id)
func InvisibleButton(str_id string, size_arg ImVec2, flags ImGuiButtonFlags) bool {
var window = GetCurrentWindow()
if window.SkipItems {
return false
}
// Cannot use zero-size for InvisibleButton(). Unlike Button() there is not way to fallback using the label size.
IM_ASSERT(size_arg.x != 0.0 && size_arg.y != 0.0)
var id = window.GetIDs(str_id)
var size = CalcItemSize(size_arg, 0.0, 0.0)
var bb = ImRect{window.DC.CursorPos, window.DC.CursorPos.Add(size)}
ItemSizeVec(&size, 0)
if !ItemAdd(&bb, id, nil, 0) {
return false
}
var hovered, held bool
var pressed = ButtonBehavior(&bb, id, &hovered, &held, flags)
return pressed
}
// square button with an arrow shape
func ArrowButton(str_id string, dir ImGuiDir) bool {
var sz = GetFrameHeight()
return ArrowButtonEx(str_id, dir, ImVec2{sz, sz}, ImGuiButtonFlags_None)
}
// use with e.g. if (RadioButton("one", my_value==1)) { my_value = 1 bool {panic("not implemented")} }
func RadioButtonBool(label string, active bool) bool {
var window = GetCurrentWindow()
if window.SkipItems {
return false
}
var g = GImGui
var style = g.Style
var id = window.GetIDs(label)
var label_size = CalcTextSize(label, true, -1)
var square_sz = GetFrameHeight()
var pos = window.DC.CursorPos
var check_bb = ImRect{pos, pos.Add(ImVec2{square_sz, square_sz})}
var padding float
if label_size.x > 0.0 {
padding = style.ItemInnerSpacing.x + label_size.x
}
var total_bb = ImRect{pos, pos.Add(ImVec2{square_sz + padding, label_size.y + style.FramePadding.y*2.0})}
ItemSizeRect(&total_bb, style.FramePadding.y)
if !ItemAdd(&total_bb, id, nil, 0) {
return false
}
var center = check_bb.GetCenter()
center.x = IM_ROUND(center.x)
center.y = IM_ROUND(center.y)
var radius = (square_sz - 1.0) * 0.5
var hovered, held bool
var pressed = ButtonBehavior(&total_bb, id, &hovered, &held, 0)
if pressed {
MarkItemEdited(id)
}
RenderNavHighlight(&total_bb, id, 0)
var c = ImGuiCol_FrameBg
if held && hovered {
c = ImGuiCol_FrameBgHovered
} else if hovered {
c = ImGuiCol_FrameBgHovered
}
window.DrawList.AddCircleFilled(center, radius, GetColorU32FromID(c, 1), 16)
if active {
var pad = ImMax(1.0, IM_FLOOR(square_sz/6.0))
window.DrawList.AddCircleFilled(center, radius-pad, GetColorU32FromID(ImGuiCol_CheckMark, 1), 16)
}
if style.FrameBorderSize > 0.0 {
window.DrawList.AddCircle(center.Add(ImVec2{1, 1}), radius, GetColorU32FromID(ImGuiCol_BorderShadow, 1), 16, style.FrameBorderSize)
window.DrawList.AddCircle(center, radius, GetColorU32FromID(ImGuiCol_Border, 1), 16, style.FrameBorderSize)
}
var label_pos = ImVec2{check_bb.Max.x + style.ItemInnerSpacing.x, check_bb.Min.y + style.FramePadding.y}
if g.LogEnabled {
s := "( )"
if active {
s = "(X)"
}
LogRenderedText(&label_pos, s)
}
if label_size.x > 0.0 {
RenderText(label_pos, label, true)
}
return pressed
}
// shortcut to handle the above pattern when value is an integer
func RadioButtonInt(label string, v *int, v_button int) bool {
// FIXME: This would work nicely if it was a public template, e.g. 'template<T> RadioButton(const char* label, T* v, T v_button)', but I'm not sure how we would expose it..
var pressed = RadioButtonBool(label, *v == v_button)
if pressed {
*v = v_button
}
return pressed
}
// Button to close a window
func CloseButton(id ImGuiID, pos *ImVec2) bool {
var g = GImGui
var window = g.CurrentWindow
// Tweak 1: Shrink hit-testing area if button covers an abnormally large proportion of the visible region. That's in order to facilitate moving the window away. (#3825)
// This may better be applied as a general hit-rect reduction mechanism for all widgets to ensure the area to move window is always accessible?
var bb = ImRect{*pos, pos.Add(ImVec2{g.FontSize, g.FontSize}).Add(g.Style.FramePadding.Scale(2.0))}
var bb_interact = bb
var area_to_visible_ratio = window.OuterRectClipped.GetArea() / bb.GetArea()
if area_to_visible_ratio < 1.5 {
expansion := bb_interact.GetSize().Scale(-0.25)
bb_interact.ExpandVec(*ImFloorVec(&expansion))
}
// Tweak 2: We intentionally allow interaction when clipped so that a mechanical Alt,Right,Activate sequence can always close a window.
// (this isn't the regular behavior of buttons, but it doesn't affect the user much because navigation tends to keep items visible).
var is_clipped = !ItemAdd(&bb_interact, id, nil, 0)
var hovered, held bool
var pressed = ButtonBehavior(&bb_interact, id, &hovered, &held, 0)
if is_clipped {
return pressed
}
// Render
// FIXME: Clarify this mess
var c = ImGuiCol_ButtonHovered
if held {
c = ImGuiCol_ButtonActive
}
var col = GetColorU32FromID(c, 1)
var center = bb.GetCenter()
if hovered {
window.DrawList.AddCircleFilled(center, ImMax(2.0, g.FontSize*0.5+1.0), col, 12)
}
var cross_extent = g.FontSize*0.5*0.7071 - 1.0
var cross_col = GetColorU32FromID(ImGuiCol_Text, 1)
center = center.Sub(ImVec2{0.5, 0.5})
a, b := center.Add(ImVec2{+cross_extent, +cross_extent}), center.Add(ImVec2{-cross_extent, -cross_extent})
window.DrawList.AddLine(&a, &b, cross_col, 1.0)
a, b = center.Add(ImVec2{+cross_extent, -cross_extent}), center.Add(ImVec2{-cross_extent, +cross_extent})
window.DrawList.AddLine(&a, &b, cross_col, 1.0)
return pressed
}
func ArrowButtonEx(str_id string, dir ImGuiDir, size ImVec2, flags ImGuiButtonFlags) bool {
var window = GetCurrentWindow()
if window.SkipItems {
return false
}
var g = GImGui
var id = window.GetIDs(str_id)
var bb = ImRect{window.DC.CursorPos, window.DC.CursorPos.Add(size)}
var default_size = GetFrameHeight()
var baseline float = -1
if size.y >= default_size {
baseline = g.Style.FramePadding.y
}
ItemSizeVec(&size, baseline)
if !ItemAdd(&bb, id, nil, 0) {
return false
}
if (g.LastItemData.InFlags & ImGuiItemFlags_ButtonRepeat) != 0 {
flags |= ImGuiButtonFlags_Repeat
}
var hovered, held bool
var pressed = ButtonBehavior(&bb, id, &hovered, &held, flags)
var c = ImGuiCol_Button
if held && hovered {
c = ImGuiCol_ButtonActive
} else if hovered {
c = ImGuiCol_ButtonHovered
}
// Render
var bg_col = GetColorU32FromID(c, 1.0)
var text_col = GetColorU32FromID(ImGuiCol_Text, 1)
RenderNavHighlight(&bb, id, 0)
RenderFrame(bb.Min, bb.Max, bg_col, true, g.Style.FrameRounding)
RenderArrow(window.DrawList, bb.Min.Add(ImVec2{ImMax(0.0, (size.x-g.FontSize)*0.5), ImMax(0.0, (size.y-g.FontSize)*0.5)}), text_col, dir, 1)
return pressed
}