-
Notifications
You must be signed in to change notification settings - Fork 7
/
oui.h
2025 lines (1696 loc) · 65.4 KB
/
oui.h
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
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
930
931
932
933
934
935
936
937
938
939
940
941
942
943
944
945
946
947
948
949
950
951
952
953
954
955
956
957
958
959
960
961
962
963
964
965
966
967
968
969
970
971
972
973
974
975
976
977
978
979
980
981
982
983
984
985
986
987
988
989
990
991
992
993
994
995
996
997
998
999
1000
/*
OUI - A minimal semi-immediate GUI handling & layouting library
Copyright (c) 2014 Leonard Ritter <[email protected]>
Permission is hereby granted, free of charge, to any person obtaining a copy
of this software and associated documentation files (the "Software"), to deal
in the Software without restriction, including without limitation the rights
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
copies of the Software, and to permit persons to whom the Software is
furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in
all copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
THE SOFTWARE.
*/
#ifndef _OUI_H_
#define _OUI_H_
#ifdef __cplusplus
extern "C" {
#endif
/*
Revision 4 (2014-12-17)
OUI (short for "Open UI", spoken like the french "oui" for "yes") is a
platform agnostic single-header C library for layouting GUI elements and
handling related user input. Together with a set of widget drawing and logic
routines it can be used to build complex user interfaces.
OUI is a semi-immediate GUI. Widget declarations are persistent for the duration
of the setup and evaluation, but do not need to be kept around longer than one
frame.
OUI has no widget types; instead, it provides only one kind of element, "Items",
which can be tailored to the application by the user and expanded with custom
buffers and event handlers to behave as containers, buttons, sliders, radio
buttons, and so on.
OUI also does not draw anything; Instead it provides a set of functions to
iterate and query the layouted items in order to allow client code to render
each widget with its current state using a preferred graphics library.
See example.cpp in the repository for a full usage example.
A basic setup for OUI usage in C looks like this:
=================================================
// a header for each widget
typedef struct Data {
int type;
UIhandler handler;
} Data;
/// global event dispatch
void ui_handler(int item, UIevent event) {
Data *data = (Data *)uiGetHandle(item);
if (data && data->handler) {
data->handler(item, event);
}
}
void app_main(...) {
UIcontext *context = uiCreateContext(4096, 1<<20);
uiMakeCurrent(context);
uiSetHandler(ui_handler);
while (app_running()) {
// update position of mouse cursor; the ui can also be updated
// from received events.
uiSetCursor(app_get_mouse_x(), app_get_mouse_y());
// update button state
for (int i = 0; i < 3; ++i)
uiSetButton(i, app_get_button_state(i));
// you can also send keys and scroll events; see example.cpp for more
// --------------
// this section does not have to be regenerated on frame; a good
// policy is to invalidate it on events, as this usually alters
// structure and layout.
// begin new UI declarations
uiBeginLayout();
// - UI setup code goes here -
app_setup_ui();
// layout UI
uiEndLayout();
// --------------
// draw UI, starting with the first item, index 0
app_draw_ui(render_context,0);
// update states and fire handlers
uiProcess(get_time_ms());
}
uiDestroyContext(context);
}
Here's an example setup for a checkbox control:
===============================================
typedef struct CheckBoxData {
Data head;
const char *label;
bool *checked;
} CheckBoxData;
// called when the item is clicked (see checkbox())
void app_checkbox_handler(int item, UIevent event) {
// retrieve custom data (see checkbox())
CheckBoxData *data = (CheckBoxData *)uiGetHandle(item);
switch(event) {
default: break;
case UI_BUTTON0_DOWN: {
// toggle value
*data->checked = !(*data->checked);
} break;
}
}
// creates a checkbox control for a pointer to a boolean
int checkbox(const char *label, bool *checked) {
// create new ui item
int item = uiItem();
// set minimum size of wiget; horizontal size is dynamic, vertical is fixed
uiSetSize(item, 0, APP_WIDGET_HEIGHT);
// store some custom data with the checkbox that we use for rendering
// and value changes.
CheckBoxData *data = (CheckBoxData *)uiAllocHandle(item, sizeof(CheckBoxData));
// assign a custom typeid to the data so the renderer knows how to
// render this control, and our event handler
data->head.type = APP_WIDGET_CHECKBOX;
data->head.handler = app_checkbox_handler;
data->label = label;
data->checked = checked;
// set to fire as soon as the left button is
// pressed; UI_BUTTON0_HOT_UP is also a popular alternative.
uiSetEvents(item, UI_BUTTON0_DOWN);
return item;
}
A simple recursive drawing routine can look like this:
======================================================
void app_draw_ui(AppRenderContext *ctx, int item) {
// retrieve custom data and cast it to Data; we assume the first member
// of every widget data item to be a Data field.
Data *head = (Data *)uiGetHandle(item);
// if a handle is set, this is a specialized widget
if (head) {
// get the widgets absolute rectangle
UIrect rect = uiGetRect(item);
switch(head->type) {
default: break;
case APP_WIDGET_LABEL: {
// ...
} break;
case APP_WIDGET_BUTTON: {
// ...
} break;
case APP_WIDGET_CHECKBOX: {
// cast to the full data type
CheckBoxData *data = (CheckBoxData*)head;
// get the widgets current state
int state = uiGetState(item);
// if the value is set, the state is always active
if (*data->checked)
state = UI_ACTIVE;
// draw the checkbox
app_draw_checkbox(ctx, rect, state, data->label);
} break;
}
}
// iterate through all children and draw
int kid = uiFirstChild(item);
while (kid != -1) {
app_draw_ui(ctx, kid);
kid = uiNextSibling(kid);
}
}
Layouting items works like this:
================================
void layout_window(int w, int h) {
// create root item; the first item always has index 0
int parent = uiItem();
// assign fixed size
uiSetSize(parent, w, h);
// create column box and use as new parent
parent = uiInsert(parent, uiItem());
// configure as column
uiSetBox(parent, UI_COLUMN);
// span horizontally, attach to top
uiSetLayout(parent, UI_HFILL | UI_TOP);
// add a label - we're assuming custom control functions to exist
int item = uiInsert(parent, label("Hello World"));
// set a fixed height for the label
uiSetSize(item, 0, APP_WIDGET_HEIGHT);
// span the label horizontally
uiSetLayout(item, UI_HFILL);
static bool checked = false;
// add a checkbox to the same parent as item; this is faster than
// calling uiInsert on the same parent repeatedly.
item = uiAppend(item, checkbox("Checked:", &checked));
// set a fixed height for the checkbox
uiSetSize(item, 0, APP_WIDGET_HEIGHT);
// span the checkbox in the same way as the label
uiSetLayout(item, UI_HFILL);
}
*/
// you can override this from the outside to pick
// the export level you need
#ifndef OUI_EXPORT
#define OUI_EXPORT
#endif
// limits
enum {
// maximum size in bytes of a single data buffer passed to uiAllocData().
UI_MAX_DATASIZE = 4096,
// maximum depth of nested containers
UI_MAX_DEPTH = 64,
// maximum number of buffered input events
UI_MAX_INPUT_EVENTS = 64,
// consecutive click threshold in ms
UI_CLICK_THRESHOLD = 250,
};
typedef unsigned int UIuint;
// opaque UI context
typedef struct UIcontext UIcontext;
// item states as returned by uiGetState()
typedef enum UIitemState {
// the item is inactive
UI_COLD = 0,
// the item is inactive, but the cursor is hovering over this item
UI_HOT = 1,
// the item is toggled, activated, focused (depends on item kind)
UI_ACTIVE = 2,
// the item is unresponsive
UI_FROZEN = 3,
} UIitemState;
// container flags to pass to uiSetBox()
typedef enum UIboxFlags {
// flex-direction (bit 0+1)
// left to right
UI_ROW = 0x002,
// top to bottom
UI_COLUMN = 0x003,
// model (bit 1)
// free layout
UI_LAYOUT = 0x000,
// flex model
UI_FLEX = 0x002,
// flex-wrap (bit 2)
// single-line
UI_NOWRAP = 0x000,
// multi-line, wrap left to right
UI_WRAP = 0x004,
// justify-content (start, end, center, space-between)
// at start of row/column
UI_START = 0x008,
// at center of row/column
UI_MIDDLE = 0x000,
// at end of row/column
UI_END = 0x010,
// insert spacing to stretch across whole row/column
UI_JUSTIFY = 0x018,
// align-items
// can be implemented by putting a flex container in a layout container,
// then using UI_TOP, UI_DOWN, UI_VFILL, UI_VCENTER, etc.
// FILL is equivalent to stretch/grow
// align-content (start, end, center, stretch)
// can be implemented by putting a flex container in a layout container,
// then using UI_TOP, UI_DOWN, UI_VFILL, UI_VCENTER, etc.
// FILL is equivalent to stretch; space-between is not supported.
} UIboxFlags;
// child layout flags to pass to uiSetLayout()
typedef enum UIlayoutFlags {
// attachments (bit 5-8)
// fully valid when parent uses UI_LAYOUT model
// partially valid when in UI_FLEX model
// anchor to left item or left side of parent
UI_LEFT = 0x020,
// anchor to top item or top side of parent
UI_TOP = 0x040,
// anchor to right item or right side of parent
UI_RIGHT = 0x080,
// anchor to bottom item or bottom side of parent
UI_DOWN = 0x100,
// anchor to both left and right item or parent borders
UI_HFILL = 0x0a0,
// anchor to both top and bottom item or parent borders
UI_VFILL = 0x140,
// center horizontally, with left margin as offset
UI_HCENTER = 0x000,
// center vertically, with top margin as offset
UI_VCENTER = 0x000,
// center in both directions, with left/top margin as offset
UI_CENTER = 0x000,
// anchor to all four directions
UI_FILL = 0x1e0,
// when wrapping, put this element on a new line
// wrapping layout code auto-inserts UI_BREAK flags,
// drawing routines can read them with uiGetLayout()
UI_BREAK = 0x200
} UIlayoutFlags;
// event flags
typedef enum UIevent {
// on button 0 down
UI_BUTTON0_DOWN = 0x0400,
// on button 0 up
// when this event has a handler, uiGetState() will return UI_ACTIVE as
// long as button 0 is down.
UI_BUTTON0_UP = 0x0800,
// on button 0 up while item is hovered
// when this event has a handler, uiGetState() will return UI_ACTIVE
// when the cursor is hovering the items rectangle; this is the
// behavior expected for buttons.
UI_BUTTON0_HOT_UP = 0x1000,
// item is being captured (button 0 constantly pressed);
// when this event has a handler, uiGetState() will return UI_ACTIVE as
// long as button 0 is down.
UI_BUTTON0_CAPTURE = 0x2000,
// on button 2 down (right mouse button, usually triggers context menu)
UI_BUTTON2_DOWN = 0x4000,
// item has received a scrollwheel event
// the accumulated wheel offset can be queried with uiGetScroll()
UI_SCROLL = 0x8000,
// item is focused and has received a key-down event
// the respective key can be queried using uiGetKey() and uiGetModifier()
UI_KEY_DOWN = 0x10000,
// item is focused and has received a key-up event
// the respective key can be queried using uiGetKey() and uiGetModifier()
UI_KEY_UP = 0x20000,
// item is focused and has received a character event
// the respective character can be queried using uiGetKey()
UI_CHAR = 0x40000,
} UIevent;
enum {
// these bits, starting at bit 24, can be safely assigned by the
// application, e.g. as item types, other event types, drop targets, etc.
// they can be set and queried using uiSetFlags() and uiGetFlags()
UI_USERMASK = 0xff000000,
// a special mask passed to uiFindItem()
UI_ANY = 0xffffffff,
};
// handler callback; event is one of UI_EVENT_*
typedef void (*UIhandler)(UIcontext* ui_context, int item, UIevent event);
// for cursor positions, mainly
typedef struct UIvec2 {
union {
int v[2];
struct { int x, y; };
};
} UIvec2;
// layout rectangle
typedef struct UIrect {
union {
int v[4];
struct { int x, y, w, h; };
};
} UIrect;
// unless declared otherwise, all operations have the complexity O(1).
// Context Management
// ------------------
// create a new UI context; call uiMakeCurrent() to make this context the
// current context. The context is managed by the client and must be released
// using uiDestroyContext()
// item_capacity is the maximum of number of items that can be declared.
// buffer_capacity is the maximum total size of bytes that can be allocated
// using uiAllocHandle(); you may pass 0 if you don't need to allocate
// handles.
// 4096 and (1<<20) are good starting values.
OUI_EXPORT UIcontext *uiCreateContext(
unsigned int item_capacity,
unsigned int buffer_capacity);
// release the memory of an UI context created with uiCreateContext(); if the
// context is the current context, the current context will be set to NULL
OUI_EXPORT void uiDestroyContext(UIcontext *ctx);
// User Data
OUI_EXPORT void uiSetContextHandle(UIcontext *ui_context, void *handle);
OUI_EXPORT void *uiGetContextHandle(UIcontext *ui_context);
// Input Control
// -------------
// sets the current cursor position (usually belonging to a mouse) to the
// screen coordinates at (x,y)
OUI_EXPORT void uiSetCursor(UIcontext *ui_context, int x, int y);
// returns the current cursor position in screen coordinates as set by
// uiSetCursor()
OUI_EXPORT UIvec2 uiGetCursor(UIcontext *ui_context);
// returns the offset of the cursor relative to the last call to uiProcess()
OUI_EXPORT UIvec2 uiGetCursorDelta(UIcontext *ui_context);
// returns the beginning point of a drag operation.
OUI_EXPORT UIvec2 uiGetCursorStart(UIcontext *ui_context);
// returns the offset of the cursor relative to the beginning point of a drag
// operation.
OUI_EXPORT UIvec2 uiGetCursorStartDelta(UIcontext *ui_context);
// sets a mouse or gamepad button as pressed/released
// button is in the range 0..63 and maps to an application defined input
// source.
// mod is an application defined set of flags for modifier keys
// enabled is 1 for pressed, 0 for released
OUI_EXPORT void uiSetButton(UIcontext *ui_context, unsigned int button, unsigned int mod, bool enabled);
// returns the current state of an application dependent input button
// as set by uiSetButton().
// the function returns 1 if the button has been set to pressed, 0 for released.
OUI_EXPORT int uiGetButton(UIcontext *ui_context, unsigned int button);
// returns the number of chained clicks; 1 is a single click,
// 2 is a double click, etc.
OUI_EXPORT int uiGetClicks(UIcontext *ui_context);
// sets a key as down/up; the key can be any application defined keycode
// mod is an application defined set of flags for modifier keys
// enabled is 1 for key down, 0 for key up
// all key events are being buffered until the next call to uiProcess()
OUI_EXPORT void uiSetKey(UIcontext *ui_context, unsigned int key, unsigned int mod, bool enabled);
// sends a single character for text input; the character is usually in the
// unicode range, but can be application defined.
// all char events are being buffered until the next call to uiProcess()
OUI_EXPORT void uiSetChar(UIcontext *ui_context, unsigned int value);
// accumulates scroll wheel offsets for the current frame
// all offsets are being accumulated until the next call to uiProcess()
OUI_EXPORT void uiSetScroll(UIcontext *ui_context, int x, int y);
// returns the currently accumulated scroll wheel offsets for this frame
OUI_EXPORT UIvec2 uiGetScroll(UIcontext *ui_context);
// Stages
// ------
// clear the item buffer; uiBeginLayout() should be called before the first
// UI declaration for this frame to avoid concatenation of the same UI multiple
// times.
// After the call, all previously declared item IDs are invalid, and all
// application dependent context data has been freed.
// uiBeginLayout() must be followed by uiEndLayout().
OUI_EXPORT void uiBeginLayout(UIcontext *ui_context);
// layout all added items starting from the root item 0.
// after calling uiEndLayout(), no further modifications to the item tree should
// be done until the next call to uiBeginLayout().
// It is safe to immediately draw the items after a call to uiEndLayout().
// this is an O(N) operation for N = number of declared items.
OUI_EXPORT void uiEndLayout(UIcontext *ui_context);
// update the current hot item; this only needs to be called if items are kept
// for more than one frame and uiEndLayout() is not called
OUI_EXPORT void uiUpdateHotItem(UIcontext *ui_context);
// update the internal state according to the current cursor position and
// button states, and call all registered handlers.
// timestamp is the time in milliseconds relative to the last call to uiProcess()
// and is used to estimate the threshold for double-clicks
// after calling uiProcess(), no further modifications to the item tree should
// be done until the next call to uiBeginLayout().
// Items should be drawn before a call to uiProcess()
// this is an O(N) operation for N = number of declared items.
OUI_EXPORT void uiProcess(UIcontext *ui_context, int timestamp);
// reset the currently stored hot/active etc. handles; this should be called when
// a re-declaration of the UI changes the item indices, to avoid state
// related glitches because item identities have changed.
OUI_EXPORT void uiClearState(UIcontext *ui_context);
// UI Declaration
// --------------
// create a new UI item and return the new items ID.
OUI_EXPORT int uiItem(UIcontext *ui_context);
// set an items state to frozen; the UI will not recurse into frozen items
// when searching for hot or active items; subsequently, frozen items and
// their child items will not cause mouse event notifications.
// The frozen state is not applied recursively; uiGetState() will report
// UI_COLD for child items. Upon encountering a frozen item, the drawing
// routine needs to handle rendering of child items appropriately.
// see example.cpp for a demonstration.
OUI_EXPORT void uiSetFrozen(UIcontext *ui_context, int item, bool enable);
// set the application-dependent handle of an item.
// handle is an application defined 64-bit handle. If handle is NULL, the item
// will not be interactive.
OUI_EXPORT void uiSetHandle(UIcontext *ui_context, int item, void *handle);
// allocate space for application-dependent context data and assign it
// as the handle to the item.
// The memory of the pointer is managed by the UI context and released
// upon the next call to uiBeginLayout()
OUI_EXPORT void *uiAllocHandle(UIcontext *ui_context, int item, unsigned int size);
// set the global handler callback for interactive items.
// the handler will be called for each item whose event flags are set using
// uiSetEvents.
OUI_EXPORT void uiSetHandler(UIcontext *ui_context, UIhandler handler);
// flags is a combination of UI_EVENT_* and designates for which events the
// handler should be called.
OUI_EXPORT void uiSetEvents(UIcontext *ui_context, int item, unsigned int flags);
// flags is a user-defined set of flags defined by UI_USERMASK.
OUI_EXPORT void uiSetFlags(UIcontext *ui_context, int item, unsigned int flags);
// assign an item to a container.
// an item ID of 0 refers to the root item.
// the function returns the child item ID
// if the container has already added items, the function searches
// for the last item and calls uiAppend() on it, which is an
// O(N) operation for N siblings.
// it is usually more efficient to call uiInsert() for the first child,
// then chain additional siblings using uiAppend().
OUI_EXPORT int uiInsert(UIcontext *ui_context, int item, int child);
// assign an item to the same container as another item
// sibling is inserted after item.
OUI_EXPORT int uiAppend(UIcontext *ui_context, int item, int sibling);
// insert child into container item like uiInsert(), but prepend
// it to the first child item, effectively putting it in
// the background.
// it is efficient to call uiInsertBack() repeatedly
// in cases where drawing or layout order doesn't matter.
OUI_EXPORT int uiInsertBack(UIcontext *ui_context, int item, int child);
// same as uiInsert()
OUI_EXPORT int uiInsertFront(UIcontext *ui_context, int item, int child);
// set the size of the item; a size of 0 indicates the dimension to be
// dynamic; if the size is set, the item can not expand beyond that size.
OUI_EXPORT void uiSetSize(UIcontext *ui_context, int item, int w, int h);
// set the anchoring behavior of the item to one or multiple UIlayoutFlags
OUI_EXPORT void uiSetLayout(UIcontext *ui_context, int item, unsigned int flags);
// set the box model behavior of the item to one or multiple UIboxFlags
OUI_EXPORT void uiSetBox(UIcontext *ui_context, int item, unsigned int flags);
// set the left, top, right and bottom margins of an item; when the item is
// anchored to the parent or another item, the margin controls the distance
// from the neighboring element.
OUI_EXPORT void uiSetMargins(UIcontext *ui_context, int item, short l, short t, short r, short b);
// set item as recipient of all keyboard events; if item is -1, no item will
// be focused.
OUI_EXPORT void uiFocus(UIcontext *ui_context, int item);
// Iteration
// ---------
// returns the first child item of a container item. If the item is not
// a container or does not contain any items, -1 is returned.
// if item is 0, the first child item of the root item will be returned.
OUI_EXPORT int uiFirstChild(UIcontext *ui_context, int item);
// returns an items next sibling in the list of the parent containers children.
// if item is 0 or the item is the last child item, -1 will be returned.
OUI_EXPORT int uiNextSibling(UIcontext *ui_context, int item);
// Querying
// --------
// return the total number of allocated items
OUI_EXPORT int uiGetItemCount(UIcontext *ui_context);
// return the total bytes that have been allocated by uiAllocHandle()
OUI_EXPORT unsigned int uiGetAllocSize(UIcontext *ui_context);
// return the current state of the item. This state is only valid after
// a call to uiProcess().
// The returned value is one of UI_COLD, UI_HOT, UI_ACTIVE, UI_FROZEN.
OUI_EXPORT UIitemState uiGetState(UIcontext *ui_context, int item);
// return the application-dependent handle of the item as passed to uiSetHandle()
// or uiAllocHandle().
OUI_EXPORT void *uiGetHandle(UIcontext *ui_context, int item);
// return the item that is currently under the cursor or -1 for none
OUI_EXPORT int uiGetHotItem(UIcontext *ui_context);
// return the item that is currently focused or -1 for none
OUI_EXPORT int uiGetFocusedItem(UIcontext *ui_context);
// returns the topmost item containing absolute location (x,y), starting with
// item as parent, using a set of flags and masks as filter:
// if both flags and mask are UI_ANY, the first topmost item is returned.
// if mask is UI_ANY, the first topmost item matching *any* of flags is returned.
// otherwise the first item matching (item.flags & flags) == mask is returned.
// you may combine box, layout, event and user flags.
// frozen items will always be ignored.
OUI_EXPORT int uiFindItem(UIcontext *ui_context, int item, int x, int y,
unsigned int flags, unsigned int mask);
// return the handler callback as passed to uiSetHandler()
OUI_EXPORT UIhandler uiGetHandler(UIcontext *ui_context);
// return the event flags for an item as passed to uiSetEvents()
OUI_EXPORT unsigned int uiGetEvents(UIcontext *ui_context, int item);
// return the user-defined flags for an item as passed to uiSetFlags()
OUI_EXPORT unsigned int uiGetFlags(UIcontext *ui_context, int item);
// when handling a KEY_DOWN/KEY_UP event: the key that triggered this event
OUI_EXPORT unsigned int uiGetKey(UIcontext *ui_context);
// when handling a keyboard or mouse event: the active modifier keys
OUI_EXPORT unsigned int uiGetModifier(UIcontext *ui_context);
// returns the items layout rectangle in absolute coordinates. If
// uiGetRect() is called before uiEndLayout(), the values of the returned
// rectangle are undefined.
OUI_EXPORT UIrect uiGetRect(UIcontext *ui_context, int item);
// returns 1 if an items absolute rectangle contains a given coordinate
// otherwise 0
OUI_EXPORT int uiContains(UIcontext *ui_context, int item, int x, int y);
// return the width of the item as set by uiSetSize()
OUI_EXPORT int uiGetWidth(UIcontext *ui_context, int item);
// return the height of the item as set by uiSetSize()
OUI_EXPORT int uiGetHeight(UIcontext *ui_context, int item);
// return the anchoring behavior as set by uiSetLayout()
OUI_EXPORT unsigned int uiGetLayout(UIcontext *ui_context, int item);
// return the box model as set by uiSetBox()
OUI_EXPORT unsigned int uiGetBox(UIcontext *ui_context, int item);
// return the left margin of the item as set with uiSetMargins()
OUI_EXPORT short uiGetMarginLeft(UIcontext *ui_context, int item);
// return the top margin of the item as set with uiSetMargins()
OUI_EXPORT short uiGetMarginTop(UIcontext *ui_context, int item);
// return the right margin of the item as set with uiSetMargins()
OUI_EXPORT short uiGetMarginRight(UIcontext *ui_context, int item);
// return the bottom margin of the item as set with uiSetMargins()
OUI_EXPORT short uiGetMarginDown(UIcontext *ui_context, int item);
// when uiBeginLayout() is called, the most recently declared items are retained.
// when uiEndLayout() completes, it matches the old item hierarchy to the new one
// and attempts to map old items to new items as well as possible.
// when passed an item Id from the previous frame, uiRecoverItem() returns the
// items new assumed Id, or -1 if the item could not be mapped.
// it is valid to pass -1 as item.
OUI_EXPORT int uiRecoverItem(UIcontext *ui_context, int olditem);
// in cases where it is important to recover old state over changes in
// the view, and the built-in remapping fails, the UI declaration can manually
// remap old items to new IDs in cases where e.g. the previous item ID has been
// temporarily saved; uiRemapItem() would then be called after creating the
// new item using uiItem().
OUI_EXPORT void uiRemapItem(UIcontext *ui_context, int olditem, int newitem);
// returns the number if items that have been allocated in the last frame
OUI_EXPORT int uiGetLastItemCount(UIcontext *ui_context);
#ifdef __cplusplus
};
#endif
#endif // _OUI_H_
#ifdef OUI_IMPLEMENTATION
#include <assert.h>
#ifdef _MSC_VER
#pragma warning (disable: 4996) // Switch off security warnings
#pragma warning (disable: 4100) // Switch off unreferenced formal parameter warnings
#pragma warning (disable: 4244)
#pragma warning (disable: 4305)
#ifdef __cplusplus
#define UI_INLINE inline
#else
#define UI_INLINE
#endif
#else
#ifdef __clang__
#define UI_INLINE static inline
#else
#define UI_INLINE inline
#endif
#endif
#define UI_MAX_KIND 16
#define UI_ANY_BUTTON0_INPUT (UI_BUTTON0_DOWN \
|UI_BUTTON0_UP \
|UI_BUTTON0_HOT_UP \
|UI_BUTTON0_CAPTURE)
#define UI_ANY_BUTTON2_INPUT (UI_BUTTON2_DOWN)
#define UI_ANY_MOUSE_INPUT (UI_ANY_BUTTON0_INPUT \
|UI_ANY_BUTTON2_INPUT)
#define UI_ANY_KEY_INPUT (UI_KEY_DOWN \
|UI_KEY_UP \
|UI_CHAR)
#define UI_ANY_INPUT (UI_ANY_MOUSE_INPUT \
|UI_ANY_KEY_INPUT)
enum {
// extra item flags
// bit 0-2
UI_ITEM_BOX_MODEL_MASK = 0x000007,
// bit 0-4
UI_ITEM_BOX_MASK = 0x00001F,
// bit 5-8
UI_ITEM_LAYOUT_MASK = 0x0003E0,
// bit 9-18
UI_ITEM_EVENT_MASK = 0x07FC00,
// item is frozen (bit 19)
UI_ITEM_FROZEN = 0x080000,
// item handle is pointer to data (bit 20)
UI_ITEM_DATA = 0x100000,
// item has been inserted (bit 21)
UI_ITEM_INSERTED = 0x200000,
// horizontal size has been explicitly set (bit 22)
UI_ITEM_HFIXED = 0x400000,
// vertical size has been explicitly set (bit 23)
UI_ITEM_VFIXED = 0x800000,
// bit 22-23
UI_ITEM_FIXED_MASK = 0xC00000,
// which flag bits will be compared
UI_ITEM_COMPARE_MASK = UI_ITEM_BOX_MODEL_MASK
| (UI_ITEM_LAYOUT_MASK & ~UI_BREAK)
| UI_ITEM_EVENT_MASK
| UI_USERMASK,
};
typedef struct UIitem {
// data handle
void *handle;
// about 27 bits worth of flags
unsigned int flags;
// index of first kid
// if old item: index of equivalent new item
int firstkid;
// index of next sibling with same parent
int nextitem;
// margin offsets, interpretation depends on flags
// after layouting, the first two components are absolute coordinates
short margins[4];
// size
short size[2];
} UIitem;
typedef enum UIstate {
UI_STATE_IDLE = 0,
UI_STATE_CAPTURE,
} UIstate;
typedef enum UIstage {
UI_STAGE_LAYOUT = 0,
UI_STAGE_POST_LAYOUT,
UI_STAGE_PROCESS,
} UIstage;
typedef struct UIhandleEntry {
unsigned int key;
int item;
} UIhandleEntry;
typedef struct UIinputEvent {
unsigned int key;
unsigned int mod;
UIevent event;
} UIinputEvent;
struct UIcontext {
unsigned int item_capacity;
unsigned int buffer_capacity;
// handler
UIhandler handler;
// User data
void *handle;
// button state in this frame
unsigned long long buttons;
// button state in the previous frame
unsigned long long last_buttons;
// where the cursor was at the beginning of the active state
UIvec2 start_cursor;
// where the cursor was last frame
UIvec2 last_cursor;
// where the cursor is currently
UIvec2 cursor;
// accumulated scroll wheel offsets
UIvec2 scroll;
int active_item;
int focus_item;
int last_hot_item;
int last_click_item;
int hot_item;
UIstate state;
UIstage stage;
unsigned int active_key;
unsigned int active_modifier;
unsigned int active_button_modifier;
int last_timestamp;
int last_click_timestamp;
int clicks;
int count;
int last_count;
int eventcount;
unsigned int datasize;
UIitem *items;
unsigned char *data;
UIitem *last_items;
int *item_map;
UIinputEvent events[UI_MAX_INPUT_EVENTS];
};
UI_INLINE int ui_max(int a, int b) {
return (a>b)?a:b;
}
UI_INLINE int ui_min(int a, int b) {
return (a<b)?a:b;
}
UI_INLINE float ui_maxf(float a, float b) {
return (a>b)?a:b;
}
UI_INLINE float ui_minf(float a, float b) {
return (a<b)?a:b;
}
void uiClear(UIcontext *ui_context) {
int i;
ui_context->last_count = ui_context->count;
ui_context->count = 0;
ui_context->datasize = 0;
ui_context->hot_item = -1;
// swap buffers
UIitem *items = ui_context->items;
ui_context->items = ui_context->last_items;
ui_context->last_items = items;
for (i = 0; i < ui_context->last_count; ++i) {
ui_context->item_map[i] = -1;
}
}
static UIcontext *uiInitializeContext(
UIcontext *ctx,
unsigned int item_capacity,
unsigned int buffer_capacity) {
memset(ctx, 0, sizeof(UIcontext));
ctx->item_capacity = item_capacity;
ctx->buffer_capacity = buffer_capacity;
ctx->stage = UI_STAGE_PROCESS;
ctx->items = (UIitem *)malloc(sizeof(UIitem) * item_capacity);
ctx->last_items = (UIitem *)malloc(sizeof(UIitem) * item_capacity);
ctx->item_map = (int *)malloc(sizeof(int) * item_capacity);
if (buffer_capacity) {
ctx->data = (unsigned char *)malloc(buffer_capacity);
}
return ctx;
}
UIcontext *uiCreateContext(
unsigned int item_capacity,
unsigned int buffer_capacity) {
assert(item_capacity);
UIcontext *ctx = (UIcontext *)malloc(sizeof(UIcontext));
uiInitializeContext(ctx, item_capacity, buffer_capacity);
uiClear(ctx);
uiClearState(ctx);
return ctx;
}
void uiDestroyContext(UIcontext *ctx) {
free(ctx->items);
free(ctx->last_items);
free(ctx->item_map);
free(ctx->data);
free(ctx);
}
void uiSetContextHandle(UIcontext *ui_context, void *handle) {
assert(ui_context);
ui_context->handle = handle;
}
void *uiGetContextHandle(UIcontext *ui_context) {
assert(ui_context);
return ui_context->handle;
}
void uiSetButton(UIcontext *ui_context, unsigned int button, unsigned int mod, bool enabled) {
assert(ui_context);
unsigned long long mask = 1ull<<button;
// set new bit
ui_context->buttons = (enabled)?
(ui_context->buttons | mask):
(ui_context->buttons & ~mask);
ui_context->active_button_modifier = mod;
}
static void uiAddInputEvent(UIcontext *ui_context, UIinputEvent event) {
assert(ui_context);
if (ui_context->eventcount == UI_MAX_INPUT_EVENTS) return;
ui_context->events[ui_context->eventcount++] = event;
}
static void uiClearInputEvents(UIcontext *ui_context) {
assert(ui_context);
ui_context->eventcount = 0;
ui_context->scroll.x = 0;
ui_context->scroll.y = 0;
}