2024-09-20 08:46:48 +00:00
// VERSION: 0.11
2024-08-23 04:05:23 +00:00
2024-09-20 08:55:45 +00:00
/*
NOTE : In order to use this library you must define
the following macro in exactly one file , _before_ including clay . h :
# define CLAY_IMPLEMENTATION
# include "clay.h"
See the examples folder for details .
*/
2024-12-30 00:09:14 +00:00
# include <stdint.h>
# include <stdbool.h>
# include <stddef.h>
2024-08-23 04:05:23 +00:00
2024-09-20 08:46:48 +00:00
// -----------------------------------------
// HEADER DECLARATIONS ---------------------
// -----------------------------------------
2024-08-23 04:05:23 +00:00
2024-09-20 08:46:48 +00:00
# ifndef CLAY_HEADER
# define CLAY_HEADER
2024-08-23 04:05:23 +00:00
2024-09-20 20:00:55 +00:00
# ifdef CLAY_WASM
# define CLAY_WASM_EXPORT(name) __attribute__((export_name(name)))
# else
# define CLAY_WASM_EXPORT(null)
# endif
2024-09-20 08:46:48 +00:00
// Public Macro API ------------------------
2024-08-23 04:05:23 +00:00
2024-10-22 07:41:35 +00:00
# ifdef __cplusplus
# define CLAY__CONFIG_WRAPPER(type, ...) __VA_ARGS__
# else
# define CLAY__CONFIG_WRAPPER(type, ...) (type) __VA_ARGS__
# endif
2024-12-18 20:29:12 +00:00
# define CLAY__MAX(x, y) (((x) > (y)) ? (x) : (y))
# define CLAY__MIN(x, y) (((x) < (y)) ? (x) : (y))
2024-10-22 07:41:35 +00:00
# define CLAY_LAYOUT(...) Clay__AttachLayoutConfig(Clay__StoreLayoutConfig(CLAY__CONFIG_WRAPPER(Clay_LayoutConfig, __VA_ARGS__)))
2024-08-23 04:05:23 +00:00
2024-10-22 07:41:35 +00:00
# define CLAY_RECTANGLE(...) Clay__AttachElementConfig(CLAY__CONFIG_WRAPPER(Clay_ElementConfigUnion, { .rectangleElementConfig = Clay__StoreRectangleElementConfig(CLAY__INIT(Clay_RectangleElementConfig) __VA_ARGS__) }, CLAY__ELEMENT_CONFIG_TYPE_RECTANGLE))
2024-08-23 04:05:23 +00:00
2024-10-22 07:41:35 +00:00
# define CLAY_TEXT_CONFIG(...) Clay__StoreTextElementConfig(CLAY__CONFIG_WRAPPER(Clay_TextElementConfig, __VA_ARGS__))
2024-08-23 04:05:23 +00:00
2024-10-22 07:41:35 +00:00
# define CLAY_IMAGE(...) Clay__AttachElementConfig(CLAY__CONFIG_WRAPPER(Clay_ElementConfigUnion, { .imageElementConfig = Clay__StoreImageElementConfig(CLAY__INIT(Clay_ImageElementConfig) __VA_ARGS__) }, CLAY__ELEMENT_CONFIG_TYPE_IMAGE))
2024-08-23 04:05:23 +00:00
2024-10-22 07:41:35 +00:00
# define CLAY_FLOATING(...) Clay__AttachElementConfig(CLAY__CONFIG_WRAPPER(Clay_ElementConfigUnion, { .floatingElementConfig = Clay__StoreFloatingElementConfig(CLAY__INIT(Clay_FloatingElementConfig) __VA_ARGS__) }, CLAY__ELEMENT_CONFIG_TYPE_FLOATING_CONTAINER))
2024-08-23 04:05:23 +00:00
2024-10-22 07:41:35 +00:00
# define CLAY_CUSTOM_ELEMENT(...) Clay__AttachElementConfig(CLAY__CONFIG_WRAPPER(Clay_ElementConfigUnion, { .customElementConfig = Clay__StoreCustomElementConfig(CLAY__INIT(Clay_CustomElementConfig) __VA_ARGS__) }, CLAY__ELEMENT_CONFIG_TYPE_CUSTOM))
2024-08-23 04:05:23 +00:00
2024-10-22 07:41:35 +00:00
# define CLAY_SCROLL(...) Clay__AttachElementConfig(CLAY__CONFIG_WRAPPER(Clay_ElementConfigUnion, { .scrollElementConfig = Clay__StoreScrollElementConfig(CLAY__INIT(Clay_ScrollElementConfig) __VA_ARGS__) }, CLAY__ELEMENT_CONFIG_TYPE_SCROLL_CONTAINER))
2024-08-23 04:05:23 +00:00
2024-10-22 07:41:35 +00:00
# define CLAY_BORDER(...) Clay__AttachElementConfig(CLAY__CONFIG_WRAPPER(Clay_ElementConfigUnion, { .borderElementConfig = Clay__StoreBorderElementConfig(CLAY__INIT(Clay_BorderElementConfig) __VA_ARGS__) }, CLAY__ELEMENT_CONFIG_TYPE_BORDER_CONTAINER))
2024-08-23 04:05:23 +00:00
2024-10-22 07:41:35 +00:00
# define CLAY_BORDER_OUTSIDE(...) Clay__AttachElementConfig(CLAY__CONFIG_WRAPPER(Clay_ElementConfigUnion, { .borderElementConfig = Clay__StoreBorderElementConfig(CLAY__INIT(Clay_BorderElementConfig) { .left = __VA_ARGS__, .right = __VA_ARGS__, .top = __VA_ARGS__, .bottom = __VA_ARGS__ }) }, CLAY__ELEMENT_CONFIG_TYPE_BORDER_CONTAINER))
2024-08-23 04:05:23 +00:00
2024-10-22 07:41:35 +00:00
# define CLAY_BORDER_OUTSIDE_RADIUS(width, color, radius) Clay__AttachElementConfig(CLAY__CONFIG_WRAPPER(Clay_ElementConfigUnion, { .borderElementConfig = Clay__StoreBorderElementConfig(CLAY__INIT(Clay_BorderElementConfig) { .left = { width, color }, .right = { width, color }, .top = { width, color }, .bottom = { width, color }, .cornerRadius = { radius, radius, radius, radius } })}, CLAY__ELEMENT_CONFIG_TYPE_BORDER_CONTAINER))
2024-08-23 04:05:23 +00:00
2024-10-22 07:41:35 +00:00
# define CLAY_BORDER_ALL(...) Clay__AttachElementConfig(CLAY__CONFIG_WRAPPER(Clay_ElementConfigUnion, { .borderElementConfig = Clay__StoreBorderElementConfig(CLAY__INIT(Clay_BorderElementConfig) { .left = __VA_ARGS__, .right = __VA_ARGS__, .top = __VA_ARGS__, .bottom = __VA_ARGS__, .betweenChildren = __VA_ARGS__ }) }, CLAY__ELEMENT_CONFIG_TYPE_BORDER_CONTAINER))
2024-08-23 04:05:23 +00:00
2024-10-22 07:41:35 +00:00
# define CLAY_BORDER_ALL_RADIUS(width, color, radius) Clay__AttachElementConfig(CLAY__CONFIG_WRAPPER(Clay_ElementConfigUnion, { .borderElementConfig = Clay__StoreBorderElementConfig(CLAY__INIT(Clay_BorderElementConfig) { .left = { width, color }, .right = { width, color }, .top = { width, color }, .bottom = { width, color }, .betweenChildren = { width, color }, .cornerRadius = { radius, radius, radius, radius }}) }))
2024-08-23 04:05:23 +00:00
2024-12-30 00:09:14 +00:00
# define CLAY_CORNER_RADIUS(radius) (CLAY__INIT(Clay_CornerRadius) { radius, radius, radius, radius })
2024-08-23 04:05:23 +00:00
2024-10-22 07:41:35 +00:00
# define CLAY__STRUCT_1_ARGS(a) a
2024-12-30 00:09:14 +00:00
# define CLAY__STRUCT_0_ARGS() {0}
2024-10-22 07:41:35 +00:00
# define CLAY__STRUCT_OVERRIDE(_0, _1, NAME, ...) NAME
2024-08-23 04:05:23 +00:00
2024-12-30 00:09:14 +00:00
# define CLAY__SIZING_FIT_INTERNAL(...) (CLAY__INIT(Clay_SizingAxis) { .size = { .minMax = __VA_ARGS__ }, .type = CLAY__SIZING_TYPE_FIT })
2024-10-22 07:41:35 +00:00
# define CLAY_SIZING_FIT(...) CLAY__SIZING_FIT_INTERNAL(CLAY__STRUCT_OVERRIDE("empty", ##__VA_ARGS__, CLAY__STRUCT_1_ARGS, CLAY__STRUCT_0_ARGS)(__VA_ARGS__))
2024-08-23 04:05:23 +00:00
2024-12-30 00:09:14 +00:00
# define CLAY__SIZING_GROW_INTERNAL(...) (CLAY__INIT(Clay_SizingAxis) { .size = { .minMax = __VA_ARGS__ }, .type = CLAY__SIZING_TYPE_GROW })
2024-10-22 07:41:35 +00:00
# define CLAY_SIZING_GROW(...) CLAY__SIZING_GROW_INTERNAL(CLAY__STRUCT_OVERRIDE("empty", ##__VA_ARGS__, CLAY__STRUCT_1_ARGS, CLAY__STRUCT_0_ARGS)(__VA_ARGS__))
2024-12-30 00:09:14 +00:00
# define CLAY_SIZING_FIXED(fixedSize) (CLAY__INIT(Clay_SizingAxis) { .size = { .minMax = { fixedSize, fixedSize } }, .type = CLAY__SIZING_TYPE_FIXED })
2024-08-23 04:05:23 +00:00
2024-12-30 00:09:14 +00:00
# define CLAY_SIZING_PERCENT(percentOfParent) (CLAY__INIT(Clay_SizingAxis) { .size = { .percent = (percentOfParent) }, .type = CLAY__SIZING_TYPE_PERCENT })
2024-08-23 04:05:23 +00:00
2024-10-22 07:41:35 +00:00
# define CLAY_ID(label) Clay__AttachId(Clay__HashString(CLAY_STRING(label), 0, 0))
2024-08-23 04:05:23 +00:00
2024-10-22 07:41:35 +00:00
# define CLAY_IDI(label, index) Clay__AttachId(Clay__HashString(CLAY_STRING(label), index, 0))
2024-08-23 04:05:23 +00:00
2024-09-25 02:04:28 +00:00
# define CLAY_ID_LOCAL(label) CLAY_IDI_LOCAL(label, 0)
2024-10-22 07:41:35 +00:00
# define CLAY_IDI_LOCAL(label, index) Clay__AttachId(Clay__HashString(CLAY_STRING(label), Clay_LayoutElementArray_Get(&Clay__layoutElements, Clay__int32_tArray_Get(&Clay__openLayoutElementStack, Clay__openLayoutElementStack.length - 2))->children.length + 1, Clay__GetOpenLayoutElement()->id))
2024-09-16 09:34:59 +00:00
2024-09-20 08:46:48 +00:00
# define CLAY__STRING_LENGTH(s) ((sizeof(s) / sizeof((s)[0])) - sizeof((s)[0]))
2024-08-23 04:05:23 +00:00
2024-12-30 00:09:14 +00:00
# define CLAY_STRING(string) (CLAY__INIT(Clay_String) { .length = CLAY__STRING_LENGTH(string), .chars = (string) })
2024-08-23 04:05:23 +00:00
2025-01-02 21:02:58 +00:00
static uint8_t CLAY__ELEMENT_DEFINITION_LATCH ;
2024-09-28 03:43:29 +00:00
2025-01-02 21:02:58 +00:00
// Publicly visible layout element macros -----------------------------------------------------
2025-01-02 21:19:59 +00:00
/* This macro looks scary on the surface, but is actually quite simple.
It turns a macro call like this :
CLAY (
CLAY_RECTANGLE ( ) ,
CLAY_ID ( )
) {
. . . children declared here
}
Into calls like this :
Clay_OpenElement ( ) ;
CLAY_RECTANGLE ( ) ;
CLAY_ID ( ) ;
Clay_ElementPostConfiguration ( ) ;
. . . children declared here
Clay_CloseElement ( ) ;
The for loop will only ever run a single iteration , putting Clay__CloseElement ( ) in the increment of the loop
means that it will run after the body - where the children are declared . It just exists to make sure you don ' t forget
to call Clay_CloseElement ( ) .
*/
2025-01-02 21:02:58 +00:00
# define CLAY(...) \
2024-09-28 03:43:29 +00:00
for ( \
2025-01-02 21:02:58 +00:00
CLAY__ELEMENT_DEFINITION_LATCH = ( Clay__OpenElement ( ) , # # __VA_ARGS__ , Clay__ElementPostConfiguration ( ) , 0 ) ; \
2024-09-28 03:43:29 +00:00
CLAY__ELEMENT_DEFINITION_LATCH < 1 ; \
2024-10-22 07:41:35 +00:00
+ + CLAY__ELEMENT_DEFINITION_LATCH , Clay__CloseElement ( ) \
2024-09-28 03:43:29 +00:00
)
2024-10-22 07:41:35 +00:00
# define CLAY_TEXT(text, textConfig) Clay__OpenTextElement(text, textConfig)
2024-09-20 08:46:48 +00:00
2024-09-28 23:53:46 +00:00
# ifdef __cplusplus
# define CLAY__INIT(type) type
2024-12-30 00:09:14 +00:00
# define CLAY__TYPEDEF(name, ...) typedef __VA_ARGS__ name
2024-09-28 23:53:46 +00:00
# define CLAY__ALIGNMENT(type) alignof(type)
2024-12-30 00:09:14 +00:00
# define CLAY__POINTER_ALIGNMENT alignof(void *)
# define CLAY_PACKED_ENUM enum : uint8_t
2024-09-28 23:53:46 +00:00
# else
2024-12-30 00:09:14 +00:00
2024-09-28 23:53:46 +00:00
# define CLAY__INIT(type) (type)
2024-12-30 00:09:14 +00:00
# define CLAY__ALIGNMENT_STRUCT(type) struct Clay__Align##type { char c; type x; }
# define CLAY__TYPEDEF(name, ...) typedef __VA_ARGS__ name; CLAY__ALIGNMENT_STRUCT(name)
# define CLAY__ALIGNMENT(type) (offsetof(struct Clay__Align##type, x))
# define CLAY__POINTER_ALIGNMENT CLAY__ALIGNMENT(pointer)
// NOTE: If you need to get the offset for other standard types in the future, add them here.
struct Clay__Alignpointer { char c ; void * x ; } ;
CLAY__ALIGNMENT_STRUCT ( bool ) ;
CLAY__ALIGNMENT_STRUCT ( uint8_t ) ;
CLAY__ALIGNMENT_STRUCT ( int32_t ) ;
2024-12-31 04:51:18 +00:00
# if defined(_MSC_VER) && !defined(__clang__)
2024-12-30 00:09:14 +00:00
# define CLAY_PACKED_ENUM __pragma(pack(push, 1)) enum __pragma(pack(pop))
# else
# define CLAY_PACKED_ENUM enum __attribute__((__packed__))
# endif
2024-09-28 23:53:46 +00:00
# endif
2024-09-20 08:46:48 +00:00
# ifdef __cplusplus
extern " C " {
# endif
// Utility Structs -------------------------
// Note: Clay_String is not guaranteed to be null terminated. It may be if created from a literal C string,
// but it is also used to represent slices.
2024-12-30 00:09:14 +00:00
CLAY__TYPEDEF ( Clay_String , struct {
size_t length ;
2024-09-20 08:46:48 +00:00
const char * chars ;
2024-12-30 00:09:14 +00:00
} ) ;
2024-09-20 08:46:48 +00:00
2024-12-30 00:09:14 +00:00
CLAY__TYPEDEF ( Clay__StringArray , struct {
uint32_t capacity ;
uint32_t length ;
Clay_String * internalArray ;
} ) ;
2024-10-22 07:41:35 +00:00
2024-12-30 00:09:14 +00:00
CLAY__TYPEDEF ( Clay_Arena , struct {
2024-12-26 06:58:59 +00:00
uintptr_t nextAllocation ;
size_t capacity ;
2024-09-20 08:46:48 +00:00
char * memory ;
2024-12-30 00:09:14 +00:00
} ) ;
2024-09-20 08:46:48 +00:00
2024-12-30 00:09:14 +00:00
CLAY__TYPEDEF ( Clay_Dimensions , struct {
2024-09-20 08:46:48 +00:00
float width , height ;
2024-12-30 00:09:14 +00:00
} ) ;
2024-09-20 08:46:48 +00:00
2024-12-30 00:09:14 +00:00
CLAY__TYPEDEF ( Clay_Vector2 , struct {
2024-09-20 08:46:48 +00:00
float x , y ;
2024-12-30 00:09:14 +00:00
} ) ;
2024-09-20 08:46:48 +00:00
2024-12-30 00:09:14 +00:00
CLAY__TYPEDEF ( Clay_Color , struct {
2024-09-20 08:46:48 +00:00
float r , g , b , a ;
2024-12-30 00:09:14 +00:00
} ) ;
2024-09-20 08:46:48 +00:00
2024-12-30 00:09:14 +00:00
CLAY__TYPEDEF ( Clay_BoundingBox , struct {
2024-09-20 08:46:48 +00:00
float x , y , width , height ;
2024-12-30 00:09:14 +00:00
} ) ;
2024-09-20 08:46:48 +00:00
// baseId + offset = id
2024-12-30 00:09:14 +00:00
CLAY__TYPEDEF ( Clay_ElementId , struct {
2024-09-20 08:46:48 +00:00
uint32_t id ;
uint32_t offset ;
uint32_t baseId ;
Clay_String stringId ;
2024-12-30 00:09:14 +00:00
} ) ;
2024-09-20 08:46:48 +00:00
2024-12-30 00:09:14 +00:00
CLAY__TYPEDEF ( Clay_CornerRadius , struct {
2024-09-20 08:46:48 +00:00
float topLeft ;
float topRight ;
float bottomLeft ;
float bottomRight ;
2024-12-30 00:09:14 +00:00
} ) ;
2024-09-20 08:46:48 +00:00
2024-12-30 00:09:14 +00:00
CLAY__TYPEDEF ( Clay__ElementConfigType , CLAY_PACKED_ENUM {
CLAY__ELEMENT_CONFIG_TYPE_NONE = 0 ,
2024-10-22 07:41:35 +00:00
CLAY__ELEMENT_CONFIG_TYPE_RECTANGLE = 1 ,
CLAY__ELEMENT_CONFIG_TYPE_BORDER_CONTAINER = 2 ,
CLAY__ELEMENT_CONFIG_TYPE_FLOATING_CONTAINER = 4 ,
CLAY__ELEMENT_CONFIG_TYPE_SCROLL_CONTAINER = 8 ,
CLAY__ELEMENT_CONFIG_TYPE_IMAGE = 16 ,
CLAY__ELEMENT_CONFIG_TYPE_TEXT = 32 ,
CLAY__ELEMENT_CONFIG_TYPE_CUSTOM = 64 ,
2024-12-30 00:09:14 +00:00
} ) ;
2024-10-22 07:41:35 +00:00
2024-09-20 08:46:48 +00:00
// Element Configs ---------------------------
// Layout
2024-12-30 00:09:14 +00:00
CLAY__TYPEDEF ( Clay_LayoutDirection , CLAY_PACKED_ENUM {
2024-09-20 08:46:48 +00:00
CLAY_LEFT_TO_RIGHT ,
CLAY_TOP_TO_BOTTOM ,
2024-12-30 00:09:14 +00:00
} ) ;
2024-09-20 08:46:48 +00:00
2024-12-30 00:09:14 +00:00
CLAY__TYPEDEF ( Clay_LayoutAlignmentX , CLAY_PACKED_ENUM {
2024-09-20 08:46:48 +00:00
CLAY_ALIGN_X_LEFT ,
CLAY_ALIGN_X_RIGHT ,
CLAY_ALIGN_X_CENTER ,
2024-12-30 00:09:14 +00:00
} ) ;
2024-09-20 08:46:48 +00:00
2024-12-30 00:09:14 +00:00
CLAY__TYPEDEF ( Clay_LayoutAlignmentY , CLAY_PACKED_ENUM {
2024-09-20 08:46:48 +00:00
CLAY_ALIGN_Y_TOP ,
CLAY_ALIGN_Y_BOTTOM ,
CLAY_ALIGN_Y_CENTER ,
2024-12-30 00:09:14 +00:00
} ) ;
2024-09-20 08:46:48 +00:00
2024-12-30 00:09:14 +00:00
CLAY__TYPEDEF ( Clay__SizingType , CLAY_PACKED_ENUM {
2024-09-20 08:46:48 +00:00
CLAY__SIZING_TYPE_FIT ,
CLAY__SIZING_TYPE_GROW ,
CLAY__SIZING_TYPE_PERCENT ,
2024-10-07 05:56:13 +00:00
CLAY__SIZING_TYPE_FIXED ,
2024-12-30 00:09:14 +00:00
} ) ;
2024-09-20 08:46:48 +00:00
2024-12-30 00:09:14 +00:00
CLAY__TYPEDEF ( Clay_ChildAlignment , struct {
2024-09-20 08:46:48 +00:00
Clay_LayoutAlignmentX x ;
Clay_LayoutAlignmentY y ;
2024-12-30 00:09:14 +00:00
} ) ;
2024-09-20 08:46:48 +00:00
2024-12-30 00:09:14 +00:00
CLAY__TYPEDEF ( Clay_SizingMinMax , struct {
2024-09-20 08:46:48 +00:00
float min ;
float max ;
2024-12-30 00:09:14 +00:00
} ) ;
2024-09-20 08:46:48 +00:00
2024-12-30 00:09:14 +00:00
CLAY__TYPEDEF ( Clay_SizingAxis , struct {
2024-09-20 08:46:48 +00:00
union {
2024-12-26 22:12:59 +00:00
Clay_SizingMinMax minMax ;
float percent ;
} size ;
2024-09-20 08:46:48 +00:00
Clay__SizingType type ;
2024-12-30 00:09:14 +00:00
} ) ;
2024-09-20 08:46:48 +00:00
2024-12-30 00:09:14 +00:00
CLAY__TYPEDEF ( Clay_Sizing , struct {
2024-09-20 08:46:48 +00:00
Clay_SizingAxis width ;
Clay_SizingAxis height ;
2024-12-30 00:09:14 +00:00
} ) ;
2024-09-20 08:46:48 +00:00
2024-12-30 00:09:14 +00:00
CLAY__TYPEDEF ( Clay_Padding , struct {
2024-09-20 08:46:48 +00:00
uint16_t x ;
uint16_t y ;
2024-12-30 00:09:14 +00:00
} ) ;
2024-09-20 08:46:48 +00:00
2024-12-30 00:09:14 +00:00
CLAY__TYPEDEF ( Clay_LayoutConfig , struct {
2024-09-20 08:46:48 +00:00
Clay_Sizing sizing ;
Clay_Padding padding ;
uint16_t childGap ;
Clay_ChildAlignment childAlignment ;
2024-09-28 23:53:46 +00:00
Clay_LayoutDirection layoutDirection ;
2024-12-30 00:09:14 +00:00
} ) ;
2024-09-20 08:46:48 +00:00
2024-09-20 20:00:55 +00:00
extern Clay_LayoutConfig CLAY_LAYOUT_DEFAULT ;
2024-09-20 08:46:48 +00:00
// Rectangle
2024-12-30 00:09:14 +00:00
// NOTE: Not declared in the typedef as an ifdef inside macro arguments is UB
struct Clay_RectangleElementConfig {
2024-09-20 08:46:48 +00:00
Clay_Color color ;
Clay_CornerRadius cornerRadius ;
# ifdef CLAY_EXTEND_CONFIG_RECTANGLE
CLAY_EXTEND_CONFIG_RECTANGLE
# endif
2024-12-30 00:09:14 +00:00
} ;
CLAY__TYPEDEF ( Clay_RectangleElementConfig , struct Clay_RectangleElementConfig ) ;
2024-09-20 08:46:48 +00:00
// Text
2024-12-30 00:09:14 +00:00
CLAY__TYPEDEF ( Clay_TextElementConfigWrapMode , enum {
2024-09-20 08:46:48 +00:00
CLAY_TEXT_WRAP_WORDS ,
CLAY_TEXT_WRAP_NEWLINES ,
CLAY_TEXT_WRAP_NONE ,
2024-12-30 00:09:14 +00:00
} ) ;
2024-09-20 08:46:48 +00:00
2024-12-30 00:09:14 +00:00
struct Clay_TextElementConfig {
2024-09-20 08:46:48 +00:00
Clay_Color textColor ;
uint16_t fontId ;
uint16_t fontSize ;
uint16_t letterSpacing ;
2024-10-05 07:57:52 +00:00
uint16_t lineHeight ;
2024-09-20 08:46:48 +00:00
Clay_TextElementConfigWrapMode wrapMode ;
# ifdef CLAY_EXTEND_CONFIG_TEXT
CLAY_EXTEND_CONFIG_TEXT
# endif
2024-12-30 00:09:14 +00:00
} ;
CLAY__TYPEDEF ( Clay_TextElementConfig , struct Clay_TextElementConfig ) ;
2024-09-20 08:46:48 +00:00
// Image
2024-12-30 00:09:14 +00:00
struct Clay_ImageElementConfig {
void * imageData ;
2024-09-20 08:46:48 +00:00
Clay_Dimensions sourceDimensions ;
# ifdef CLAY_EXTEND_CONFIG_IMAGE
CLAY_EXTEND_CONFIG_IMAGE
# endif
2024-12-30 00:09:14 +00:00
} ;
CLAY__TYPEDEF ( Clay_ImageElementConfig , struct Clay_ImageElementConfig ) ;
2024-09-20 08:46:48 +00:00
// Floating
2024-12-30 00:09:14 +00:00
CLAY__TYPEDEF ( Clay_FloatingAttachPointType , CLAY_PACKED_ENUM {
2024-09-20 08:46:48 +00:00
CLAY_ATTACH_POINT_LEFT_TOP ,
CLAY_ATTACH_POINT_LEFT_CENTER ,
CLAY_ATTACH_POINT_LEFT_BOTTOM ,
CLAY_ATTACH_POINT_CENTER_TOP ,
CLAY_ATTACH_POINT_CENTER_CENTER ,
CLAY_ATTACH_POINT_CENTER_BOTTOM ,
CLAY_ATTACH_POINT_RIGHT_TOP ,
CLAY_ATTACH_POINT_RIGHT_CENTER ,
CLAY_ATTACH_POINT_RIGHT_BOTTOM ,
2024-12-30 00:09:14 +00:00
} ) ;
2024-09-20 08:46:48 +00:00
2024-12-30 00:09:14 +00:00
CLAY__TYPEDEF ( Clay_FloatingAttachPoints , struct {
2024-09-20 08:46:48 +00:00
Clay_FloatingAttachPointType element ;
Clay_FloatingAttachPointType parent ;
2024-12-30 00:09:14 +00:00
} ) ;
2024-09-20 08:46:48 +00:00
2024-12-30 00:09:14 +00:00
CLAY__TYPEDEF ( Clay_PointerCaptureMode , enum {
2024-12-20 17:36:34 +00:00
CLAY_POINTER_CAPTURE_MODE_CAPTURE ,
// CLAY_POINTER_CAPTURE_MODE_PARENT, TODO pass pointer through to attached parent
CLAY_POINTER_CAPTURE_MODE_PASSTHROUGH ,
2024-12-30 00:09:14 +00:00
} ) ;
2024-12-20 17:36:34 +00:00
2024-12-30 00:09:14 +00:00
CLAY__TYPEDEF ( Clay_FloatingElementConfig , struct {
2024-09-20 08:46:48 +00:00
Clay_Vector2 offset ;
Clay_Dimensions expand ;
uint16_t zIndex ;
uint32_t parentId ;
Clay_FloatingAttachPoints attachment ;
2024-12-20 17:36:34 +00:00
Clay_PointerCaptureMode pointerCaptureMode ;
2024-12-30 00:09:14 +00:00
} ) ;
2024-09-20 08:46:48 +00:00
// Custom
2024-12-30 00:09:14 +00:00
struct Clay_CustomElementConfig {
2024-09-20 08:46:48 +00:00
# ifndef CLAY_EXTEND_CONFIG_CUSTOM
2024-12-30 00:09:14 +00:00
void * customData ;
2024-09-20 08:46:48 +00:00
# else
CLAY_EXTEND_CONFIG_CUSTOM
# endif
2024-12-30 00:09:14 +00:00
} ;
CLAY__TYPEDEF ( Clay_CustomElementConfig , struct Clay_CustomElementConfig ) ;
2024-09-20 08:46:48 +00:00
// Scroll
2024-12-30 00:09:14 +00:00
CLAY__TYPEDEF ( Clay_ScrollElementConfig , struct {
2024-09-20 08:46:48 +00:00
bool horizontal ;
bool vertical ;
2024-12-30 00:09:14 +00:00
} ) ;
2024-09-20 08:46:48 +00:00
// Border
2024-12-30 00:09:14 +00:00
CLAY__TYPEDEF ( Clay_Border , struct {
2024-09-20 08:46:48 +00:00
uint32_t width ;
Clay_Color color ;
2024-12-30 00:09:14 +00:00
} ) ;
2024-09-20 08:46:48 +00:00
2024-12-30 00:09:14 +00:00
CLAY__TYPEDEF ( Clay_BorderElementConfig , struct {
2024-09-20 08:46:48 +00:00
Clay_Border left ;
Clay_Border right ;
Clay_Border top ;
Clay_Border bottom ;
Clay_Border betweenChildren ;
Clay_CornerRadius cornerRadius ;
2024-12-30 00:09:14 +00:00
} ) ;
2024-09-20 08:46:48 +00:00
2024-12-30 00:09:14 +00:00
CLAY__TYPEDEF ( Clay_ElementConfigUnion , union {
2024-09-20 08:46:48 +00:00
Clay_RectangleElementConfig * rectangleElementConfig ;
Clay_TextElementConfig * textElementConfig ;
Clay_ImageElementConfig * imageElementConfig ;
Clay_FloatingElementConfig * floatingElementConfig ;
Clay_CustomElementConfig * customElementConfig ;
Clay_ScrollElementConfig * scrollElementConfig ;
Clay_BorderElementConfig * borderElementConfig ;
2024-12-30 00:09:14 +00:00
} ) ;
2024-09-20 08:46:48 +00:00
2024-12-30 00:09:14 +00:00
CLAY__TYPEDEF ( Clay_ElementConfig , struct {
2024-10-22 07:41:35 +00:00
Clay__ElementConfigType type ;
Clay_ElementConfigUnion config ;
2024-12-30 00:09:14 +00:00
} ) ;
2024-10-22 07:41:35 +00:00
2024-09-20 08:46:48 +00:00
// Miscellaneous Structs & Enums ---------------------------------
2024-12-30 00:09:14 +00:00
CLAY__TYPEDEF ( Clay_ScrollContainerData , struct {
2024-09-20 08:46:48 +00:00
// Note: This is a pointer to the real internal scroll position, mutating it may cause a change in final layout.
// Intended for use with external functionality that modifies scroll position, such as scroll bars or auto scrolling.
Clay_Vector2 * scrollPosition ;
Clay_Dimensions scrollContainerDimensions ;
Clay_Dimensions contentDimensions ;
Clay_ScrollElementConfig config ;
// Indicates whether an actual scroll container matched the provided ID or if the default struct was returned.
bool found ;
2024-12-30 00:09:14 +00:00
} ) ;
2024-09-20 08:46:48 +00:00
2025-01-02 20:59:09 +00:00
CLAY__TYPEDEF ( Clay_RenderCommandType , CLAY_PACKED_ENUM {
2024-09-20 08:46:48 +00:00
CLAY_RENDER_COMMAND_TYPE_NONE ,
CLAY_RENDER_COMMAND_TYPE_RECTANGLE ,
CLAY_RENDER_COMMAND_TYPE_BORDER ,
CLAY_RENDER_COMMAND_TYPE_TEXT ,
CLAY_RENDER_COMMAND_TYPE_IMAGE ,
CLAY_RENDER_COMMAND_TYPE_SCISSOR_START ,
CLAY_RENDER_COMMAND_TYPE_SCISSOR_END ,
CLAY_RENDER_COMMAND_TYPE_CUSTOM ,
2024-12-30 00:09:14 +00:00
} ) ;
2024-09-20 08:46:48 +00:00
2024-12-30 00:09:14 +00:00
CLAY__TYPEDEF ( Clay_RenderCommand , struct {
2024-09-20 08:46:48 +00:00
Clay_BoundingBox boundingBox ;
Clay_ElementConfigUnion config ;
Clay_String text ; // TODO I wish there was a way to avoid having to have this on every render command
uint32_t id ;
Clay_RenderCommandType commandType ;
2024-12-30 00:09:14 +00:00
} ) ;
2024-09-20 08:46:48 +00:00
2024-12-30 00:09:14 +00:00
CLAY__TYPEDEF ( Clay_RenderCommandArray , struct {
uint32_t capacity ;
uint32_t length ;
Clay_RenderCommand * internalArray ;
} ) ;
2024-09-20 08:46:48 +00:00
2024-12-30 00:09:14 +00:00
CLAY__TYPEDEF ( Clay_PointerDataInteractionState , enum {
2024-10-22 20:44:39 +00:00
CLAY_POINTER_DATA_PRESSED_THIS_FRAME ,
CLAY_POINTER_DATA_PRESSED ,
CLAY_POINTER_DATA_RELEASED_THIS_FRAME ,
CLAY_POINTER_DATA_RELEASED ,
2024-12-30 00:09:14 +00:00
} ) ;
2024-10-22 07:41:35 +00:00
2024-12-30 00:09:14 +00:00
CLAY__TYPEDEF ( Clay_PointerData , struct {
2024-10-22 07:41:35 +00:00
Clay_Vector2 position ;
2024-10-22 20:44:39 +00:00
Clay_PointerDataInteractionState state ;
2024-12-30 00:09:14 +00:00
} ) ;
2024-10-22 07:41:35 +00:00
2024-12-30 00:09:14 +00:00
CLAY__TYPEDEF ( Clay_ErrorType , enum {
2024-12-26 06:58:59 +00:00
CLAY_ERROR_TYPE_TEXT_MEASUREMENT_FUNCTION_NOT_PROVIDED ,
CLAY_ERROR_TYPE_ARENA_CAPACITY_EXCEEDED ,
CLAY_ERROR_TYPE_ELEMENTS_CAPACITY_EXCEEDED ,
CLAY_ERROR_TYPE_TEXT_MEASUREMENT_CAPACITY_EXCEEDED ,
CLAY_ERROR_TYPE_DUPLICATE_ID ,
CLAY_ERROR_TYPE_FLOATING_CONTAINER_PARENT_NOT_FOUND ,
CLAY_ERROR_TYPE_INTERNAL_ERROR ,
2024-12-30 00:09:14 +00:00
} ) ;
2024-12-26 06:58:59 +00:00
2024-12-30 00:09:14 +00:00
CLAY__TYPEDEF ( Clay_ErrorData , struct {
2024-12-26 06:58:59 +00:00
Clay_ErrorType errorType ;
Clay_String errorText ;
uintptr_t userData ;
2024-12-30 00:09:14 +00:00
} ) ;
2024-12-26 06:58:59 +00:00
2024-12-30 00:09:14 +00:00
CLAY__TYPEDEF ( Clay_ErrorHandler , struct {
2024-12-26 06:58:59 +00:00
void ( * errorHandlerFunction ) ( Clay_ErrorData errorText ) ;
uintptr_t userData ;
2024-12-30 00:09:14 +00:00
} ) ;
2024-12-26 06:58:59 +00:00
2024-09-20 08:46:48 +00:00
// Function Forward Declarations ---------------------------------
2024-10-22 07:41:35 +00:00
// Public API functions ---
2024-12-26 22:12:59 +00:00
uint32_t Clay_MinMemorySize ( void ) ;
2024-09-20 08:46:48 +00:00
Clay_Arena Clay_CreateArenaWithCapacityAndMemory ( uint32_t capacity , void * offset ) ;
void Clay_SetPointerState ( Clay_Vector2 position , bool pointerDown ) ;
2024-12-26 06:58:59 +00:00
void Clay_Initialize ( Clay_Arena arena , Clay_Dimensions layoutDimensions , Clay_ErrorHandler errorHandler ) ;
2024-09-20 20:13:22 +00:00
void Clay_UpdateScrollContainers ( bool enableDragScrolling , Clay_Vector2 scrollDelta , float deltaTime ) ;
2024-09-20 08:46:48 +00:00
void Clay_SetLayoutDimensions ( Clay_Dimensions dimensions ) ;
2024-12-26 22:12:59 +00:00
void Clay_BeginLayout ( void ) ;
Clay_RenderCommandArray Clay_EndLayout ( void ) ;
2024-10-22 07:41:35 +00:00
Clay_ElementId Clay_GetElementId ( Clay_String idString ) ;
2024-10-24 08:38:02 +00:00
Clay_ElementId Clay_GetElementIdWithIndex ( Clay_String idString , uint32_t index ) ;
2024-12-26 22:12:59 +00:00
bool Clay_Hovered ( void ) ;
2024-10-22 20:44:39 +00:00
void Clay_OnHover ( void ( * onHoverFunction ) ( Clay_ElementId elementId , Clay_PointerData pointerData , intptr_t userData ) , intptr_t userData ) ;
2024-12-28 09:33:41 +00:00
bool Clay_PointerOver ( Clay_ElementId elementId ) ;
2024-09-20 08:46:48 +00:00
Clay_ScrollContainerData Clay_GetScrollContainerData ( Clay_ElementId id ) ;
void Clay_SetMeasureTextFunction ( Clay_Dimensions ( * measureTextFunction ) ( Clay_String * text , Clay_TextElementConfig * config ) ) ;
2024-12-18 22:35:18 +00:00
void Clay_SetQueryScrollOffsetFunction ( Clay_Vector2 ( * queryScrollOffsetFunction ) ( uint32_t elementId ) ) ;
2024-09-20 08:46:48 +00:00
Clay_RenderCommand * Clay_RenderCommandArray_Get ( Clay_RenderCommandArray * array , int32_t index ) ;
void Clay_SetDebugModeEnabled ( bool enabled ) ;
2024-12-29 23:04:48 +00:00
bool Clay_IsDebugModeEnabled ( void ) ;
2024-12-18 22:35:18 +00:00
void Clay_SetCullingEnabled ( bool enabled ) ;
2024-12-26 06:58:59 +00:00
void Clay_SetMaxElementCount ( uint32_t maxElementCount ) ;
void Clay_SetMaxMeasureTextCacheWordCount ( uint32_t maxMeasureTextCacheWordCount ) ;
2024-09-20 08:46:48 +00:00
// Internal API functions required by macros
2024-12-26 22:12:59 +00:00
void Clay__OpenElement ( void ) ;
void Clay__CloseElement ( void ) ;
2024-09-20 08:46:48 +00:00
Clay_LayoutConfig * Clay__StoreLayoutConfig ( Clay_LayoutConfig config ) ;
2024-12-26 22:12:59 +00:00
void Clay__ElementPostConfiguration ( void ) ;
2024-10-22 07:41:35 +00:00
void Clay__AttachId ( Clay_ElementId id ) ;
void Clay__AttachLayoutConfig ( Clay_LayoutConfig * config ) ;
void Clay__AttachElementConfig ( Clay_ElementConfigUnion config , Clay__ElementConfigType type ) ;
2024-09-20 08:46:48 +00:00
Clay_RectangleElementConfig * Clay__StoreRectangleElementConfig ( Clay_RectangleElementConfig config ) ;
Clay_TextElementConfig * Clay__StoreTextElementConfig ( Clay_TextElementConfig config ) ;
Clay_ImageElementConfig * Clay__StoreImageElementConfig ( Clay_ImageElementConfig config ) ;
Clay_FloatingElementConfig * Clay__StoreFloatingElementConfig ( Clay_FloatingElementConfig config ) ;
Clay_CustomElementConfig * Clay__StoreCustomElementConfig ( Clay_CustomElementConfig config ) ;
Clay_ScrollElementConfig * Clay__StoreScrollElementConfig ( Clay_ScrollElementConfig config ) ;
Clay_BorderElementConfig * Clay__StoreBorderElementConfig ( Clay_BorderElementConfig config ) ;
2024-09-25 02:04:28 +00:00
Clay_ElementId Clay__HashString ( Clay_String key , uint32_t offset , uint32_t seed ) ;
2024-11-18 04:01:55 +00:00
void Clay__OpenTextElement ( Clay_String text , Clay_TextElementConfig * textConfig ) ;
2024-09-20 20:00:55 +00:00
extern Clay_Color Clay__debugViewHighlightColor ;
extern uint32_t Clay__debugViewWidth ;
2024-09-20 08:46:48 +00:00
# ifdef __cplusplus
}
# endif
2024-12-30 00:09:14 +00:00
# endif // CLAY_HEADER
2024-09-20 08:46:48 +00:00
// -----------------------------------------
// IMPLEMENTATION --------------------------
// -----------------------------------------
# ifdef CLAY_IMPLEMENTATION
# undef CLAY_IMPLEMENTATION
# ifndef CLAY__NULL
# define CLAY__NULL 0
# endif
# ifndef CLAY__MAXFLOAT
# define CLAY__MAXFLOAT 3.40282346638528859812e+38F
# endif
2024-09-16 09:34:59 +00:00
bool Clay__warningsEnabled = true ;
2024-12-26 06:58:59 +00:00
uint32_t Clay__maxElementCount = 8192 ;
uint32_t Clay__maxMeasureTextCacheWordCount = 16384 ;
2024-12-30 00:09:14 +00:00
void Clay__ErrorHandlerFunctionDefault ( Clay_ErrorData errorText ) {
( void ) errorText ;
}
Clay_ErrorHandler Clay__errorHandler = { . errorHandlerFunction = Clay__ErrorHandlerFunctionDefault } ;
2024-09-16 09:34:59 +00:00
2024-12-30 00:09:14 +00:00
Clay_String CLAY__SPACECHAR = { . length = 1 , . chars = " " } ;
Clay_String CLAY__STRING_DEFAULT = { . length = 0 , . chars = NULL } ;
2024-08-23 04:05:23 +00:00
2024-12-30 00:09:14 +00:00
CLAY__TYPEDEF ( Clay_BooleanWarnings , struct {
2024-12-01 08:05:56 +00:00
bool maxElementsExceeded ;
bool maxRenderCommandsExceeded ;
2024-12-26 06:58:59 +00:00
bool maxTextMeasureCacheExceeded ;
2024-12-30 00:09:14 +00:00
} ) ;
2024-12-01 08:05:56 +00:00
Clay_BooleanWarnings Clay__booleanWarnings ;
2024-12-30 00:09:14 +00:00
CLAY__TYPEDEF ( Clay__Warning , struct {
2024-09-16 09:34:59 +00:00
Clay_String baseMessage ;
Clay_String dynamicMessage ;
2024-12-30 00:09:14 +00:00
} ) ;
2024-08-23 04:05:23 +00:00
2024-12-30 00:09:14 +00:00
Clay__Warning CLAY__WARNING_DEFAULT = { 0 } ;
2024-08-23 04:05:23 +00:00
2024-09-16 09:34:59 +00:00
# pragma region generated
2024-12-30 00:09:14 +00:00
CLAY__TYPEDEF ( Clay__WarningArray , struct {
uint32_t capacity ;
uint32_t length ;
Clay__Warning * internalArray ;
} ) ;
2024-09-16 09:34:59 +00:00
Clay__WarningArray Clay__WarningArray_Allocate_Arena ( uint32_t capacity , Clay_Arena * arena ) {
2024-12-22 06:22:10 +00:00
size_t totalSizeBytes = capacity * sizeof ( Clay_String ) ;
2024-12-30 00:09:14 +00:00
Clay__WarningArray array = { . capacity = capacity , . length = 0 } ;
2024-12-26 06:58:59 +00:00
uintptr_t nextAllocAddress = arena - > nextAllocation + ( uintptr_t ) arena - > memory ;
2024-12-22 06:22:10 +00:00
uintptr_t arenaOffsetAligned = nextAllocAddress + ( CLAY__ALIGNMENT ( Clay_String ) - ( nextAllocAddress % CLAY__ALIGNMENT ( Clay_String ) ) ) ;
arenaOffsetAligned - = ( uintptr_t ) arena - > memory ;
2024-09-16 09:34:59 +00:00
if ( arenaOffsetAligned + totalSizeBytes < = arena - > capacity ) {
2024-12-22 06:22:10 +00:00
array . internalArray = ( Clay__Warning * ) ( ( uintptr_t ) arena - > memory + ( uintptr_t ) arenaOffsetAligned ) ;
2024-09-16 09:34:59 +00:00
arena - > nextAllocation = arenaOffsetAligned + totalSizeBytes ;
2024-08-23 04:05:23 +00:00
}
else {
2024-12-26 06:58:59 +00:00
Clay__errorHandler . errorHandlerFunction ( CLAY__INIT ( Clay_ErrorData ) {
. errorType = CLAY_ERROR_TYPE_ARENA_CAPACITY_EXCEEDED ,
. errorText = CLAY_STRING ( " Clay attempted to allocate memory in its arena, but ran out of capacity. Try increasing the capacity of the arena passed to Clay_Initialize() " ) ,
. userData = Clay__errorHandler . userData } ) ;
2024-08-23 04:05:23 +00:00
}
2024-09-16 09:34:59 +00:00
return array ;
2024-08-23 04:05:23 +00:00
}
2024-12-30 00:09:14 +00:00
Clay__WarningArray Clay_warnings = { 0 } ;
2024-09-16 09:34:59 +00:00
Clay__Warning * Clay__WarningArray_Add ( Clay__WarningArray * array , Clay__Warning item )
2024-08-23 04:05:23 +00:00
{
2024-09-16 09:34:59 +00:00
if ( array - > length < array - > capacity ) {
array - > internalArray [ array - > length + + ] = item ;
return & array - > internalArray [ array - > length - 1 ] ;
2024-08-23 04:05:23 +00:00
}
2024-09-16 09:34:59 +00:00
return & CLAY__WARNING_DEFAULT ;
2024-08-23 04:05:23 +00:00
}
2024-08-24 03:11:39 +00:00
void * Clay__Array_Allocate_Arena ( uint32_t capacity , uint32_t itemSize , uint32_t alignment , Clay_Arena * arena )
2024-08-23 04:05:23 +00:00
{
2024-12-22 06:22:10 +00:00
size_t totalSizeBytes = capacity * itemSize ;
2024-12-26 06:58:59 +00:00
uintptr_t nextAllocAddress = arena - > nextAllocation + ( uintptr_t ) arena - > memory ;
2024-12-22 06:22:10 +00:00
uintptr_t arenaOffsetAligned = nextAllocAddress + ( alignment - ( nextAllocAddress % alignment ) ) ;
arenaOffsetAligned - = ( uintptr_t ) arena - > memory ;
2024-08-23 04:05:23 +00:00
if ( arenaOffsetAligned + totalSizeBytes < = arena - > capacity ) {
arena - > nextAllocation = arenaOffsetAligned + totalSizeBytes ;
2024-12-22 06:22:10 +00:00
return ( void * ) ( ( uintptr_t ) arena - > memory + ( uintptr_t ) arenaOffsetAligned ) ;
2024-08-23 04:05:23 +00:00
}
else {
2024-12-26 06:58:59 +00:00
Clay__errorHandler . errorHandlerFunction ( CLAY__INIT ( Clay_ErrorData ) {
. errorType = CLAY_ERROR_TYPE_ARENA_CAPACITY_EXCEEDED ,
. errorText = CLAY_STRING ( " Clay attempted to allocate memory in its arena, but ran out of capacity. Try increasing the capacity of the arena passed to Clay_Initialize() " ) ,
. userData = Clay__errorHandler . userData } ) ;
2024-08-23 04:05:23 +00:00
}
return CLAY__NULL ;
}
bool Clay__Array_RangeCheck ( int index , uint32_t length )
{
if ( index < length & & index > = 0 ) {
return true ;
}
2024-12-26 06:58:59 +00:00
Clay__errorHandler . errorHandlerFunction ( CLAY__INIT ( Clay_ErrorData ) {
. errorType = CLAY_ERROR_TYPE_INTERNAL_ERROR ,
. errorText = CLAY_STRING ( " Clay attempted to make an out of bounds array access. This is an internal error and is likely a bug. " ) ,
. userData = Clay__errorHandler . userData } ) ;
2024-08-23 04:05:23 +00:00
return false ;
}
2024-12-26 06:58:59 +00:00
bool Clay__Array_AddCapacityCheck ( uint32_t length , uint32_t capacity )
2024-08-23 04:05:23 +00:00
{
if ( length < capacity ) {
return true ;
}
2024-12-26 06:58:59 +00:00
Clay__errorHandler . errorHandlerFunction ( CLAY__INIT ( Clay_ErrorData ) {
. errorType = CLAY_ERROR_TYPE_INTERNAL_ERROR ,
. errorText = CLAY_STRING ( " Clay attempted to make an out of bounds array access. This is an internal error and is likely a bug. " ) ,
. userData = Clay__errorHandler . userData } ) ;
2024-08-23 04:05:23 +00:00
return false ;
}
bool CLAY__BOOL_DEFAULT = false ;
2024-10-22 07:41:35 +00:00
// __GENERATED__ template array_define,array_allocate TYPE=bool NAME=Clay__BoolArray
2024-08-23 04:05:23 +00:00
# pragma region generated
2024-12-30 00:09:14 +00:00
CLAY__TYPEDEF ( Clay__BoolArray , struct
2024-08-23 04:05:23 +00:00
{
uint32_t capacity ;
uint32_t length ;
bool * internalArray ;
2024-12-30 00:09:14 +00:00
} ) ;
2024-08-23 04:05:23 +00:00
Clay__BoolArray Clay__BoolArray_Allocate_Arena ( uint32_t capacity , Clay_Arena * arena ) {
2024-09-28 23:53:46 +00:00
return CLAY__INIT ( Clay__BoolArray ) { . capacity = capacity , . length = 0 , . internalArray = ( bool * ) Clay__Array_Allocate_Arena ( capacity , sizeof ( bool ) , CLAY__ALIGNMENT ( bool ) , arena ) } ;
2024-08-23 04:05:23 +00:00
}
# pragma endregion
// __GENERATED__ template
2024-12-30 00:09:14 +00:00
Clay_ElementId CLAY__ELEMENT_ID_DEFAULT = { 0 } ;
2024-09-16 09:34:59 +00:00
2024-10-22 07:41:35 +00:00
// __GENERATED__ template array_define,array_allocate,array_get,array_add TYPE=Clay_ElementId NAME=Clay__ElementIdArray DEFAULT_VALUE=&CLAY__ELEMENT_ID_DEFAULT
2024-09-16 09:34:59 +00:00
# pragma region generated
2024-12-30 00:09:14 +00:00
CLAY__TYPEDEF ( Clay__ElementIdArray , struct
2024-09-16 09:34:59 +00:00
{
uint32_t capacity ;
uint32_t length ;
Clay_ElementId * internalArray ;
2024-12-30 00:09:14 +00:00
} ) ;
2024-09-16 09:34:59 +00:00
Clay__ElementIdArray Clay__ElementIdArray_Allocate_Arena ( uint32_t capacity , Clay_Arena * arena ) {
2024-09-28 23:53:46 +00:00
return CLAY__INIT ( Clay__ElementIdArray ) { . capacity = capacity , . length = 0 , . internalArray = ( Clay_ElementId * ) Clay__Array_Allocate_Arena ( capacity , sizeof ( Clay_ElementId ) , CLAY__ALIGNMENT ( Clay_ElementId ) , arena ) } ;
2024-09-16 09:34:59 +00:00
}
Clay_ElementId * Clay__ElementIdArray_Get ( Clay__ElementIdArray * array , int index ) {
return Clay__Array_RangeCheck ( index , array - > length ) ? & array - > internalArray [ index ] : & CLAY__ELEMENT_ID_DEFAULT ;
}
Clay_ElementId * Clay__ElementIdArray_Add ( Clay__ElementIdArray * array , Clay_ElementId item ) {
2024-12-26 06:58:59 +00:00
if ( Clay__Array_AddCapacityCheck ( array - > length , array - > capacity ) ) {
2024-09-16 09:34:59 +00:00
array - > internalArray [ array - > length + + ] = item ;
return & array - > internalArray [ array - > length - 1 ] ;
}
return & CLAY__ELEMENT_ID_DEFAULT ;
}
# pragma endregion
// __GENERATED__ template
2024-12-30 00:09:14 +00:00
Clay_ElementConfig CLAY__ELEMENT_CONFIG_DEFAULT = { CLAY__ELEMENT_CONFIG_TYPE_NONE , { 0 } } ;
2024-10-22 07:41:35 +00:00
// __GENERATED__ template array_define,array_define_slice,array_allocate,array_get,array_add,array_get_slice TYPE=Clay_ElementConfig NAME=Clay__ElementConfigArray DEFAULT_VALUE=&CLAY__ELEMENT_CONFIG_DEFAULT
# pragma region generated
2024-12-30 00:09:14 +00:00
CLAY__TYPEDEF ( Clay__ElementConfigArray , struct
2024-10-22 07:41:35 +00:00
{
uint32_t capacity ;
uint32_t length ;
Clay_ElementConfig * internalArray ;
2024-12-30 00:09:14 +00:00
} ) ;
CLAY__TYPEDEF ( Clay__ElementConfigArraySlice , struct
2024-10-22 07:41:35 +00:00
{
uint32_t length ;
Clay_ElementConfig * internalArray ;
2024-12-30 00:09:14 +00:00
} ) ;
2024-10-22 07:41:35 +00:00
Clay__ElementConfigArray Clay__ElementConfigArray_Allocate_Arena ( uint32_t capacity , Clay_Arena * arena ) {
return CLAY__INIT ( Clay__ElementConfigArray ) { . capacity = capacity , . length = 0 , . internalArray = ( Clay_ElementConfig * ) Clay__Array_Allocate_Arena ( capacity , sizeof ( Clay_ElementConfig ) , CLAY__ALIGNMENT ( Clay_ElementConfig ) , arena ) } ;
}
Clay_ElementConfig * Clay__ElementConfigArray_Get ( Clay__ElementConfigArray * array , int index ) {
return Clay__Array_RangeCheck ( index , array - > length ) ? & array - > internalArray [ index ] : & CLAY__ELEMENT_CONFIG_DEFAULT ;
}
Clay_ElementConfig * Clay__ElementConfigArray_Add ( Clay__ElementConfigArray * array , Clay_ElementConfig item ) {
2024-12-26 06:58:59 +00:00
if ( Clay__Array_AddCapacityCheck ( array - > length , array - > capacity ) ) {
2024-10-22 07:41:35 +00:00
array - > internalArray [ array - > length + + ] = item ;
return & array - > internalArray [ array - > length - 1 ] ;
}
return & CLAY__ELEMENT_CONFIG_DEFAULT ;
}
Clay_ElementConfig * Clay__ElementConfigArraySlice_Get ( Clay__ElementConfigArraySlice * slice , int index ) {
return Clay__Array_RangeCheck ( index , slice - > length ) ? & slice - > internalArray [ index ] : & CLAY__ELEMENT_CONFIG_DEFAULT ;
}
# pragma endregion
// __GENERATED__ template
2024-08-23 04:05:23 +00:00
2024-12-30 00:09:14 +00:00
Clay_LayoutConfig CLAY_LAYOUT_DEFAULT = { . sizing = { . width = { . size = { . minMax = { 0 , CLAY__MAXFLOAT } } , . type = CLAY__SIZING_TYPE_FIT } , . height = { . size = { . minMax = { 0 , CLAY__MAXFLOAT } } , . type = CLAY__SIZING_TYPE_FIT } } } ;
2024-08-23 04:05:23 +00:00
2024-10-22 07:41:35 +00:00
// __GENERATED__ template array_define,array_allocate,array_add TYPE=Clay_LayoutConfig NAME=Clay__LayoutConfigArray DEFAULT_VALUE=&CLAY_LAYOUT_DEFAULT
2024-08-23 04:05:23 +00:00
# pragma region generated
2024-12-30 00:09:14 +00:00
CLAY__TYPEDEF ( Clay__LayoutConfigArray , struct
2024-08-23 04:05:23 +00:00
{
uint32_t capacity ;
uint32_t length ;
Clay_LayoutConfig * internalArray ;
2024-12-30 00:09:14 +00:00
} ) ;
2024-08-30 06:44:36 +00:00
Clay__LayoutConfigArray Clay__LayoutConfigArray_Allocate_Arena ( uint32_t capacity , Clay_Arena * arena ) {
2024-09-28 23:53:46 +00:00
return CLAY__INIT ( Clay__LayoutConfigArray ) { . capacity = capacity , . length = 0 , . internalArray = ( Clay_LayoutConfig * ) Clay__Array_Allocate_Arena ( capacity , sizeof ( Clay_LayoutConfig ) , CLAY__ALIGNMENT ( Clay_LayoutConfig ) , arena ) } ;
2024-08-23 04:05:23 +00:00
}
2024-08-30 06:44:36 +00:00
Clay_LayoutConfig * Clay__LayoutConfigArray_Add ( Clay__LayoutConfigArray * array , Clay_LayoutConfig item ) {
2024-12-26 06:58:59 +00:00
if ( Clay__Array_AddCapacityCheck ( array - > length , array - > capacity ) ) {
2024-08-23 04:05:23 +00:00
array - > internalArray [ array - > length + + ] = item ;
return & array - > internalArray [ array - > length - 1 ] ;
}
return & CLAY_LAYOUT_DEFAULT ;
}
# pragma endregion
// __GENERATED__ template
2024-12-30 00:09:14 +00:00
Clay_RectangleElementConfig CLAY__RECTANGLE_ELEMENT_CONFIG_DEFAULT = { 0 } ;
2024-08-23 04:05:23 +00:00
2024-10-22 07:41:35 +00:00
// __GENERATED__ template array_define,array_allocate,array_add TYPE=Clay_RectangleElementConfig NAME=Clay__RectangleElementConfigArray DEFAULT_VALUE=&CLAY__RECTANGLE_ELEMENT_CONFIG_DEFAULT
2024-08-23 04:05:23 +00:00
# pragma region generated
2024-12-30 00:09:14 +00:00
CLAY__TYPEDEF ( Clay__RectangleElementConfigArray , struct
2024-08-23 04:05:23 +00:00
{
uint32_t capacity ;
uint32_t length ;
Clay_RectangleElementConfig * internalArray ;
2024-12-30 00:09:14 +00:00
} ) ;
2024-08-30 06:44:36 +00:00
Clay__RectangleElementConfigArray Clay__RectangleElementConfigArray_Allocate_Arena ( uint32_t capacity , Clay_Arena * arena ) {
2024-09-28 23:53:46 +00:00
return CLAY__INIT ( Clay__RectangleElementConfigArray ) { . capacity = capacity , . length = 0 , . internalArray = ( Clay_RectangleElementConfig * ) Clay__Array_Allocate_Arena ( capacity , sizeof ( Clay_RectangleElementConfig ) , CLAY__ALIGNMENT ( Clay_RectangleElementConfig ) , arena ) } ;
2024-08-23 04:05:23 +00:00
}
2024-08-30 06:44:36 +00:00
Clay_RectangleElementConfig * Clay__RectangleElementConfigArray_Add ( Clay__RectangleElementConfigArray * array , Clay_RectangleElementConfig item ) {
2024-12-26 06:58:59 +00:00
if ( Clay__Array_AddCapacityCheck ( array - > length , array - > capacity ) ) {
2024-08-23 04:05:23 +00:00
array - > internalArray [ array - > length + + ] = item ;
return & array - > internalArray [ array - > length - 1 ] ;
}
return & CLAY__RECTANGLE_ELEMENT_CONFIG_DEFAULT ;
}
# pragma endregion
// __GENERATED__ template
2024-12-30 00:09:14 +00:00
Clay_TextElementConfig CLAY__TEXT_ELEMENT_CONFIG_DEFAULT = { 0 } ;
2024-08-23 04:05:23 +00:00
2024-10-22 07:41:35 +00:00
// __GENERATED__ template array_define,array_allocate,array_add TYPE=Clay_TextElementConfig NAME=Clay__TextElementConfigArray DEFAULT_VALUE=&CLAY__TEXT_ELEMENT_CONFIG_DEFAULT
2024-08-23 04:05:23 +00:00
# pragma region generated
2024-12-30 00:09:14 +00:00
CLAY__TYPEDEF ( Clay__TextElementConfigArray , struct
2024-08-23 04:05:23 +00:00
{
uint32_t capacity ;
uint32_t length ;
Clay_TextElementConfig * internalArray ;
2024-12-30 00:09:14 +00:00
} ) ;
2024-08-30 06:44:36 +00:00
Clay__TextElementConfigArray Clay__TextElementConfigArray_Allocate_Arena ( uint32_t capacity , Clay_Arena * arena ) {
2024-09-28 23:53:46 +00:00
return CLAY__INIT ( Clay__TextElementConfigArray ) { . capacity = capacity , . length = 0 , . internalArray = ( Clay_TextElementConfig * ) Clay__Array_Allocate_Arena ( capacity , sizeof ( Clay_TextElementConfig ) , CLAY__ALIGNMENT ( Clay_TextElementConfig ) , arena ) } ;
2024-08-23 04:05:23 +00:00
}
2024-08-30 06:44:36 +00:00
Clay_TextElementConfig * Clay__TextElementConfigArray_Add ( Clay__TextElementConfigArray * array , Clay_TextElementConfig item ) {
2024-12-26 06:58:59 +00:00
if ( Clay__Array_AddCapacityCheck ( array - > length , array - > capacity ) ) {
2024-08-23 04:05:23 +00:00
array - > internalArray [ array - > length + + ] = item ;
return & array - > internalArray [ array - > length - 1 ] ;
}
return & CLAY__TEXT_ELEMENT_CONFIG_DEFAULT ;
}
# pragma endregion
// __GENERATED__ template
2024-12-30 00:09:14 +00:00
Clay_ImageElementConfig CLAY__IMAGE_ELEMENT_CONFIG_DEFAULT = { 0 } ;
2024-08-23 04:05:23 +00:00
2024-10-22 07:41:35 +00:00
// __GENERATED__ template array_define,array_allocate,array_add TYPE=Clay_ImageElementConfig NAME=Clay__ImageElementConfigArray DEFAULT_VALUE=&CLAY__IMAGE_ELEMENT_CONFIG_DEFAULT
2024-08-23 04:05:23 +00:00
# pragma region generated
2024-12-30 00:09:14 +00:00
CLAY__TYPEDEF ( Clay__ImageElementConfigArray , struct
2024-08-23 04:05:23 +00:00
{
uint32_t capacity ;
uint32_t length ;
Clay_ImageElementConfig * internalArray ;
2024-12-30 00:09:14 +00:00
} ) ;
2024-08-30 06:44:36 +00:00
Clay__ImageElementConfigArray Clay__ImageElementConfigArray_Allocate_Arena ( uint32_t capacity , Clay_Arena * arena ) {
2024-09-28 23:53:46 +00:00
return CLAY__INIT ( Clay__ImageElementConfigArray ) { . capacity = capacity , . length = 0 , . internalArray = ( Clay_ImageElementConfig * ) Clay__Array_Allocate_Arena ( capacity , sizeof ( Clay_ImageElementConfig ) , CLAY__ALIGNMENT ( Clay_ImageElementConfig ) , arena ) } ;
2024-08-23 04:05:23 +00:00
}
2024-08-30 06:44:36 +00:00
Clay_ImageElementConfig * Clay__ImageElementConfigArray_Add ( Clay__ImageElementConfigArray * array , Clay_ImageElementConfig item ) {
2024-12-26 06:58:59 +00:00
if ( Clay__Array_AddCapacityCheck ( array - > length , array - > capacity ) ) {
2024-08-23 04:05:23 +00:00
array - > internalArray [ array - > length + + ] = item ;
return & array - > internalArray [ array - > length - 1 ] ;
}
return & CLAY__IMAGE_ELEMENT_CONFIG_DEFAULT ;
}
# pragma endregion
// __GENERATED__ template
2024-12-30 00:09:14 +00:00
Clay_FloatingElementConfig CLAY__FLOATING_ELEMENT_CONFIG_DEFAULT = { 0 } ;
2024-08-23 04:05:23 +00:00
2024-10-22 07:41:35 +00:00
// __GENERATED__ template array_define,array_allocate,array_add TYPE=Clay_FloatingElementConfig NAME=Clay__FloatingElementConfigArray DEFAULT_VALUE=&CLAY__FLOATING_ELEMENT_CONFIG_DEFAULT
2024-08-23 04:05:23 +00:00
# pragma region generated
2024-12-30 00:09:14 +00:00
CLAY__TYPEDEF ( Clay__FloatingElementConfigArray , struct
2024-08-23 04:05:23 +00:00
{
uint32_t capacity ;
uint32_t length ;
Clay_FloatingElementConfig * internalArray ;
2024-12-30 00:09:14 +00:00
} ) ;
2024-08-30 06:44:36 +00:00
Clay__FloatingElementConfigArray Clay__FloatingElementConfigArray_Allocate_Arena ( uint32_t capacity , Clay_Arena * arena ) {
2024-09-28 23:53:46 +00:00
return CLAY__INIT ( Clay__FloatingElementConfigArray ) { . capacity = capacity , . length = 0 , . internalArray = ( Clay_FloatingElementConfig * ) Clay__Array_Allocate_Arena ( capacity , sizeof ( Clay_FloatingElementConfig ) , CLAY__ALIGNMENT ( Clay_FloatingElementConfig ) , arena ) } ;
2024-08-23 04:05:23 +00:00
}
2024-08-30 06:44:36 +00:00
Clay_FloatingElementConfig * Clay__FloatingElementConfigArray_Add ( Clay__FloatingElementConfigArray * array , Clay_FloatingElementConfig item ) {
2024-12-26 06:58:59 +00:00
if ( Clay__Array_AddCapacityCheck ( array - > length , array - > capacity ) ) {
2024-08-23 04:05:23 +00:00
array - > internalArray [ array - > length + + ] = item ;
return & array - > internalArray [ array - > length - 1 ] ;
}
return & CLAY__FLOATING_ELEMENT_CONFIG_DEFAULT ;
}
# pragma endregion
// __GENERATED__ template
2024-12-30 00:09:14 +00:00
Clay_CustomElementConfig CLAY__CUSTOM_ELEMENT_CONFIG_DEFAULT = { 0 } ;
2024-08-23 04:05:23 +00:00
2024-10-22 07:41:35 +00:00
// __GENERATED__ template array_define,array_allocate,array_add TYPE=Clay_CustomElementConfig NAME=Clay__CustomElementConfigArray DEFAULT_VALUE=&CLAY__CUSTOM_ELEMENT_CONFIG_DEFAULT
2024-08-23 04:05:23 +00:00
# pragma region generated
2024-12-30 00:09:14 +00:00
CLAY__TYPEDEF ( Clay__CustomElementConfigArray , struct
2024-08-23 04:05:23 +00:00
{
uint32_t capacity ;
uint32_t length ;
Clay_CustomElementConfig * internalArray ;
2024-12-30 00:09:14 +00:00
} ) ;
2024-08-30 06:44:36 +00:00
Clay__CustomElementConfigArray Clay__CustomElementConfigArray_Allocate_Arena ( uint32_t capacity , Clay_Arena * arena ) {
2024-09-28 23:53:46 +00:00
return CLAY__INIT ( Clay__CustomElementConfigArray ) { . capacity = capacity , . length = 0 , . internalArray = ( Clay_CustomElementConfig * ) Clay__Array_Allocate_Arena ( capacity , sizeof ( Clay_CustomElementConfig ) , CLAY__ALIGNMENT ( Clay_CustomElementConfig ) , arena ) } ;
2024-08-23 04:05:23 +00:00
}
2024-08-30 06:44:36 +00:00
Clay_CustomElementConfig * Clay__CustomElementConfigArray_Add ( Clay__CustomElementConfigArray * array , Clay_CustomElementConfig item ) {
2024-12-26 06:58:59 +00:00
if ( Clay__Array_AddCapacityCheck ( array - > length , array - > capacity ) ) {
2024-08-23 04:05:23 +00:00
array - > internalArray [ array - > length + + ] = item ;
return & array - > internalArray [ array - > length - 1 ] ;
}
return & CLAY__CUSTOM_ELEMENT_CONFIG_DEFAULT ;
}
# pragma endregion
// __GENERATED__ template
2024-12-30 00:09:14 +00:00
Clay_ScrollElementConfig CLAY__SCROLL_ELEMENT_CONFIG_DEFAULT = { 0 } ;
2024-08-23 04:05:23 +00:00
2024-12-01 08:05:56 +00:00
// __GENERATED__ template array_define,array_allocate,array_add TYPE=Clay_ScrollElementConfig NAME=Clay__ScrollElementConfigArray DEFAULT_VALUE=&CLAY__SCROLL_ELEMENT_CONFIG_DEFAULT
2024-08-23 04:05:23 +00:00
# pragma region generated
2024-12-30 00:09:14 +00:00
CLAY__TYPEDEF ( Clay__ScrollElementConfigArray , struct
2024-08-23 04:05:23 +00:00
{
uint32_t capacity ;
uint32_t length ;
2024-08-30 06:44:36 +00:00
Clay_ScrollElementConfig * internalArray ;
2024-12-30 00:09:14 +00:00
} ) ;
2024-08-30 06:44:36 +00:00
Clay__ScrollElementConfigArray Clay__ScrollElementConfigArray_Allocate_Arena ( uint32_t capacity , Clay_Arena * arena ) {
2024-09-28 23:53:46 +00:00
return CLAY__INIT ( Clay__ScrollElementConfigArray ) { . capacity = capacity , . length = 0 , . internalArray = ( Clay_ScrollElementConfig * ) Clay__Array_Allocate_Arena ( capacity , sizeof ( Clay_ScrollElementConfig ) , CLAY__ALIGNMENT ( Clay_ScrollElementConfig ) , arena ) } ;
2024-08-23 04:05:23 +00:00
}
2024-08-30 06:44:36 +00:00
Clay_ScrollElementConfig * Clay__ScrollElementConfigArray_Add ( Clay__ScrollElementConfigArray * array , Clay_ScrollElementConfig item ) {
2024-12-26 06:58:59 +00:00
if ( Clay__Array_AddCapacityCheck ( array - > length , array - > capacity ) ) {
2024-08-23 04:05:23 +00:00
array - > internalArray [ array - > length + + ] = item ;
return & array - > internalArray [ array - > length - 1 ] ;
}
2024-12-01 08:05:56 +00:00
return & CLAY__SCROLL_ELEMENT_CONFIG_DEFAULT ;
2024-08-23 04:05:23 +00:00
}
# pragma endregion
// __GENERATED__ template
2024-10-22 07:41:35 +00:00
// __GENERATED__ template array_define_slice,array_allocate,array_add TYPE=Clay_String NAME=Clay__StringArray DEFAULT_VALUE=&CLAY__STRING_DEFAULT
# pragma region generated
2024-12-30 00:09:14 +00:00
CLAY__TYPEDEF ( Clay__StringArraySlice , struct
2024-09-16 09:34:59 +00:00
{
2024-10-22 07:41:35 +00:00
uint32_t length ;
Clay_String * internalArray ;
2024-12-30 00:09:14 +00:00
} ) ;
2024-10-22 07:41:35 +00:00
Clay__StringArray Clay__StringArray_Allocate_Arena ( uint32_t capacity , Clay_Arena * arena ) {
return CLAY__INIT ( Clay__StringArray ) { . capacity = capacity , . length = 0 , . internalArray = ( Clay_String * ) Clay__Array_Allocate_Arena ( capacity , sizeof ( Clay_String ) , CLAY__ALIGNMENT ( Clay_String ) , arena ) } ;
}
Clay_String * Clay__StringArray_Add ( Clay__StringArray * array , Clay_String item ) {
2024-12-26 06:58:59 +00:00
if ( Clay__Array_AddCapacityCheck ( array - > length , array - > capacity ) ) {
2024-10-22 07:41:35 +00:00
array - > internalArray [ array - > length + + ] = item ;
return & array - > internalArray [ array - > length - 1 ] ;
}
return & CLAY__STRING_DEFAULT ;
}
# pragma endregion
// __GENERATED__ template
2024-12-30 00:09:14 +00:00
CLAY__TYPEDEF ( Clay__WrappedTextLine , struct {
2024-12-28 06:15:22 +00:00
Clay_Dimensions dimensions ;
Clay_String line ;
2024-12-30 00:09:14 +00:00
} ) ;
2024-12-28 06:15:22 +00:00
2024-12-30 00:09:14 +00:00
Clay__WrappedTextLine CLAY__WRAPPED_TEXT_LINE_DEFAULT = { 0 } ;
2024-12-28 06:15:22 +00:00
// __GENERATED__ template array_define,array_define_slice,array_allocate,array_add,array_get TYPE=Clay__WrappedTextLine NAME=Clay__WrappedTextLineArray DEFAULT_VALUE=&CLAY__WRAPPED_TEXT_LINE_DEFAULT
# pragma region generated
2024-12-30 00:09:14 +00:00
CLAY__TYPEDEF ( Clay__WrappedTextLineArray , struct
2024-12-28 06:15:22 +00:00
{
2024-12-30 00:09:14 +00:00
uint32_t capacity ;
uint32_t length ;
Clay__WrappedTextLine * internalArray ;
} ) ;
CLAY__TYPEDEF ( Clay__WrappedTextLineArraySlice , struct
2024-12-28 06:15:22 +00:00
{
2024-12-30 00:09:14 +00:00
uint32_t length ;
Clay__WrappedTextLine * internalArray ;
} ) ;
2024-12-28 06:15:22 +00:00
Clay__WrappedTextLineArray Clay__WrappedTextLineArray_Allocate_Arena ( uint32_t capacity , Clay_Arena * arena ) {
return CLAY__INIT ( Clay__WrappedTextLineArray ) { . capacity = capacity , . length = 0 , . internalArray = ( Clay__WrappedTextLine * ) Clay__Array_Allocate_Arena ( capacity , sizeof ( Clay__WrappedTextLine ) , CLAY__ALIGNMENT ( Clay__WrappedTextLine ) , arena ) } ;
}
Clay__WrappedTextLine * Clay__WrappedTextLineArray_Add ( Clay__WrappedTextLineArray * array , Clay__WrappedTextLine item ) {
2024-12-30 00:09:14 +00:00
if ( Clay__Array_AddCapacityCheck ( array - > length , array - > capacity ) ) {
array - > internalArray [ array - > length + + ] = item ;
return & array - > internalArray [ array - > length - 1 ] ;
}
return & CLAY__WRAPPED_TEXT_LINE_DEFAULT ;
2024-12-28 06:15:22 +00:00
}
Clay__WrappedTextLine * Clay__WrappedTextLineArray_Get ( Clay__WrappedTextLineArray * array , int index ) {
return Clay__Array_RangeCheck ( index , array - > length ) ? & array - > internalArray [ index ] : & CLAY__WRAPPED_TEXT_LINE_DEFAULT ;
}
# pragma endregion
// __GENERATED__ template
2024-12-30 00:09:14 +00:00
CLAY__TYPEDEF ( Clay__TextElementData , struct {
2024-10-22 07:41:35 +00:00
Clay_String text ;
2024-09-16 09:34:59 +00:00
Clay_Dimensions preferredDimensions ;
2024-10-22 07:41:35 +00:00
uint32_t elementIndex ;
2024-12-28 06:15:22 +00:00
Clay__WrappedTextLineArraySlice wrappedLines ;
2024-12-30 00:09:14 +00:00
} ) ;
2024-09-16 09:34:59 +00:00
2024-12-30 00:09:14 +00:00
Clay__TextElementData CLAY__TEXT_ELEMENT_DATA_DEFAULT = { 0 } ;
2024-09-16 09:34:59 +00:00
2024-10-22 07:41:35 +00:00
// __GENERATED__ template array_define,array_allocate,array_get,array_add TYPE=Clay__TextElementData NAME=Clay__TextElementDataArray DEFAULT_VALUE=&CLAY__TEXT_ELEMENT_DATA_DEFAULT
2024-09-16 09:34:59 +00:00
# pragma region generated
2024-12-30 00:09:14 +00:00
CLAY__TYPEDEF ( Clay__TextElementDataArray , struct
2024-09-16 09:34:59 +00:00
{
uint32_t capacity ;
uint32_t length ;
Clay__TextElementData * internalArray ;
2024-12-30 00:09:14 +00:00
} ) ;
2024-09-16 09:34:59 +00:00
Clay__TextElementDataArray Clay__TextElementDataArray_Allocate_Arena ( uint32_t capacity , Clay_Arena * arena ) {
2024-09-28 23:53:46 +00:00
return CLAY__INIT ( Clay__TextElementDataArray ) { . capacity = capacity , . length = 0 , . internalArray = ( Clay__TextElementData * ) Clay__Array_Allocate_Arena ( capacity , sizeof ( Clay__TextElementData ) , CLAY__ALIGNMENT ( Clay__TextElementData ) , arena ) } ;
2024-09-16 09:34:59 +00:00
}
Clay__TextElementData * Clay__TextElementDataArray_Get ( Clay__TextElementDataArray * array , int index ) {
return Clay__Array_RangeCheck ( index , array - > length ) ? & array - > internalArray [ index ] : & CLAY__TEXT_ELEMENT_DATA_DEFAULT ;
}
Clay__TextElementData * Clay__TextElementDataArray_Add ( Clay__TextElementDataArray * array , Clay__TextElementData item ) {
2024-12-26 06:58:59 +00:00
if ( Clay__Array_AddCapacityCheck ( array - > length , array - > capacity ) ) {
2024-09-16 09:34:59 +00:00
array - > internalArray [ array - > length + + ] = item ;
return & array - > internalArray [ array - > length - 1 ] ;
}
return & CLAY__TEXT_ELEMENT_DATA_DEFAULT ;
}
# pragma endregion
// __GENERATED__ template
2024-12-30 00:09:14 +00:00
Clay_BorderElementConfig CLAY__BORDER_ELEMENT_CONFIG_DEFAULT = { 0 } ;
2024-08-23 04:05:23 +00:00
2024-12-01 08:05:56 +00:00
// __GENERATED__ template array_define,array_allocate,array_add TYPE=Clay_BorderElementConfig NAME=Clay__BorderElementConfigArray DEFAULT_VALUE=&CLAY__BORDER_ELEMENT_CONFIG_DEFAULT
2024-08-23 04:05:23 +00:00
# pragma region generated
2024-12-30 00:09:14 +00:00
CLAY__TYPEDEF ( Clay__BorderElementConfigArray , struct
2024-08-23 04:05:23 +00:00
{
uint32_t capacity ;
uint32_t length ;
2024-08-30 06:44:36 +00:00
Clay_BorderElementConfig * internalArray ;
2024-12-30 00:09:14 +00:00
} ) ;
2024-08-30 06:44:36 +00:00
Clay__BorderElementConfigArray Clay__BorderElementConfigArray_Allocate_Arena ( uint32_t capacity , Clay_Arena * arena ) {
2024-09-28 23:53:46 +00:00
return CLAY__INIT ( Clay__BorderElementConfigArray ) { . capacity = capacity , . length = 0 , . internalArray = ( Clay_BorderElementConfig * ) Clay__Array_Allocate_Arena ( capacity , sizeof ( Clay_BorderElementConfig ) , CLAY__ALIGNMENT ( Clay_BorderElementConfig ) , arena ) } ;
2024-08-23 04:05:23 +00:00
}
2024-08-30 06:44:36 +00:00
Clay_BorderElementConfig * Clay__BorderElementConfigArray_Add ( Clay__BorderElementConfigArray * array , Clay_BorderElementConfig item ) {
2024-12-26 06:58:59 +00:00
if ( Clay__Array_AddCapacityCheck ( array - > length , array - > capacity ) ) {
2024-08-23 04:05:23 +00:00
array - > internalArray [ array - > length + + ] = item ;
return & array - > internalArray [ array - > length - 1 ] ;
}
2024-12-01 08:05:56 +00:00
return & CLAY__BORDER_ELEMENT_CONFIG_DEFAULT ;
2024-08-23 04:05:23 +00:00
}
# pragma endregion
// __GENERATED__ template
2024-12-30 00:09:14 +00:00
CLAY__TYPEDEF ( Clay__LayoutElementChildren , struct {
2024-09-16 09:34:59 +00:00
int32_t * elements ;
2024-08-23 04:05:23 +00:00
uint16_t length ;
2024-12-30 00:09:14 +00:00
} ) ;
2024-08-23 04:05:23 +00:00
2024-12-30 00:09:14 +00:00
CLAY__TYPEDEF ( Clay_LayoutElement , struct {
2024-08-23 04:05:23 +00:00
union {
Clay__LayoutElementChildren children ;
2024-10-22 07:41:35 +00:00
Clay__TextElementData * textElementData ;
2024-12-30 00:09:14 +00:00
} childrenOrTextContent ;
2024-08-23 04:05:23 +00:00
Clay_Dimensions dimensions ;
Clay_Dimensions minDimensions ;
Clay_LayoutConfig * layoutConfig ;
2024-10-22 07:41:35 +00:00
Clay__ElementConfigArraySlice elementConfigs ;
uint32_t configsEnabled ;
2024-08-23 04:05:23 +00:00
uint32_t id ;
2024-12-30 00:09:14 +00:00
} ) ;
2024-08-23 04:05:23 +00:00
2024-12-30 00:09:14 +00:00
Clay_LayoutElement CLAY__LAYOUT_ELEMENT_DEFAULT = { 0 } ;
2024-08-23 04:05:23 +00:00
2024-10-22 07:41:35 +00:00
// __GENERATED__ template array_define,array_allocate,array_add,array_get TYPE=Clay_LayoutElement NAME=Clay_LayoutElementArray DEFAULT_VALUE=&CLAY__LAYOUT_ELEMENT_DEFAULT
2024-08-23 04:05:23 +00:00
# pragma region generated
2024-12-30 00:09:14 +00:00
CLAY__TYPEDEF ( Clay_LayoutElementArray , struct
2024-08-23 04:05:23 +00:00
{
uint32_t capacity ;
uint32_t length ;
Clay_LayoutElement * internalArray ;
2024-12-30 00:09:14 +00:00
} ) ;
2024-08-23 04:05:23 +00:00
Clay_LayoutElementArray Clay_LayoutElementArray_Allocate_Arena ( uint32_t capacity , Clay_Arena * arena ) {
2024-09-28 23:53:46 +00:00
return CLAY__INIT ( Clay_LayoutElementArray ) { . capacity = capacity , . length = 0 , . internalArray = ( Clay_LayoutElement * ) Clay__Array_Allocate_Arena ( capacity , sizeof ( Clay_LayoutElement ) , CLAY__ALIGNMENT ( Clay_LayoutElement ) , arena ) } ;
2024-08-23 04:05:23 +00:00
}
Clay_LayoutElement * Clay_LayoutElementArray_Add ( Clay_LayoutElementArray * array , Clay_LayoutElement item ) {
2024-12-26 06:58:59 +00:00
if ( Clay__Array_AddCapacityCheck ( array - > length , array - > capacity ) ) {
2024-08-23 04:05:23 +00:00
array - > internalArray [ array - > length + + ] = item ;
return & array - > internalArray [ array - > length - 1 ] ;
}
return & CLAY__LAYOUT_ELEMENT_DEFAULT ;
}
Clay_LayoutElement * Clay_LayoutElementArray_Get ( Clay_LayoutElementArray * array , int index ) {
return Clay__Array_RangeCheck ( index , array - > length ) ? & array - > internalArray [ index ] : & CLAY__LAYOUT_ELEMENT_DEFAULT ;
}
# pragma endregion
// __GENERATED__ template
2024-12-30 00:09:14 +00:00
// __GENERATED__ template array_define,array_allocate_pointer,array_add,array_get_value,array_remove_swapback TYPE=Clay_LayoutElement* NAME=Clay__LayoutElementPointerArray DEFAULT_VALUE=CLAY__NULL
2024-08-23 04:05:23 +00:00
# pragma region generated
2024-12-30 00:09:14 +00:00
CLAY__TYPEDEF ( Clay__LayoutElementPointerArray , struct
2024-08-23 04:05:23 +00:00
{
uint32_t capacity ;
uint32_t length ;
Clay_LayoutElement * * internalArray ;
2024-12-30 00:09:14 +00:00
} ) ;
2024-08-23 04:05:23 +00:00
Clay__LayoutElementPointerArray Clay__LayoutElementPointerArray_Allocate_Arena ( uint32_t capacity , Clay_Arena * arena ) {
2024-12-30 00:09:14 +00:00
return CLAY__INIT ( Clay__LayoutElementPointerArray ) { . capacity = capacity , . length = 0 , . internalArray = ( Clay_LayoutElement * * ) Clay__Array_Allocate_Arena ( capacity , sizeof ( Clay_LayoutElement * ) , CLAY__POINTER_ALIGNMENT , arena ) } ;
2024-08-23 04:05:23 +00:00
}
Clay_LayoutElement * * Clay__LayoutElementPointerArray_Add ( Clay__LayoutElementPointerArray * array , Clay_LayoutElement * item ) {
2024-12-26 06:58:59 +00:00
if ( Clay__Array_AddCapacityCheck ( array - > length , array - > capacity ) ) {
2024-08-23 04:05:23 +00:00
array - > internalArray [ array - > length + + ] = item ;
return & array - > internalArray [ array - > length - 1 ] ;
}
return CLAY__NULL ;
}
2024-10-22 07:41:35 +00:00
Clay_LayoutElement * Clay__LayoutElementPointerArray_Get ( Clay__LayoutElementPointerArray * array , int index ) {
return Clay__Array_RangeCheck ( index , array - > length ) ? array - > internalArray [ index ] : CLAY__NULL ;
2024-08-23 04:05:23 +00:00
}
Clay_LayoutElement * Clay__LayoutElementPointerArray_RemoveSwapback ( Clay__LayoutElementPointerArray * array , int index ) {
if ( Clay__Array_RangeCheck ( index , array - > length ) ) {
array - > length - - ;
Clay_LayoutElement * removed = array - > internalArray [ index ] ;
array - > internalArray [ index ] = array - > internalArray [ array - > length ] ;
return removed ;
}
return CLAY__NULL ;
}
# pragma endregion
// __GENERATED__ template
2024-12-30 00:09:14 +00:00
Clay_RenderCommand CLAY__RENDER_COMMAND_DEFAULT = { 0 } ;
2024-08-23 04:05:23 +00:00
2024-10-22 07:41:35 +00:00
// __GENERATED__ template array_allocate,array_add,array_get TYPE=Clay_RenderCommand NAME=Clay_RenderCommandArray DEFAULT_VALUE=&CLAY__RENDER_COMMAND_DEFAULT
2024-08-23 04:05:23 +00:00
# pragma region generated
Clay_RenderCommandArray Clay_RenderCommandArray_Allocate_Arena ( uint32_t capacity , Clay_Arena * arena ) {
2024-09-28 23:53:46 +00:00
return CLAY__INIT ( Clay_RenderCommandArray ) { . capacity = capacity , . length = 0 , . internalArray = ( Clay_RenderCommand * ) Clay__Array_Allocate_Arena ( capacity , sizeof ( Clay_RenderCommand ) , CLAY__ALIGNMENT ( Clay_RenderCommand ) , arena ) } ;
2024-08-23 04:05:23 +00:00
}
Clay_RenderCommand * Clay_RenderCommandArray_Add ( Clay_RenderCommandArray * array , Clay_RenderCommand item ) {
2024-12-26 06:58:59 +00:00
if ( Clay__Array_AddCapacityCheck ( array - > length , array - > capacity ) ) {
2024-08-23 04:05:23 +00:00
array - > internalArray [ array - > length + + ] = item ;
return & array - > internalArray [ array - > length - 1 ] ;
}
return & CLAY__RENDER_COMMAND_DEFAULT ;
}
2024-12-26 06:58:59 +00:00
Clay_RenderCommand * Clay_RenderCommandArray_Get ( Clay_RenderCommandArray * array , int index ) {
2024-08-23 04:05:23 +00:00
return Clay__Array_RangeCheck ( index , array - > length ) ? & array - > internalArray [ index ] : & CLAY__RENDER_COMMAND_DEFAULT ;
}
# pragma endregion
// __GENERATED__ template
2024-12-30 00:09:14 +00:00
CLAY__TYPEDEF ( Clay__ScrollContainerDataInternal , struct {
2024-08-23 04:05:23 +00:00
Clay_LayoutElement * layoutElement ;
2024-08-28 04:13:28 +00:00
Clay_BoundingBox boundingBox ;
2024-08-23 04:05:23 +00:00
Clay_Dimensions contentSize ;
Clay_Vector2 scrollOrigin ;
Clay_Vector2 pointerOrigin ;
Clay_Vector2 scrollMomentum ;
Clay_Vector2 scrollPosition ;
2024-08-24 09:22:54 +00:00
Clay_Vector2 previousDelta ;
2024-08-23 04:05:23 +00:00
float momentumTime ;
uint32_t elementId ;
bool openThisFrame ;
bool pointerScrollActive ;
2024-12-30 00:09:14 +00:00
} ) ;
2024-08-23 04:05:23 +00:00
2024-12-30 00:09:14 +00:00
Clay__ScrollContainerDataInternal CLAY__SCROLL_CONTAINER_DEFAULT = { 0 } ;
2024-08-23 04:05:23 +00:00
2024-10-22 07:41:35 +00:00
// __GENERATED__ template array_define,array_allocate,array_add,array_get TYPE=Clay__ScrollContainerDataInternal NAME=Clay__ScrollContainerDataInternalArray DEFAULT_VALUE=&CLAY__SCROLL_CONTAINER_DEFAULT
2024-08-23 04:05:23 +00:00
# pragma region generated
2024-12-30 00:09:14 +00:00
CLAY__TYPEDEF ( Clay__ScrollContainerDataInternalArray , struct
2024-08-23 04:05:23 +00:00
{
uint32_t capacity ;
uint32_t length ;
2024-08-26 07:05:43 +00:00
Clay__ScrollContainerDataInternal * internalArray ;
2024-12-30 00:09:14 +00:00
} ) ;
2024-08-26 07:05:43 +00:00
Clay__ScrollContainerDataInternalArray Clay__ScrollContainerDataInternalArray_Allocate_Arena ( uint32_t capacity , Clay_Arena * arena ) {
2024-09-28 23:53:46 +00:00
return CLAY__INIT ( Clay__ScrollContainerDataInternalArray ) { . capacity = capacity , . length = 0 , . internalArray = ( Clay__ScrollContainerDataInternal * ) Clay__Array_Allocate_Arena ( capacity , sizeof ( Clay__ScrollContainerDataInternal ) , CLAY__ALIGNMENT ( Clay__ScrollContainerDataInternal ) , arena ) } ;
2024-08-23 04:05:23 +00:00
}
2024-08-26 07:05:43 +00:00
Clay__ScrollContainerDataInternal * Clay__ScrollContainerDataInternalArray_Add ( Clay__ScrollContainerDataInternalArray * array , Clay__ScrollContainerDataInternal item ) {
2024-12-26 06:58:59 +00:00
if ( Clay__Array_AddCapacityCheck ( array - > length , array - > capacity ) ) {
2024-08-23 04:05:23 +00:00
array - > internalArray [ array - > length + + ] = item ;
return & array - > internalArray [ array - > length - 1 ] ;
}
return & CLAY__SCROLL_CONTAINER_DEFAULT ;
}
2024-08-26 07:05:43 +00:00
Clay__ScrollContainerDataInternal * Clay__ScrollContainerDataInternalArray_Get ( Clay__ScrollContainerDataInternalArray * array , int index ) {
2024-08-23 04:05:23 +00:00
return Clay__Array_RangeCheck ( index , array - > length ) ? & array - > internalArray [ index ] : & CLAY__SCROLL_CONTAINER_DEFAULT ;
}
# pragma endregion
// __GENERATED__ template
2024-08-26 07:05:43 +00:00
// __GENERATED__ template array_remove_swapback TYPE=Clay__ScrollContainerDataInternal NAME=Clay__ScrollContainerDataInternalArray DEFAULT_VALUE=CLAY__SCROLL_CONTAINER_DEFAULT
2024-08-23 04:05:23 +00:00
# pragma region generated
2024-08-26 07:05:43 +00:00
Clay__ScrollContainerDataInternal Clay__ScrollContainerDataInternalArray_RemoveSwapback ( Clay__ScrollContainerDataInternalArray * array , int index ) {
2024-08-23 04:05:23 +00:00
if ( Clay__Array_RangeCheck ( index , array - > length ) ) {
array - > length - - ;
2024-08-26 07:05:43 +00:00
Clay__ScrollContainerDataInternal removed = array - > internalArray [ index ] ;
2024-08-23 04:05:23 +00:00
array - > internalArray [ index ] = array - > internalArray [ array - > length ] ;
return removed ;
}
return CLAY__SCROLL_CONTAINER_DEFAULT ;
}
# pragma endregion
// __GENERATED__ template
2024-12-30 00:09:14 +00:00
CLAY__TYPEDEF ( Clay__DebugElementData , struct {
2024-09-16 09:34:59 +00:00
bool collision ;
bool collapsed ;
2024-12-30 00:09:14 +00:00
} ) ;
2024-09-16 09:34:59 +00:00
2024-12-30 00:09:14 +00:00
Clay__DebugElementData CLAY__DEBUG_ELEMENT_DATA_DEFAULT = { 0 } ;
2024-09-16 09:34:59 +00:00
2024-10-22 07:41:35 +00:00
// __GENERATED__ template array_define,array_allocate,array_add,array_get TYPE=Clay__DebugElementData NAME=Clay__DebugElementDataArray DEFAULT_VALUE=&CLAY__DEBUG_ELEMENT_DATA_DEFAULT
2024-09-16 09:34:59 +00:00
# pragma region generated
2024-12-30 00:09:14 +00:00
CLAY__TYPEDEF ( Clay__DebugElementDataArray , struct
2024-09-16 09:34:59 +00:00
{
uint32_t capacity ;
uint32_t length ;
Clay__DebugElementData * internalArray ;
2024-12-30 00:09:14 +00:00
} ) ;
2024-09-16 09:34:59 +00:00
Clay__DebugElementDataArray Clay__DebugElementDataArray_Allocate_Arena ( uint32_t capacity , Clay_Arena * arena ) {
2024-09-28 23:53:46 +00:00
return CLAY__INIT ( Clay__DebugElementDataArray ) { . capacity = capacity , . length = 0 , . internalArray = ( Clay__DebugElementData * ) Clay__Array_Allocate_Arena ( capacity , sizeof ( Clay__DebugElementData ) , CLAY__ALIGNMENT ( Clay__DebugElementData ) , arena ) } ;
2024-09-16 09:34:59 +00:00
}
Clay__DebugElementData * Clay__DebugElementDataArray_Add ( Clay__DebugElementDataArray * array , Clay__DebugElementData item ) {
2024-12-26 06:58:59 +00:00
if ( Clay__Array_AddCapacityCheck ( array - > length , array - > capacity ) ) {
2024-09-16 09:34:59 +00:00
array - > internalArray [ array - > length + + ] = item ;
return & array - > internalArray [ array - > length - 1 ] ;
}
return & CLAY__DEBUG_ELEMENT_DATA_DEFAULT ;
}
Clay__DebugElementData * Clay__DebugElementDataArray_Get ( Clay__DebugElementDataArray * array , int index ) {
return Clay__Array_RangeCheck ( index , array - > length ) ? & array - > internalArray [ index ] : & CLAY__DEBUG_ELEMENT_DATA_DEFAULT ;
}
# pragma endregion
// __GENERATED__ template
2024-12-30 00:09:14 +00:00
CLAY__TYPEDEF ( Clay_LayoutElementHashMapItem , struct { // todo get this struct into a single cache line
2024-08-28 04:13:28 +00:00
Clay_BoundingBox boundingBox ;
2024-09-16 09:34:59 +00:00
Clay_ElementId elementId ;
2024-08-23 04:05:23 +00:00
Clay_LayoutElement * layoutElement ;
2024-10-22 20:44:39 +00:00
void ( * onHoverFunction ) ( Clay_ElementId elementId , Clay_PointerData pointerInfo , intptr_t userData ) ;
2024-10-22 07:41:35 +00:00
intptr_t hoverFunctionUserData ;
2024-08-23 04:05:23 +00:00
int32_t nextIndex ;
2024-09-16 09:34:59 +00:00
uint32_t generation ;
Clay__DebugElementData * debugData ;
2024-12-30 00:09:14 +00:00
} ) ;
2024-08-23 04:05:23 +00:00
2024-12-30 00:09:14 +00:00
Clay_LayoutElementHashMapItem CLAY__LAYOUT_ELEMENT_HASH_MAP_ITEM_DEFAULT = { . layoutElement = & CLAY__LAYOUT_ELEMENT_DEFAULT } ;
2024-08-23 04:05:23 +00:00
2024-10-22 07:41:35 +00:00
// __GENERATED__ template array_define,array_allocate,array_get,array_add TYPE=Clay_LayoutElementHashMapItem NAME=Clay__LayoutElementHashMapItemArray DEFAULT_VALUE=&CLAY__LAYOUT_ELEMENT_HASH_MAP_ITEM_DEFAULT
2024-08-23 04:05:23 +00:00
# pragma region generated
2024-12-30 00:09:14 +00:00
CLAY__TYPEDEF ( Clay__LayoutElementHashMapItemArray , struct
2024-08-23 04:05:23 +00:00
{
uint32_t capacity ;
uint32_t length ;
Clay_LayoutElementHashMapItem * internalArray ;
2024-12-30 00:09:14 +00:00
} ) ;
2024-08-23 04:05:23 +00:00
Clay__LayoutElementHashMapItemArray Clay__LayoutElementHashMapItemArray_Allocate_Arena ( uint32_t capacity , Clay_Arena * arena ) {
2024-09-28 23:53:46 +00:00
return CLAY__INIT ( Clay__LayoutElementHashMapItemArray ) { . capacity = capacity , . length = 0 , . internalArray = ( Clay_LayoutElementHashMapItem * ) Clay__Array_Allocate_Arena ( capacity , sizeof ( Clay_LayoutElementHashMapItem ) , CLAY__ALIGNMENT ( Clay_LayoutElementHashMapItem ) , arena ) } ;
2024-08-23 04:05:23 +00:00
}
Clay_LayoutElementHashMapItem * Clay__LayoutElementHashMapItemArray_Get ( Clay__LayoutElementHashMapItemArray * array , int index ) {
return Clay__Array_RangeCheck ( index , array - > length ) ? & array - > internalArray [ index ] : & CLAY__LAYOUT_ELEMENT_HASH_MAP_ITEM_DEFAULT ;
}
Clay_LayoutElementHashMapItem * Clay__LayoutElementHashMapItemArray_Add ( Clay__LayoutElementHashMapItemArray * array , Clay_LayoutElementHashMapItem item ) {
2024-12-26 06:58:59 +00:00
if ( Clay__Array_AddCapacityCheck ( array - > length , array - > capacity ) ) {
2024-08-23 04:05:23 +00:00
array - > internalArray [ array - > length + + ] = item ;
return & array - > internalArray [ array - > length - 1 ] ;
}
return & CLAY__LAYOUT_ELEMENT_HASH_MAP_ITEM_DEFAULT ;
}
# pragma endregion
// __GENERATED__ template
2024-12-30 00:09:14 +00:00
CLAY__TYPEDEF ( Clay__MeasuredWord , struct {
2024-10-22 07:41:35 +00:00
uint32_t startOffset ;
uint32_t length ;
float width ;
2024-12-01 08:05:56 +00:00
int32_t next ;
2024-12-30 00:09:14 +00:00
} ) ;
2024-10-22 07:41:35 +00:00
2024-12-30 00:09:14 +00:00
Clay__MeasuredWord CLAY__MEASURED_WORD_DEFAULT = { . next = - 1 } ;
2024-10-22 07:41:35 +00:00
2024-12-01 08:05:56 +00:00
// __GENERATED__ template array_define,array_allocate,array_get,array_set,array_add TYPE=Clay__MeasuredWord NAME=Clay__MeasuredWordArray DEFAULT_VALUE=&CLAY__MEASURED_WORD_DEFAULT
2024-10-22 07:41:35 +00:00
# pragma region generated
2024-12-30 00:09:14 +00:00
CLAY__TYPEDEF ( Clay__MeasuredWordArray , struct
2024-10-22 07:41:35 +00:00
{
uint32_t capacity ;
uint32_t length ;
Clay__MeasuredWord * internalArray ;
2024-12-30 00:09:14 +00:00
} ) ;
2024-10-22 07:41:35 +00:00
Clay__MeasuredWordArray Clay__MeasuredWordArray_Allocate_Arena ( uint32_t capacity , Clay_Arena * arena ) {
return CLAY__INIT ( Clay__MeasuredWordArray ) { . capacity = capacity , . length = 0 , . internalArray = ( Clay__MeasuredWord * ) Clay__Array_Allocate_Arena ( capacity , sizeof ( Clay__MeasuredWord ) , CLAY__ALIGNMENT ( Clay__MeasuredWord ) , arena ) } ;
}
Clay__MeasuredWord * Clay__MeasuredWordArray_Get ( Clay__MeasuredWordArray * array , int index ) {
return Clay__Array_RangeCheck ( index , array - > length ) ? & array - > internalArray [ index ] : & CLAY__MEASURED_WORD_DEFAULT ;
}
2024-11-26 23:13:11 +00:00
void Clay__MeasuredWordArray_Set ( Clay__MeasuredWordArray * array , int index , Clay__MeasuredWord value ) {
2024-12-26 06:58:59 +00:00
if ( Clay__Array_RangeCheck ( index , array - > capacity ) ) {
2024-11-26 23:13:11 +00:00
array - > internalArray [ index ] = value ;
array - > length = index < array - > length ? array - > length : index + 1 ;
}
}
2024-10-22 07:41:35 +00:00
Clay__MeasuredWord * Clay__MeasuredWordArray_Add ( Clay__MeasuredWordArray * array , Clay__MeasuredWord item ) {
2024-12-26 06:58:59 +00:00
if ( Clay__Array_AddCapacityCheck ( array - > length , array - > capacity ) ) {
2024-10-22 07:41:35 +00:00
array - > internalArray [ array - > length + + ] = item ;
return & array - > internalArray [ array - > length - 1 ] ;
}
return & CLAY__MEASURED_WORD_DEFAULT ;
}
# pragma endregion
// __GENERATED__ template
2024-12-30 00:09:14 +00:00
CLAY__TYPEDEF ( Clay__MeasureTextCacheItem , struct {
2024-10-22 07:41:35 +00:00
Clay_Dimensions unwrappedDimensions ;
2024-12-01 08:05:56 +00:00
int32_t measuredWordsStartIndex ;
2024-10-22 07:41:35 +00:00
// Hash map data
2024-08-23 04:05:23 +00:00
uint32_t id ;
int32_t nextIndex ;
2024-10-15 23:11:01 +00:00
uint32_t generation ;
2024-12-30 00:09:14 +00:00
} ) ;
2024-08-23 04:05:23 +00:00
2024-12-30 00:09:14 +00:00
Clay__MeasureTextCacheItem CLAY__MEASURE_TEXT_CACHE_ITEM_DEFAULT = { . measuredWordsStartIndex = - 1 } ;
2024-08-23 04:05:23 +00:00
2024-10-22 07:41:35 +00:00
// __GENERATED__ template array_define,array_allocate,array_get,array_add,array_set TYPE=Clay__MeasureTextCacheItem NAME=Clay__MeasureTextCacheItemArray DEFAULT_VALUE=&CLAY__MEASURE_TEXT_CACHE_ITEM_DEFAULT
2024-08-23 04:05:23 +00:00
# pragma region generated
2024-12-30 00:09:14 +00:00
CLAY__TYPEDEF ( Clay__MeasureTextCacheItemArray , struct
2024-08-23 04:05:23 +00:00
{
uint32_t capacity ;
uint32_t length ;
Clay__MeasureTextCacheItem * internalArray ;
2024-12-30 00:09:14 +00:00
} ) ;
2024-08-23 04:05:23 +00:00
Clay__MeasureTextCacheItemArray Clay__MeasureTextCacheItemArray_Allocate_Arena ( uint32_t capacity , Clay_Arena * arena ) {
2024-09-28 23:53:46 +00:00
return CLAY__INIT ( Clay__MeasureTextCacheItemArray ) { . capacity = capacity , . length = 0 , . internalArray = ( Clay__MeasureTextCacheItem * ) Clay__Array_Allocate_Arena ( capacity , sizeof ( Clay__MeasureTextCacheItem ) , CLAY__ALIGNMENT ( Clay__MeasureTextCacheItem ) , arena ) } ;
2024-08-23 04:05:23 +00:00
}
Clay__MeasureTextCacheItem * Clay__MeasureTextCacheItemArray_Get ( Clay__MeasureTextCacheItemArray * array , int index ) {
return Clay__Array_RangeCheck ( index , array - > length ) ? & array - > internalArray [ index ] : & CLAY__MEASURE_TEXT_CACHE_ITEM_DEFAULT ;
}
Clay__MeasureTextCacheItem * Clay__MeasureTextCacheItemArray_Add ( Clay__MeasureTextCacheItemArray * array , Clay__MeasureTextCacheItem item ) {
2024-12-26 06:58:59 +00:00
if ( Clay__Array_AddCapacityCheck ( array - > length , array - > capacity ) ) {
2024-08-23 04:05:23 +00:00
array - > internalArray [ array - > length + + ] = item ;
return & array - > internalArray [ array - > length - 1 ] ;
}
return & CLAY__MEASURE_TEXT_CACHE_ITEM_DEFAULT ;
}
void Clay__MeasureTextCacheItemArray_Set ( Clay__MeasureTextCacheItemArray * array , int index , Clay__MeasureTextCacheItem value ) {
2024-12-26 06:58:59 +00:00
if ( Clay__Array_RangeCheck ( index , array - > capacity ) ) {
2024-08-23 04:05:23 +00:00
array - > internalArray [ index ] = value ;
array - > length = index < array - > length ? array - > length : index + 1 ;
}
}
# pragma endregion
// __GENERATED__ template
2024-10-22 07:41:35 +00:00
// __GENERATED__ template array_define,array_allocate,array_get_value,array_add_value,array_set,array_remove_swapback TYPE=int32_t NAME=Clay__int32_tArray DEFAULT_VALUE=-1
2024-08-23 04:05:23 +00:00
# pragma region generated
2024-12-30 00:09:14 +00:00
CLAY__TYPEDEF ( Clay__int32_tArray , struct
2024-08-23 04:05:23 +00:00
{
uint32_t capacity ;
uint32_t length ;
int32_t * internalArray ;
2024-12-30 00:09:14 +00:00
} ) ;
2024-08-23 04:05:23 +00:00
Clay__int32_tArray Clay__int32_tArray_Allocate_Arena ( uint32_t capacity , Clay_Arena * arena ) {
2024-09-28 23:53:46 +00:00
return CLAY__INIT ( Clay__int32_tArray ) { . capacity = capacity , . length = 0 , . internalArray = ( int32_t * ) Clay__Array_Allocate_Arena ( capacity , sizeof ( int32_t ) , CLAY__ALIGNMENT ( int32_t ) , arena ) } ;
2024-08-23 04:05:23 +00:00
}
2024-09-01 23:06:14 +00:00
int32_t Clay__int32_tArray_Get ( Clay__int32_tArray * array , int index ) {
return Clay__Array_RangeCheck ( index , array - > length ) ? array - > internalArray [ index ] : - 1 ;
2024-08-23 04:05:23 +00:00
}
2024-09-01 23:06:14 +00:00
void Clay__int32_tArray_Add ( Clay__int32_tArray * array , int32_t item ) {
2024-12-26 06:58:59 +00:00
if ( Clay__Array_AddCapacityCheck ( array - > length , array - > capacity ) ) {
2024-08-23 04:05:23 +00:00
array - > internalArray [ array - > length + + ] = item ;
}
}
void Clay__int32_tArray_Set ( Clay__int32_tArray * array , int index , int32_t value ) {
2024-12-26 06:58:59 +00:00
if ( Clay__Array_RangeCheck ( index , array - > capacity ) ) {
2024-08-23 04:05:23 +00:00
array - > internalArray [ index ] = value ;
array - > length = index < array - > length ? array - > length : index + 1 ;
}
}
2024-10-22 07:41:35 +00:00
int32_t Clay__int32_tArray_RemoveSwapback ( Clay__int32_tArray * array , int index ) {
if ( Clay__Array_RangeCheck ( index , array - > length ) ) {
array - > length - - ;
int32_t removed = array - > internalArray [ index ] ;
array - > internalArray [ index ] = array - > internalArray [ array - > length ] ;
return removed ;
}
return - 1 ;
}
2024-08-23 04:05:23 +00:00
# pragma endregion
// __GENERATED__ template
2024-12-30 00:09:14 +00:00
CLAY__TYPEDEF ( Clay__LayoutElementTreeNode , struct {
2024-08-23 04:05:23 +00:00
Clay_LayoutElement * layoutElement ;
Clay_Vector2 position ;
Clay_Vector2 nextChildOffset ;
2024-12-30 00:09:14 +00:00
} ) ;
2024-08-23 04:05:23 +00:00
2024-12-30 00:09:14 +00:00
Clay__LayoutElementTreeNode CLAY__LAYOUT_ELEMENT_TREE_NODE_DEFAULT = { 0 } ;
2024-08-23 04:05:23 +00:00
2024-10-22 07:41:35 +00:00
// __GENERATED__ template array_define,array_allocate,array_add,array_get TYPE=Clay__LayoutElementTreeNode NAME=Clay__LayoutElementTreeNodeArray DEFAULT_VALUE=&CLAY__LAYOUT_ELEMENT_TREE_NODE_DEFAULT
2024-08-23 04:05:23 +00:00
# pragma region generated
2024-12-30 00:09:14 +00:00
CLAY__TYPEDEF ( Clay__LayoutElementTreeNodeArray , struct
2024-08-23 04:05:23 +00:00
{
uint32_t capacity ;
uint32_t length ;
Clay__LayoutElementTreeNode * internalArray ;
2024-12-30 00:09:14 +00:00
} ) ;
2024-08-23 04:05:23 +00:00
Clay__LayoutElementTreeNodeArray Clay__LayoutElementTreeNodeArray_Allocate_Arena ( uint32_t capacity , Clay_Arena * arena ) {
2024-09-28 23:53:46 +00:00
return CLAY__INIT ( Clay__LayoutElementTreeNodeArray ) { . capacity = capacity , . length = 0 , . internalArray = ( Clay__LayoutElementTreeNode * ) Clay__Array_Allocate_Arena ( capacity , sizeof ( Clay__LayoutElementTreeNode ) , CLAY__ALIGNMENT ( Clay__LayoutElementTreeNode ) , arena ) } ;
2024-08-23 04:05:23 +00:00
}
Clay__LayoutElementTreeNode * Clay__LayoutElementTreeNodeArray_Add ( Clay__LayoutElementTreeNodeArray * array , Clay__LayoutElementTreeNode item ) {
2024-12-26 06:58:59 +00:00
if ( Clay__Array_AddCapacityCheck ( array - > length , array - > capacity ) ) {
2024-08-23 04:05:23 +00:00
array - > internalArray [ array - > length + + ] = item ;
return & array - > internalArray [ array - > length - 1 ] ;
}
return & CLAY__LAYOUT_ELEMENT_TREE_NODE_DEFAULT ;
}
Clay__LayoutElementTreeNode * Clay__LayoutElementTreeNodeArray_Get ( Clay__LayoutElementTreeNodeArray * array , int index ) {
return Clay__Array_RangeCheck ( index , array - > length ) ? & array - > internalArray [ index ] : & CLAY__LAYOUT_ELEMENT_TREE_NODE_DEFAULT ;
}
# pragma endregion
// __GENERATED__ template
2024-12-30 00:09:14 +00:00
CLAY__TYPEDEF ( Clay__LayoutElementTreeRoot , struct {
2024-09-16 09:34:59 +00:00
uint32_t layoutElementIndex ;
2024-08-23 04:05:23 +00:00
uint32_t parentId ; // This can be zero in the case of the root layout tree
uint32_t clipElementId ; // This can be zero if there is no clip element
uint32_t zIndex ;
2024-12-18 22:35:18 +00:00
Clay_Vector2 pointerOffset ; // Only used when scroll containers are managed externally
2024-12-30 00:09:14 +00:00
} ) ;
2024-08-23 04:05:23 +00:00
2024-12-30 00:09:14 +00:00
Clay__LayoutElementTreeRoot CLAY__LAYOUT_ELEMENT_TREE_ROOT_DEFAULT = { 0 } ;
2024-08-23 04:05:23 +00:00
2024-10-22 07:41:35 +00:00
// __GENERATED__ template array_define,array_allocate,array_add,array_get TYPE=Clay__LayoutElementTreeRoot NAME=Clay__LayoutElementTreeRootArray DEFAULT_VALUE=&CLAY__LAYOUT_ELEMENT_TREE_ROOT_DEFAULT
2024-08-23 04:05:23 +00:00
# pragma region generated
2024-12-30 00:09:14 +00:00
CLAY__TYPEDEF ( Clay__LayoutElementTreeRootArray , struct
2024-08-23 04:05:23 +00:00
{
uint32_t capacity ;
uint32_t length ;
Clay__LayoutElementTreeRoot * internalArray ;
2024-12-30 00:09:14 +00:00
} ) ;
2024-08-23 04:05:23 +00:00
Clay__LayoutElementTreeRootArray Clay__LayoutElementTreeRootArray_Allocate_Arena ( uint32_t capacity , Clay_Arena * arena ) {
2024-09-28 23:53:46 +00:00
return CLAY__INIT ( Clay__LayoutElementTreeRootArray ) { . capacity = capacity , . length = 0 , . internalArray = ( Clay__LayoutElementTreeRoot * ) Clay__Array_Allocate_Arena ( capacity , sizeof ( Clay__LayoutElementTreeRoot ) , CLAY__ALIGNMENT ( Clay__LayoutElementTreeRoot ) , arena ) } ;
2024-08-23 04:05:23 +00:00
}
Clay__LayoutElementTreeRoot * Clay__LayoutElementTreeRootArray_Add ( Clay__LayoutElementTreeRootArray * array , Clay__LayoutElementTreeRoot item ) {
2024-12-26 06:58:59 +00:00
if ( Clay__Array_AddCapacityCheck ( array - > length , array - > capacity ) ) {
2024-08-23 04:05:23 +00:00
array - > internalArray [ array - > length + + ] = item ;
return & array - > internalArray [ array - > length - 1 ] ;
}
return & CLAY__LAYOUT_ELEMENT_TREE_ROOT_DEFAULT ;
}
Clay__LayoutElementTreeRoot * Clay__LayoutElementTreeRootArray_Get ( Clay__LayoutElementTreeRootArray * array , int index ) {
return Clay__Array_RangeCheck ( index , array - > length ) ? & array - > internalArray [ index ] : & CLAY__LAYOUT_ELEMENT_TREE_ROOT_DEFAULT ;
}
# pragma endregion
// __GENERATED__ template
2024-10-22 07:41:35 +00:00
// __GENERATED__ template array_define,array_allocate TYPE=uint8_t NAME=Clay__CharArray DEFAULT_VALUE=0
2024-09-16 09:34:59 +00:00
# pragma region generated
2024-12-30 00:09:14 +00:00
CLAY__TYPEDEF ( Clay__CharArray , struct
2024-09-16 09:34:59 +00:00
{
uint32_t capacity ;
uint32_t length ;
uint8_t * internalArray ;
2024-12-30 00:09:14 +00:00
} ) ;
2024-09-16 09:34:59 +00:00
Clay__CharArray Clay__CharArray_Allocate_Arena ( uint32_t capacity , Clay_Arena * arena ) {
2024-09-28 23:53:46 +00:00
return CLAY__INIT ( Clay__CharArray ) { . capacity = capacity , . length = 0 , . internalArray = ( uint8_t * ) Clay__Array_Allocate_Arena ( capacity , sizeof ( uint8_t ) , CLAY__ALIGNMENT ( uint8_t ) , arena ) } ;
2024-09-16 09:34:59 +00:00
}
# pragma endregion
// __GENERATED__ template
Clay_String Clay__WriteStringToCharBuffer ( Clay__CharArray * buffer , Clay_String string ) {
2024-12-30 00:09:14 +00:00
for ( size_t i = 0 ; i < string . length ; i + + ) {
2024-09-16 09:34:59 +00:00
buffer - > internalArray [ buffer - > length + i ] = string . chars [ i ] ;
}
buffer - > length + = string . length ;
2024-09-28 23:53:46 +00:00
return CLAY__INIT ( Clay_String ) { . length = string . length , . chars = ( const char * ) ( buffer - > internalArray + buffer - > length - string . length ) } ;
2024-09-16 09:34:59 +00:00
}
2024-11-26 23:13:11 +00:00
// Global Variable Definitions ----------------------------------------------
2024-12-30 00:09:14 +00:00
Clay_PointerData Clay__pointerInfo = { . position = { - 1 , - 1 } } ;
Clay_Dimensions Clay__layoutDimensions = { 0 } ;
Clay_ElementId Clay__dynamicElementIndexBaseHash = { . id = 128476991 , . stringId = { . length = 8 , . chars = " Auto ID " } } ;
2024-09-16 09:34:59 +00:00
uint32_t Clay__dynamicElementIndex = 0 ;
bool Clay__debugModeEnabled = false ;
2024-12-18 22:35:18 +00:00
bool Clay__disableCulling = false ;
bool Clay__externalScrollHandlingEnabled = false ;
2024-09-16 09:34:59 +00:00
uint32_t Clay__debugSelectedElementId = 0 ;
uint32_t Clay__debugViewWidth = 400 ;
2024-12-30 00:09:14 +00:00
Clay_Color Clay__debugViewHighlightColor = { 168 , 66 , 28 , 100 } ;
2024-09-16 09:34:59 +00:00
uint32_t Clay__generation = 0 ;
2024-08-23 04:05:23 +00:00
uint64_t Clay__arenaResetOffset = 0 ;
Clay_Arena Clay__internalArena ;
// Layout Elements / Render Commands
Clay_LayoutElementArray Clay__layoutElements ;
Clay_RenderCommandArray Clay__renderCommands ;
2024-10-22 07:41:35 +00:00
Clay__int32_tArray Clay__openLayoutElementStack ;
2024-09-16 09:34:59 +00:00
Clay__int32_tArray Clay__layoutElementChildren ;
Clay__int32_tArray Clay__layoutElementChildrenBuffer ;
Clay__TextElementDataArray Clay__textElementData ;
2024-08-23 04:05:23 +00:00
Clay__LayoutElementPointerArray Clay__imageElementPointers ;
2024-10-22 07:41:35 +00:00
Clay__int32_tArray Clay__reusableElementIndexBuffer ;
2024-12-18 22:35:18 +00:00
Clay__int32_tArray Clay__layoutElementClipElementIds ;
2024-08-23 04:05:23 +00:00
// Configs
2024-08-30 06:44:36 +00:00
Clay__LayoutConfigArray Clay__layoutConfigs ;
2024-10-22 07:41:35 +00:00
Clay__ElementConfigArray Clay__elementConfigBuffer ;
Clay__ElementConfigArray Clay__elementConfigs ;
2024-08-30 06:44:36 +00:00
Clay__RectangleElementConfigArray Clay__rectangleElementConfigs ;
Clay__TextElementConfigArray Clay__textElementConfigs ;
Clay__ImageElementConfigArray Clay__imageElementConfigs ;
Clay__FloatingElementConfigArray Clay__floatingElementConfigs ;
Clay__ScrollElementConfigArray Clay__scrollElementConfigs ;
Clay__CustomElementConfigArray Clay__customElementConfigs ;
Clay__BorderElementConfigArray Clay__borderElementConfigs ;
2024-08-23 04:05:23 +00:00
// Misc Data Structures
2024-09-16 09:34:59 +00:00
Clay__StringArray Clay__layoutElementIdStrings ;
2024-12-28 06:15:22 +00:00
Clay__WrappedTextLineArray Clay__wrappedTextLines ;
2024-08-23 04:05:23 +00:00
Clay__LayoutElementTreeNodeArray Clay__layoutElementTreeNodeArray1 ;
Clay__LayoutElementTreeRootArray Clay__layoutElementTreeRoots ;
Clay__LayoutElementHashMapItemArray Clay__layoutElementsHashMapInternal ;
Clay__int32_tArray Clay__layoutElementsHashMap ;
Clay__MeasureTextCacheItemArray Clay__measureTextHashMapInternal ;
2024-10-15 23:11:01 +00:00
Clay__int32_tArray Clay__measureTextHashMapInternalFreeList ;
2024-08-23 04:05:23 +00:00
Clay__int32_tArray Clay__measureTextHashMap ;
2024-10-22 07:41:35 +00:00
Clay__MeasuredWordArray Clay__measuredWords ;
2024-11-26 23:13:11 +00:00
Clay__int32_tArray Clay__measuredWordsFreeList ;
2024-08-23 04:05:23 +00:00
Clay__int32_tArray Clay__openClipElementStack ;
2024-09-16 09:34:59 +00:00
Clay__ElementIdArray Clay__pointerOverIds ;
2024-08-26 07:05:43 +00:00
Clay__ScrollContainerDataInternalArray Clay__scrollContainerDatas ;
2024-08-23 04:05:23 +00:00
Clay__BoolArray Clay__treeNodeVisited ;
2024-09-16 09:34:59 +00:00
Clay__CharArray Clay__dynamicStringData ;
Clay__DebugElementDataArray Clay__debugElementData ;
2024-08-23 04:05:23 +00:00
2024-12-30 00:11:32 +00:00
# ifdef CLAY_WASM
2024-08-23 04:05:23 +00:00
__attribute__ ( ( import_module ( " clay " ) , import_name ( " measureTextFunction " ) ) ) Clay_Dimensions Clay__MeasureText ( Clay_String * text , Clay_TextElementConfig * config ) ;
2024-12-18 22:35:18 +00:00
__attribute__ ( ( import_module ( " clay " ) , import_name ( " queryScrollOffsetFunction " ) ) ) Clay_Vector2 Clay__QueryScrollOffset ( uint32_t elementId ) ;
2024-08-23 04:05:23 +00:00
# else
Clay_Dimensions ( * Clay__MeasureText ) ( Clay_String * text , Clay_TextElementConfig * config ) ;
2024-12-18 22:35:18 +00:00
Clay_Vector2 ( * Clay__QueryScrollOffset ) ( uint32_t elementId ) ;
2024-08-23 04:05:23 +00:00
# endif
2024-12-30 00:09:14 +00:00
Clay_LayoutElement * Clay__GetOpenLayoutElement ( void ) {
2024-10-22 07:41:35 +00:00
return Clay_LayoutElementArray_Get ( & Clay__layoutElements , Clay__int32_tArray_Get ( & Clay__openLayoutElementStack , Clay__openLayoutElementStack . length - 1 ) ) ;
}
bool Clay__ElementHasConfig ( Clay_LayoutElement * element , Clay__ElementConfigType type ) {
return ( element - > configsEnabled & type ) ;
}
Clay_ElementConfigUnion Clay__FindElementConfigWithType ( Clay_LayoutElement * element , Clay__ElementConfigType type ) {
2024-12-30 00:09:14 +00:00
for ( uint32_t i = 0 ; i < element - > elementConfigs . length ; i + + ) {
2024-10-22 07:41:35 +00:00
Clay_ElementConfig * config = Clay__ElementConfigArraySlice_Get ( & element - > elementConfigs , i ) ;
if ( config - > type = = type ) {
return config - > config ;
}
}
return CLAY__INIT ( Clay_ElementConfigUnion ) { NULL } ;
}
Clay_ElementId Clay__HashNumber ( const uint32_t offset , const uint32_t seed ) {
uint32_t hash = seed ;
hash + = ( offset + 48 ) ;
hash + = ( hash < < 10 ) ;
hash ^ = ( hash > > 6 ) ;
hash + = ( hash < < 3 ) ;
hash ^ = ( hash > > 11 ) ;
hash + = ( hash < < 15 ) ;
return CLAY__INIT ( Clay_ElementId ) { . id = hash + 1 , . offset = offset , . baseId = seed , . stringId = CLAY__STRING_DEFAULT } ; // Reserve the hash result of zero as "null id"
}
2024-08-23 04:05:23 +00:00
2024-09-25 02:04:28 +00:00
Clay_ElementId Clay__HashString ( Clay_String key , const uint32_t offset , const uint32_t seed ) {
2024-08-23 04:05:23 +00:00
uint32_t hash = 0 ;
2024-09-25 02:04:28 +00:00
uint32_t base = seed ;
2024-08-23 04:05:23 +00:00
2024-12-30 00:09:14 +00:00
for ( size_t i = 0 ; i < key . length ; i + + ) {
2024-09-16 09:34:59 +00:00
base + = key . chars [ i ] ;
base + = ( base < < 10 ) ;
base ^ = ( base > > 6 ) ;
2024-08-23 04:05:23 +00:00
}
2024-09-16 09:34:59 +00:00
hash = base ;
2024-08-23 04:05:23 +00:00
hash + = offset ;
hash + = ( hash < < 10 ) ;
hash ^ = ( hash > > 6 ) ;
hash + = ( hash < < 3 ) ;
2024-09-16 09:34:59 +00:00
base + = ( base < < 3 ) ;
2024-08-23 04:05:23 +00:00
hash ^ = ( hash > > 11 ) ;
2024-09-16 09:34:59 +00:00
base ^ = ( base > > 11 ) ;
2024-08-23 04:05:23 +00:00
hash + = ( hash < < 15 ) ;
2024-09-16 09:34:59 +00:00
base + = ( base < < 15 ) ;
2024-09-28 23:53:46 +00:00
return CLAY__INIT ( Clay_ElementId ) { . id = hash + 1 , . offset = offset , . baseId = base + 1 , . stringId = key } ; // Reserve the hash result of zero as "null id"
2024-09-16 09:34:59 +00:00
}
Clay_ElementId Clay__Rehash ( Clay_ElementId elementId , uint32_t number ) {
uint32_t id = elementId . baseId ;
id + = number ;
id + = ( id < < 10 ) ;
id ^ = ( id > > 6 ) ;
id + = ( id < < 3 ) ;
id ^ = ( id > > 11 ) ;
id + = ( id < < 15 ) ;
2024-09-28 23:53:46 +00:00
return CLAY__INIT ( Clay_ElementId ) { . id = id , . offset = number , . baseId = elementId . baseId , . stringId = elementId . stringId } ;
2024-08-23 04:05:23 +00:00
}
uint32_t Clay__RehashWithNumber ( uint32_t id , uint32_t number ) {
id + = number ;
id + = ( id < < 10 ) ;
id ^ = ( id > > 6 ) ;
id + = ( id < < 3 ) ;
id ^ = ( id > > 11 ) ;
id + = ( id < < 15 ) ;
return id ;
}
uint32_t Clay__HashTextWithConfig ( Clay_String * text , Clay_TextElementConfig * config ) {
uint32_t hash = 0 ;
2024-12-22 06:22:10 +00:00
uintptr_t pointerAsNumber = ( uintptr_t ) text - > chars ;
2024-08-23 04:05:23 +00:00
hash + = pointerAsNumber ;
hash + = ( hash < < 10 ) ;
hash ^ = ( hash > > 6 ) ;
hash + = text - > length ;
hash + = ( hash < < 10 ) ;
hash ^ = ( hash > > 6 ) ;
2024-12-05 00:23:20 +00:00
hash + = config - > fontId ;
hash + = ( hash < < 10 ) ;
hash ^ = ( hash > > 6 ) ;
hash + = config - > fontSize ;
hash + = ( hash < < 10 ) ;
hash ^ = ( hash > > 6 ) ;
hash + = config - > lineHeight ;
hash + = ( hash < < 10 ) ;
hash ^ = ( hash > > 6 ) ;
hash + = config - > letterSpacing ;
hash + = ( hash < < 10 ) ;
hash ^ = ( hash > > 6 ) ;
hash + = config - > wrapMode ;
2024-08-23 04:05:23 +00:00
hash + = ( hash < < 10 ) ;
hash ^ = ( hash > > 6 ) ;
hash + = ( hash < < 3 ) ;
hash ^ = ( hash > > 11 ) ;
hash + = ( hash < < 15 ) ;
return hash + 1 ; // Reserve the hash result of zero as "null id"
}
2024-12-01 08:05:56 +00:00
Clay__MeasuredWord * Clay__AddMeasuredWord ( Clay__MeasuredWord word , Clay__MeasuredWord * previousWord ) {
2024-11-26 23:13:11 +00:00
if ( Clay__measuredWordsFreeList . length > 0 ) {
uint32_t newItemIndex = Clay__int32_tArray_Get ( & Clay__measuredWordsFreeList , ( int ) Clay__measuredWordsFreeList . length - 1 ) ;
Clay__measuredWordsFreeList . length - - ;
Clay__MeasuredWordArray_Set ( & Clay__measuredWords , ( int ) newItemIndex , word ) ;
2024-12-01 23:04:01 +00:00
previousWord - > next = ( int32_t ) newItemIndex ;
2024-12-01 08:05:56 +00:00
return Clay__MeasuredWordArray_Get ( & Clay__measuredWords , ( int ) newItemIndex ) ;
2024-11-26 23:13:11 +00:00
} else {
2024-12-01 23:04:01 +00:00
previousWord - > next = ( int32_t ) Clay__measuredWords . length ;
2024-12-01 08:05:56 +00:00
return Clay__MeasuredWordArray_Add ( & Clay__measuredWords , word ) ;
2024-11-26 23:13:11 +00:00
}
}
2024-10-22 07:41:35 +00:00
Clay__MeasureTextCacheItem * Clay__MeasureTextCached ( Clay_String * text , Clay_TextElementConfig * config ) {
2024-12-26 06:58:59 +00:00
# ifndef CLAY_WASM
if ( ! Clay__MeasureText ) {
Clay__errorHandler . errorHandlerFunction ( CLAY__INIT ( Clay_ErrorData ) {
. errorType = CLAY_ERROR_TYPE_TEXT_MEASUREMENT_FUNCTION_NOT_PROVIDED ,
. errorText = CLAY_STRING ( " Clay's internal MeasureText function is null. You may have forgotten to call Clay_SetMeasureTextFunction(), or passed a NULL function pointer by mistake. " ) ,
. userData = Clay__errorHandler . userData } ) ;
return NULL ;
}
# endif
2024-08-23 04:05:23 +00:00
uint32_t id = Clay__HashTextWithConfig ( text , config ) ;
2024-12-26 06:58:59 +00:00
uint32_t hashBucket = id % ( Clay__maxMeasureTextCacheWordCount / 32 ) ;
2024-08-23 04:05:23 +00:00
int32_t elementIndexPrevious = 0 ;
int32_t elementIndex = Clay__measureTextHashMap . internalArray [ hashBucket ] ;
while ( elementIndex ! = 0 ) {
Clay__MeasureTextCacheItem * hashEntry = Clay__MeasureTextCacheItemArray_Get ( & Clay__measureTextHashMapInternal , elementIndex ) ;
if ( hashEntry - > id = = id ) {
2024-10-15 23:11:01 +00:00
hashEntry - > generation = Clay__generation ;
2024-10-22 07:41:35 +00:00
return hashEntry ;
2024-08-23 04:05:23 +00:00
}
2024-10-15 23:11:01 +00:00
// This element hasn't been seen in a few frames, delete the hash map item
if ( Clay__generation - hashEntry - > generation > 2 ) {
2024-11-26 23:13:11 +00:00
// Add all the measured words that were included in this measurement to the freelist
2024-12-01 08:05:56 +00:00
int32_t nextWordIndex = hashEntry - > measuredWordsStartIndex ;
while ( nextWordIndex ! = - 1 ) {
Clay__MeasuredWord * measuredWord = Clay__MeasuredWordArray_Get ( & Clay__measuredWords , nextWordIndex ) ;
Clay__int32_tArray_Add ( & Clay__measuredWordsFreeList , nextWordIndex ) ;
nextWordIndex = measuredWord - > next ;
2024-11-26 23:13:11 +00:00
}
2024-10-15 23:11:01 +00:00
uint32_t nextIndex = hashEntry - > nextIndex ;
2024-12-01 08:05:56 +00:00
Clay__MeasureTextCacheItemArray_Set ( & Clay__measureTextHashMapInternal , elementIndex , CLAY__INIT ( Clay__MeasureTextCacheItem ) { . measuredWordsStartIndex = - 1 } ) ;
2024-10-15 23:11:01 +00:00
Clay__int32_tArray_Add ( & Clay__measureTextHashMapInternalFreeList , elementIndex ) ;
if ( elementIndexPrevious = = 0 ) {
Clay__measureTextHashMap . internalArray [ hashBucket ] = nextIndex ;
} else {
Clay__MeasureTextCacheItem * previousHashEntry = Clay__MeasureTextCacheItemArray_Get ( & Clay__measureTextHashMapInternal , elementIndexPrevious ) ;
previousHashEntry - > nextIndex = nextIndex ;
}
elementIndex = nextIndex ;
} else {
elementIndexPrevious = elementIndex ;
elementIndex = hashEntry - > nextIndex ;
}
2024-08-23 04:05:23 +00:00
}
2024-10-22 07:41:35 +00:00
2024-10-15 23:11:01 +00:00
uint32_t newItemIndex = 0 ;
2024-12-30 00:09:14 +00:00
Clay__MeasureTextCacheItem newCacheItem = { . measuredWordsStartIndex = - 1 , . id = id , . generation = Clay__generation } ;
2024-10-22 07:41:35 +00:00
Clay__MeasureTextCacheItem * measured = NULL ;
2024-10-15 23:11:01 +00:00
if ( Clay__measureTextHashMapInternalFreeList . length > 0 ) {
newItemIndex = Clay__int32_tArray_Get ( & Clay__measureTextHashMapInternalFreeList , Clay__measureTextHashMapInternalFreeList . length - 1 ) ;
Clay__measureTextHashMapInternalFreeList . length - - ;
2024-10-22 07:41:35 +00:00
Clay__MeasureTextCacheItemArray_Set ( & Clay__measureTextHashMapInternal , newItemIndex , newCacheItem ) ;
measured = Clay__MeasureTextCacheItemArray_Get ( & Clay__measureTextHashMapInternal , newItemIndex ) ;
2024-10-15 23:11:01 +00:00
} else {
2024-12-01 08:05:56 +00:00
if ( Clay__measureTextHashMapInternal . length = = Clay__measureTextHashMapInternal . capacity - 1 ) {
2024-12-26 06:58:59 +00:00
if ( Clay__booleanWarnings . maxTextMeasureCacheExceeded ) {
Clay__errorHandler . errorHandlerFunction ( CLAY__INIT ( Clay_ErrorData ) {
. errorType = CLAY_ERROR_TYPE_ELEMENTS_CAPACITY_EXCEEDED ,
. errorText = CLAY_STRING ( " Clay ran out of capacity while attempting to measure text elements. Try using Clay_SetMaxElementCount() with a higher value. " ) ,
. userData = Clay__errorHandler . userData } ) ;
Clay__booleanWarnings . maxTextMeasureCacheExceeded = true ;
}
2024-12-01 08:05:56 +00:00
return & CLAY__MEASURE_TEXT_CACHE_ITEM_DEFAULT ;
2024-10-15 23:11:01 +00:00
}
2024-10-22 07:41:35 +00:00
measured = Clay__MeasureTextCacheItemArray_Add ( & Clay__measureTextHashMapInternal , newCacheItem ) ;
2024-10-15 23:11:01 +00:00
newItemIndex = Clay__measureTextHashMapInternal . length - 1 ;
}
2024-10-22 07:41:35 +00:00
uint32_t start = 0 ;
uint32_t end = 0 ;
float measuredWidth = 0 ;
float measuredHeight = 0 ;
float spaceWidth = Clay__MeasureText ( & CLAY__SPACECHAR , config ) . width ;
2024-12-01 08:05:56 +00:00
Clay__MeasuredWord tempWord = { . next = - 1 } ;
Clay__MeasuredWord * previousWord = & tempWord ;
2024-10-22 07:41:35 +00:00
while ( end < text - > length ) {
2024-12-01 08:05:56 +00:00
if ( Clay__measuredWords . length = = Clay__measuredWords . capacity - 1 ) {
2024-12-26 06:58:59 +00:00
if ( ! Clay__booleanWarnings . maxTextMeasureCacheExceeded ) {
Clay__errorHandler . errorHandlerFunction ( CLAY__INIT ( Clay_ErrorData ) {
. errorType = CLAY_ERROR_TYPE_TEXT_MEASUREMENT_CAPACITY_EXCEEDED ,
. errorText = CLAY_STRING ( " Clay has run out of space in it's internal text measurement cache. Try using Clay_SetMaxMeasureTextCacheWordCount() (default 16384, with 1 unit storing 1 measured word). " ) ,
. userData = Clay__errorHandler . userData } ) ;
Clay__booleanWarnings . maxTextMeasureCacheExceeded = true ;
}
2024-12-01 08:05:56 +00:00
return & CLAY__MEASURE_TEXT_CACHE_ITEM_DEFAULT ;
}
2024-10-22 07:41:35 +00:00
char current = text - > chars [ end ] ;
if ( current = = ' ' | | current = = ' \n ' ) {
uint32_t length = end - start ;
2024-12-30 00:09:14 +00:00
Clay_String word = { . length = length , . chars = & text - > chars [ start ] } ;
2024-10-22 07:41:35 +00:00
Clay_Dimensions dimensions = Clay__MeasureText ( & word , config ) ;
if ( current = = ' ' ) {
dimensions . width + = spaceWidth ;
2024-12-01 08:05:56 +00:00
previousWord = Clay__AddMeasuredWord ( CLAY__INIT ( Clay__MeasuredWord ) { . startOffset = start , . length = length + 1 , . width = dimensions . width , . next = - 1 } , previousWord ) ;
2024-10-22 07:41:35 +00:00
}
if ( current = = ' \n ' ) {
2024-12-04 03:02:53 +00:00
if ( length > 1 ) {
previousWord = Clay__AddMeasuredWord ( CLAY__INIT ( Clay__MeasuredWord ) { . startOffset = start , . length = length , . width = dimensions . width , . next = - 1 } , previousWord ) ;
}
2024-12-01 08:05:56 +00:00
previousWord = Clay__AddMeasuredWord ( CLAY__INIT ( Clay__MeasuredWord ) { . startOffset = end + 1 , . length = 0 , . width = 0 , . next = - 1 } , previousWord ) ;
2024-10-22 07:41:35 +00:00
}
measuredWidth + = dimensions . width ;
measuredHeight = dimensions . height ;
start = end + 1 ;
}
end + + ;
}
if ( end - start > 0 ) {
2024-12-30 00:09:14 +00:00
Clay_String lastWord = { . length = end - start , . chars = & text - > chars [ start ] } ;
2024-10-22 07:41:35 +00:00
Clay_Dimensions dimensions = Clay__MeasureText ( & lastWord , config ) ;
2024-12-01 08:05:56 +00:00
Clay__AddMeasuredWord ( CLAY__INIT ( Clay__MeasuredWord ) { . startOffset = start , . length = end - start , . width = dimensions . width , . next = - 1 } , previousWord ) ;
2024-10-22 07:41:35 +00:00
measuredWidth + = dimensions . width ;
measuredHeight = dimensions . height ;
}
2024-12-01 08:05:56 +00:00
measured - > measuredWordsStartIndex = tempWord . next ;
2024-10-22 07:41:35 +00:00
measured - > unwrappedDimensions . width = measuredWidth ;
measured - > unwrappedDimensions . height = measuredHeight ;
2024-08-23 04:05:23 +00:00
if ( elementIndexPrevious ! = 0 ) {
2024-10-15 23:11:01 +00:00
Clay__MeasureTextCacheItemArray_Get ( & Clay__measureTextHashMapInternal , elementIndexPrevious ) - > nextIndex = newItemIndex ;
2024-08-23 04:05:23 +00:00
} else {
2024-10-15 23:11:01 +00:00
Clay__measureTextHashMap . internalArray [ hashBucket ] = newItemIndex ;
2024-08-23 04:05:23 +00:00
}
return measured ;
}
2024-08-28 04:13:28 +00:00
bool Clay__PointIsInsideRect ( Clay_Vector2 point , Clay_BoundingBox rect ) {
2024-08-23 04:05:23 +00:00
return point . x > = rect . x & & point . x < = rect . x + rect . width & & point . y > = rect . y & & point . y < = rect . y + rect . height ;
}
2024-09-16 09:34:59 +00:00
Clay_LayoutElementHashMapItem * Clay__AddHashMapItem ( Clay_ElementId elementId , Clay_LayoutElement * layoutElement ) {
2024-12-01 08:05:56 +00:00
if ( Clay__layoutElementsHashMapInternal . length = = Clay__layoutElementsHashMapInternal . capacity - 1 ) {
return NULL ;
}
2024-12-30 00:09:14 +00:00
Clay_LayoutElementHashMapItem item = { . elementId = elementId , . layoutElement = layoutElement , . nextIndex = - 1 , . generation = Clay__generation + 1 } ;
2024-09-16 09:34:59 +00:00
uint32_t hashBucket = elementId . id % Clay__layoutElementsHashMap . capacity ;
2024-08-23 04:05:23 +00:00
int32_t hashItemPrevious = - 1 ;
int32_t hashItemIndex = Clay__layoutElementsHashMap . internalArray [ hashBucket ] ;
while ( hashItemIndex ! = - 1 ) { // Just replace collision, not a big deal - leave it up to the end user
Clay_LayoutElementHashMapItem * hashItem = Clay__LayoutElementHashMapItemArray_Get ( & Clay__layoutElementsHashMapInternal , hashItemIndex ) ;
2024-09-16 09:34:59 +00:00
if ( hashItem - > elementId . id = = elementId . id ) { // Collision - resolve based on generation
2024-08-23 04:05:23 +00:00
item . nextIndex = hashItem - > nextIndex ;
2024-09-16 09:34:59 +00:00
if ( hashItem - > generation < = Clay__generation ) { // First collision - assume this is the "same" element
hashItem - > generation = Clay__generation + 1 ;
hashItem - > layoutElement = layoutElement ;
hashItem - > debugData - > collision = false ;
} else { // Multiple collisions this frame - two elements have the same ID
2024-12-26 06:58:59 +00:00
Clay__errorHandler . errorHandlerFunction ( CLAY__INIT ( Clay_ErrorData ) {
. errorType = CLAY_ERROR_TYPE_DUPLICATE_ID ,
. errorText = CLAY_STRING ( " An element with this ID was already previously declared during this layout. " ) ,
. userData = Clay__errorHandler . userData } ) ;
2024-09-16 09:34:59 +00:00
if ( Clay__debugModeEnabled ) {
hashItem - > debugData - > collision = true ;
}
}
return hashItem ;
2024-08-23 04:05:23 +00:00
}
hashItemPrevious = hashItemIndex ;
hashItemIndex = hashItem - > nextIndex ;
}
2024-09-16 09:34:59 +00:00
Clay_LayoutElementHashMapItem * hashItem = Clay__LayoutElementHashMapItemArray_Add ( & Clay__layoutElementsHashMapInternal , item ) ;
2024-12-30 00:09:14 +00:00
hashItem - > debugData = Clay__DebugElementDataArray_Add ( & Clay__debugElementData , CLAY__INIT ( Clay__DebugElementData ) { 0 } ) ;
2024-08-23 04:05:23 +00:00
if ( hashItemPrevious ! = - 1 ) {
2024-09-16 09:34:59 +00:00
Clay__LayoutElementHashMapItemArray_Get ( & Clay__layoutElementsHashMapInternal , hashItemPrevious ) - > nextIndex = ( int32_t ) Clay__layoutElementsHashMapInternal . length - 1 ;
} else {
Clay__layoutElementsHashMap . internalArray [ hashBucket ] = ( int32_t ) Clay__layoutElementsHashMapInternal . length - 1 ;
2024-08-23 04:05:23 +00:00
}
return hashItem ;
}
Clay_LayoutElementHashMapItem * Clay__GetHashMapItem ( uint32_t id ) {
uint32_t hashBucket = id % Clay__layoutElementsHashMap . capacity ;
int32_t elementIndex = Clay__layoutElementsHashMap . internalArray [ hashBucket ] ;
while ( elementIndex ! = - 1 ) {
2024-09-16 09:34:59 +00:00
Clay_LayoutElementHashMapItem * hashEntry = Clay__LayoutElementHashMapItemArray_Get ( & Clay__layoutElementsHashMapInternal , elementIndex ) ;
if ( hashEntry - > elementId . id = = id ) {
2024-08-23 04:05:23 +00:00
return hashEntry ;
}
elementIndex = hashEntry - > nextIndex ;
}
2024-12-01 08:05:56 +00:00
return & CLAY__LAYOUT_ELEMENT_HASH_MAP_ITEM_DEFAULT ;
2024-08-23 04:05:23 +00:00
}
2024-10-22 07:41:35 +00:00
void Clay__GenerateIdForAnonymousElement ( Clay_LayoutElement * openLayoutElement ) {
Clay_LayoutElement * parentElement = Clay_LayoutElementArray_Get ( & Clay__layoutElements , Clay__int32_tArray_Get ( & Clay__openLayoutElementStack , Clay__openLayoutElementStack . length - 2 ) ) ;
2024-12-30 00:09:14 +00:00
Clay_ElementId elementId = Clay__HashNumber ( parentElement - > childrenOrTextContent . children . length , parentElement - > id ) ;
2024-10-22 07:41:35 +00:00
openLayoutElement - > id = elementId . id ;
Clay__AddHashMapItem ( elementId , openLayoutElement ) ;
2024-09-16 09:34:59 +00:00
Clay__StringArray_Add ( & Clay__layoutElementIdStrings , elementId . stringId ) ;
2024-08-23 04:05:23 +00:00
}
2024-12-26 22:12:59 +00:00
void Clay__ElementPostConfiguration ( void ) {
2024-12-26 06:58:59 +00:00
if ( Clay__booleanWarnings . maxElementsExceeded ) {
2024-12-01 08:05:56 +00:00
return ;
}
2024-10-22 07:41:35 +00:00
Clay_LayoutElement * openLayoutElement = Clay__GetOpenLayoutElement ( ) ;
// ID
if ( openLayoutElement - > id = = 0 ) {
Clay__GenerateIdForAnonymousElement ( openLayoutElement ) ;
2024-08-23 04:05:23 +00:00
}
2024-10-22 07:41:35 +00:00
// Layout Config
if ( ! openLayoutElement - > layoutConfig ) {
openLayoutElement - > layoutConfig = & CLAY_LAYOUT_DEFAULT ;
2024-08-23 04:05:23 +00:00
}
2024-10-22 07:41:35 +00:00
// Loop through element configs and handle special cases
openLayoutElement - > elementConfigs . internalArray = & Clay__elementConfigs . internalArray [ Clay__elementConfigs . length ] ;
2024-12-30 00:09:14 +00:00
for ( uint32_t elementConfigIndex = 0 ; elementConfigIndex < openLayoutElement - > elementConfigs . length ; elementConfigIndex + + ) {
2024-10-22 07:41:35 +00:00
Clay_ElementConfig * config = Clay__ElementConfigArray_Add ( & Clay__elementConfigs , * Clay__ElementConfigArray_Get ( & Clay__elementConfigBuffer , Clay__elementConfigBuffer . length - openLayoutElement - > elementConfigs . length + elementConfigIndex ) ) ;
openLayoutElement - > configsEnabled | = config - > type ;
switch ( config - > type ) {
2024-11-26 23:13:11 +00:00
case CLAY__ELEMENT_CONFIG_TYPE_RECTANGLE :
2024-10-22 07:41:35 +00:00
case CLAY__ELEMENT_CONFIG_TYPE_BORDER_CONTAINER : break ;
case CLAY__ELEMENT_CONFIG_TYPE_FLOATING_CONTAINER : {
Clay_FloatingElementConfig * floatingConfig = config - > config . floatingElementConfig ;
// This looks dodgy but because of the auto generated root element the depth of the tree will always be at least 2 here
Clay_LayoutElement * hierarchicalParent = Clay_LayoutElementArray_Get ( & Clay__layoutElements , Clay__int32_tArray_Get ( & Clay__openLayoutElementStack , Clay__openLayoutElementStack . length - 2 ) ) ;
if ( ! hierarchicalParent ) {
break ;
}
int clipElementId = 0 ;
if ( floatingConfig - > parentId = = 0 ) {
// If no parent id was specified, attach to the elements direct hierarchical parent
Clay_FloatingElementConfig newConfig = * floatingConfig ;
newConfig . parentId = hierarchicalParent - > id ;
floatingConfig = Clay__FloatingElementConfigArray_Add ( & Clay__floatingElementConfigs , newConfig ) ;
config - > config . floatingElementConfig = floatingConfig ;
if ( Clay__openClipElementStack . length > 0 ) {
clipElementId = Clay__int32_tArray_Get ( & Clay__openClipElementStack , ( int ) Clay__openClipElementStack . length - 1 ) ;
}
} else {
Clay_LayoutElementHashMapItem * parentItem = Clay__GetHashMapItem ( floatingConfig - > parentId ) ;
2024-12-18 22:35:18 +00:00
clipElementId = Clay__int32_tArray_Get ( & Clay__layoutElementClipElementIds , parentItem - > layoutElement - Clay__layoutElements . internalArray ) ;
2024-10-22 07:41:35 +00:00
if ( ! parentItem ) {
2024-12-26 06:58:59 +00:00
Clay__errorHandler . errorHandlerFunction ( CLAY__INIT ( Clay_ErrorData ) {
. errorType = CLAY_ERROR_TYPE_FLOATING_CONTAINER_PARENT_NOT_FOUND ,
. errorText = CLAY_STRING ( " A floating element was declared with a parentId, but no element with that ID was found. " ) ,
. userData = Clay__errorHandler . userData } ) ;
2024-10-22 07:41:35 +00:00
}
}
Clay__LayoutElementTreeRootArray_Add ( & Clay__layoutElementTreeRoots , CLAY__INIT ( Clay__LayoutElementTreeRoot ) {
. layoutElementIndex = ( uint32_t ) Clay__int32_tArray_Get ( & Clay__openLayoutElementStack , Clay__openLayoutElementStack . length - 1 ) ,
. parentId = floatingConfig - > parentId ,
. clipElementId = ( uint32_t ) clipElementId ,
. zIndex = floatingConfig - > zIndex ,
} ) ;
break ;
}
case CLAY__ELEMENT_CONFIG_TYPE_SCROLL_CONTAINER : {
Clay__int32_tArray_Add ( & Clay__openClipElementStack , ( int ) openLayoutElement - > id ) ;
// Retrieve or create cached data to track scroll position across frames
Clay__ScrollContainerDataInternal * scrollOffset = CLAY__NULL ;
2024-12-30 00:09:14 +00:00
for ( uint32_t i = 0 ; i < Clay__scrollContainerDatas . length ; i + + ) {
2024-10-22 07:41:35 +00:00
Clay__ScrollContainerDataInternal * mapping = Clay__ScrollContainerDataInternalArray_Get ( & Clay__scrollContainerDatas , i ) ;
if ( openLayoutElement - > id = = mapping - > elementId ) {
scrollOffset = mapping ;
scrollOffset - > layoutElement = openLayoutElement ;
scrollOffset - > openThisFrame = true ;
}
}
if ( ! scrollOffset ) {
2024-12-18 22:35:18 +00:00
scrollOffset = Clay__ScrollContainerDataInternalArray_Add ( & Clay__scrollContainerDatas , CLAY__INIT ( Clay__ScrollContainerDataInternal ) { . layoutElement = openLayoutElement , . scrollOrigin = { - 1 , - 1 } , . elementId = openLayoutElement - > id , . openThisFrame = true } ) ;
}
if ( Clay__externalScrollHandlingEnabled ) {
scrollOffset - > scrollPosition = Clay__QueryScrollOffset ( scrollOffset - > elementId ) ;
2024-10-22 07:41:35 +00:00
}
break ;
}
case CLAY__ELEMENT_CONFIG_TYPE_CUSTOM : break ;
case CLAY__ELEMENT_CONFIG_TYPE_IMAGE : {
Clay__LayoutElementPointerArray_Add ( & Clay__imageElementPointers , openLayoutElement ) ;
break ;
}
2024-12-30 00:09:14 +00:00
case CLAY__ELEMENT_CONFIG_TYPE_TEXT :
default : break ;
2024-08-23 04:05:23 +00:00
}
}
2024-10-22 07:41:35 +00:00
Clay__elementConfigBuffer . length - = openLayoutElement - > elementConfigs . length ;
2024-08-23 04:05:23 +00:00
}
2024-12-26 22:12:59 +00:00
void Clay__CloseElement ( void ) {
2024-12-26 06:58:59 +00:00
if ( Clay__booleanWarnings . maxElementsExceeded ) {
2024-12-01 08:05:56 +00:00
return ;
}
2024-10-22 07:41:35 +00:00
Clay_LayoutElement * openLayoutElement = Clay__GetOpenLayoutElement ( ) ;
Clay_LayoutConfig * layoutConfig = openLayoutElement - > layoutConfig ;
bool elementHasScrollHorizontal = false ;
bool elementHasScrollVertical = false ;
if ( Clay__ElementHasConfig ( openLayoutElement , CLAY__ELEMENT_CONFIG_TYPE_SCROLL_CONTAINER ) ) {
Clay_ScrollElementConfig * scrollConfig = Clay__FindElementConfigWithType ( openLayoutElement , CLAY__ELEMENT_CONFIG_TYPE_SCROLL_CONTAINER ) . scrollElementConfig ;
elementHasScrollHorizontal = scrollConfig - > horizontal ;
elementHasScrollVertical = scrollConfig - > vertical ;
Clay__openClipElementStack . length - - ;
}
2024-08-23 04:05:23 +00:00
2024-10-22 07:41:35 +00:00
// Attach children to the current open element
2024-12-30 00:09:14 +00:00
openLayoutElement - > childrenOrTextContent . children . elements = & Clay__layoutElementChildren . internalArray [ Clay__layoutElementChildren . length ] ;
2024-08-23 04:05:23 +00:00
if ( layoutConfig - > layoutDirection = = CLAY_LEFT_TO_RIGHT ) {
2024-10-22 07:41:35 +00:00
openLayoutElement - > dimensions . width = layoutConfig - > padding . x * 2 ;
2024-12-30 00:09:14 +00:00
for ( int i = 0 ; i < openLayoutElement - > childrenOrTextContent . children . length ; i + + ) {
int32_t childIndex = Clay__int32_tArray_Get ( & Clay__layoutElementChildrenBuffer , ( int ) Clay__layoutElementChildrenBuffer . length - openLayoutElement - > childrenOrTextContent . children . length + i ) ;
2024-09-16 09:34:59 +00:00
Clay_LayoutElement * child = Clay_LayoutElementArray_Get ( & Clay__layoutElements , childIndex ) ;
2024-10-22 07:41:35 +00:00
openLayoutElement - > dimensions . width + = child - > dimensions . width ;
openLayoutElement - > dimensions . height = CLAY__MAX ( openLayoutElement - > dimensions . height , child - > dimensions . height + layoutConfig - > padding . y * 2 ) ;
2024-08-23 04:05:23 +00:00
// Minimum size of child elements doesn't matter to scroll containers as they can shrink and hide their contents
2024-10-22 07:41:35 +00:00
if ( ! elementHasScrollHorizontal ) {
openLayoutElement - > minDimensions . width + = child - > minDimensions . width ;
2024-08-23 04:05:23 +00:00
}
2024-10-22 07:41:35 +00:00
if ( ! elementHasScrollVertical ) {
openLayoutElement - > minDimensions . height = CLAY__MAX ( openLayoutElement - > minDimensions . height , child - > minDimensions . height + layoutConfig - > padding . y * 2 ) ;
2024-08-23 04:05:23 +00:00
}
2024-09-16 09:34:59 +00:00
Clay__int32_tArray_Add ( & Clay__layoutElementChildren , childIndex ) ;
2024-08-23 04:05:23 +00:00
}
2024-12-30 00:09:14 +00:00
float childGap = ( float ) ( CLAY__MAX ( openLayoutElement - > childrenOrTextContent . children . length - 1 , 0 ) * layoutConfig - > childGap ) ;
2024-10-22 07:41:35 +00:00
openLayoutElement - > dimensions . width + = childGap ; // TODO this is technically a bug with childgap and scroll containers
openLayoutElement - > minDimensions . width + = childGap ;
2024-08-23 04:05:23 +00:00
}
else if ( layoutConfig - > layoutDirection = = CLAY_TOP_TO_BOTTOM ) {
2024-10-22 07:41:35 +00:00
openLayoutElement - > dimensions . height = layoutConfig - > padding . y * 2 ;
2024-12-30 00:09:14 +00:00
for ( int i = 0 ; i < openLayoutElement - > childrenOrTextContent . children . length ; i + + ) {
int32_t childIndex = Clay__int32_tArray_Get ( & Clay__layoutElementChildrenBuffer , ( int ) Clay__layoutElementChildrenBuffer . length - openLayoutElement - > childrenOrTextContent . children . length + i ) ;
2024-09-16 09:34:59 +00:00
Clay_LayoutElement * child = Clay_LayoutElementArray_Get ( & Clay__layoutElements , childIndex ) ;
2024-10-22 07:41:35 +00:00
openLayoutElement - > dimensions . height + = child - > dimensions . height ;
openLayoutElement - > dimensions . width = CLAY__MAX ( openLayoutElement - > dimensions . width , child - > dimensions . width + layoutConfig - > padding . x * 2 ) ;
2024-08-23 04:05:23 +00:00
// Minimum size of child elements doesn't matter to scroll containers as they can shrink and hide their contents
2024-10-22 07:41:35 +00:00
if ( ! elementHasScrollVertical ) {
openLayoutElement - > minDimensions . height + = child - > minDimensions . height ;
2024-08-23 04:05:23 +00:00
}
2024-10-22 07:41:35 +00:00
if ( ! elementHasScrollHorizontal ) {
openLayoutElement - > minDimensions . width = CLAY__MAX ( openLayoutElement - > minDimensions . width , child - > minDimensions . width + layoutConfig - > padding . x * 2 ) ;
2024-08-23 04:05:23 +00:00
}
2024-09-16 09:34:59 +00:00
Clay__int32_tArray_Add ( & Clay__layoutElementChildren , childIndex ) ;
2024-08-23 04:05:23 +00:00
}
2024-12-30 00:09:14 +00:00
float childGap = ( float ) ( CLAY__MAX ( openLayoutElement - > childrenOrTextContent . children . length - 1 , 0 ) * layoutConfig - > childGap ) ;
2024-10-22 07:41:35 +00:00
openLayoutElement - > dimensions . height + = childGap ; // TODO this is technically a bug with childgap and scroll containers
openLayoutElement - > minDimensions . height + = childGap ;
2024-08-23 04:05:23 +00:00
}
2024-12-30 00:09:14 +00:00
Clay__layoutElementChildrenBuffer . length - = openLayoutElement - > childrenOrTextContent . children . length ;
2024-08-23 04:05:23 +00:00
2024-10-22 07:41:35 +00:00
// Clamp element min and max width to the values configured in the layout
2024-08-23 04:05:23 +00:00
if ( layoutConfig - > sizing . width . type ! = CLAY__SIZING_TYPE_PERCENT ) {
2024-12-26 22:12:59 +00:00
if ( layoutConfig - > sizing . width . size . minMax . max < = 0 ) { // Set the max size if the user didn't specify, makes calculations easier
layoutConfig - > sizing . width . size . minMax . max = CLAY__MAXFLOAT ;
2024-10-07 05:56:13 +00:00
}
2024-12-26 22:12:59 +00:00
openLayoutElement - > dimensions . width = CLAY__MIN ( CLAY__MAX ( openLayoutElement - > dimensions . width , layoutConfig - > sizing . width . size . minMax . min ) , layoutConfig - > sizing . width . size . minMax . max ) ;
openLayoutElement - > minDimensions . width = CLAY__MIN ( CLAY__MAX ( openLayoutElement - > minDimensions . width , layoutConfig - > sizing . width . size . minMax . min ) , layoutConfig - > sizing . width . size . minMax . max ) ;
2024-10-22 07:41:35 +00:00
} else {
openLayoutElement - > dimensions . width = 0 ;
2024-08-23 04:05:23 +00:00
}
2024-10-22 07:41:35 +00:00
// Clamp element min and max height to the values configured in the layout
2024-08-23 04:05:23 +00:00
if ( layoutConfig - > sizing . height . type ! = CLAY__SIZING_TYPE_PERCENT ) {
2024-12-26 22:12:59 +00:00
if ( layoutConfig - > sizing . height . size . minMax . max < = 0 ) { // Set the max size if the user didn't specify, makes calculations easier
layoutConfig - > sizing . height . size . minMax . max = CLAY__MAXFLOAT ;
2024-10-07 05:56:13 +00:00
}
2024-12-26 22:12:59 +00:00
openLayoutElement - > dimensions . height = CLAY__MIN ( CLAY__MAX ( openLayoutElement - > dimensions . height , layoutConfig - > sizing . height . size . minMax . min ) , layoutConfig - > sizing . height . size . minMax . max ) ;
openLayoutElement - > minDimensions . height = CLAY__MIN ( CLAY__MAX ( openLayoutElement - > minDimensions . height , layoutConfig - > sizing . height . size . minMax . min ) , layoutConfig - > sizing . height . size . minMax . max ) ;
2024-10-22 07:41:35 +00:00
} else {
openLayoutElement - > dimensions . height = 0 ;
2024-08-23 04:05:23 +00:00
}
2024-10-22 07:41:35 +00:00
bool elementIsFloating = Clay__ElementHasConfig ( openLayoutElement , CLAY__ELEMENT_CONFIG_TYPE_FLOATING_CONTAINER ) ;
2024-08-23 04:05:23 +00:00
2024-10-22 07:41:35 +00:00
// Close the currently open element
int32_t closingElementIndex = Clay__int32_tArray_RemoveSwapback ( & Clay__openLayoutElementStack , ( int ) Clay__openLayoutElementStack . length - 1 ) ;
openLayoutElement = Clay__GetOpenLayoutElement ( ) ;
2024-08-23 04:05:23 +00:00
2024-10-22 07:41:35 +00:00
if ( ! elementIsFloating & & Clay__openLayoutElementStack . length > 1 ) {
2024-12-30 00:09:14 +00:00
openLayoutElement - > childrenOrTextContent . children . length + + ;
2024-10-22 07:41:35 +00:00
Clay__int32_tArray_Add ( & Clay__layoutElementChildrenBuffer , closingElementIndex ) ;
}
2024-08-23 04:05:23 +00:00
}
2024-12-26 22:12:59 +00:00
void Clay__OpenElement ( void ) {
2024-12-26 06:58:59 +00:00
if ( Clay__layoutElements . length = = Clay__layoutElements . capacity - 1 | | Clay__booleanWarnings . maxElementsExceeded ) {
Clay__booleanWarnings . maxElementsExceeded = true ;
2024-12-01 08:05:56 +00:00
return ;
}
2024-12-30 00:09:14 +00:00
Clay_LayoutElement layoutElement = { 0 } ;
2024-10-22 07:41:35 +00:00
Clay_LayoutElementArray_Add ( & Clay__layoutElements , layoutElement ) ;
Clay__int32_tArray_Add ( & Clay__openLayoutElementStack , Clay__layoutElements . length - 1 ) ;
2024-12-18 22:35:18 +00:00
if ( Clay__openClipElementStack . length > 0 ) {
Clay__int32_tArray_Set ( & Clay__layoutElementClipElementIds , Clay__layoutElements . length - 1 , Clay__int32_tArray_Get ( & Clay__openClipElementStack , ( int ) Clay__openClipElementStack . length - 1 ) ) ;
} else {
Clay__int32_tArray_Set ( & Clay__layoutElementClipElementIds , Clay__layoutElements . length - 1 , 0 ) ;
}
2024-08-23 04:05:23 +00:00
}
2024-10-22 07:41:35 +00:00
void Clay__OpenTextElement ( Clay_String text , Clay_TextElementConfig * textConfig ) {
2024-12-26 06:58:59 +00:00
if ( Clay__layoutElements . length = = Clay__layoutElements . capacity - 1 | | Clay__booleanWarnings . maxElementsExceeded ) {
Clay__booleanWarnings . maxElementsExceeded = true ;
2024-12-01 08:05:56 +00:00
return ;
}
2024-10-22 07:41:35 +00:00
Clay_LayoutElement * parentElement = Clay__GetOpenLayoutElement ( ) ;
2024-12-30 00:09:14 +00:00
parentElement - > childrenOrTextContent . children . length + + ;
2024-10-22 07:41:35 +00:00
Clay__OpenElement ( ) ;
Clay_LayoutElement * openLayoutElement = Clay__GetOpenLayoutElement ( ) ;
Clay__int32_tArray_Add ( & Clay__layoutElementChildrenBuffer , Clay__layoutElements . length - 1 ) ;
Clay__MeasureTextCacheItem * textMeasured = Clay__MeasureTextCached ( & text , textConfig ) ;
2024-12-30 00:09:14 +00:00
Clay_ElementId elementId = Clay__HashString ( CLAY_STRING ( " Text " ) , parentElement - > childrenOrTextContent . children . length , parentElement - > id ) ;
2024-10-22 07:41:35 +00:00
openLayoutElement - > id = elementId . id ;
Clay__AddHashMapItem ( elementId , openLayoutElement ) ;
Clay__StringArray_Add ( & Clay__layoutElementIdStrings , elementId . stringId ) ;
2024-12-30 00:09:14 +00:00
Clay_Dimensions textDimensions = { . width = textMeasured - > unwrappedDimensions . width , . height = textConfig - > lineHeight > 0 ? textConfig - > lineHeight : textMeasured - > unwrappedDimensions . height } ;
2024-10-22 07:41:35 +00:00
openLayoutElement - > dimensions = textDimensions ;
openLayoutElement - > minDimensions = CLAY__INIT ( Clay_Dimensions ) { . width = textMeasured - > unwrappedDimensions . height , . height = textDimensions . height } ; // TODO not sure this is the best way to decide min width for text
2024-12-30 00:09:14 +00:00
openLayoutElement - > childrenOrTextContent . textElementData = Clay__TextElementDataArray_Add ( & Clay__textElementData , CLAY__INIT ( Clay__TextElementData ) { . text = text , . preferredDimensions = textMeasured - > unwrappedDimensions , . elementIndex = Clay__layoutElements . length - 1 } ) ;
2024-10-22 07:41:35 +00:00
openLayoutElement - > elementConfigs = CLAY__INIT ( Clay__ElementConfigArraySlice ) {
. length = 1 ,
. internalArray = Clay__ElementConfigArray_Add ( & Clay__elementConfigs , CLAY__INIT ( Clay_ElementConfig ) { . type = CLAY__ELEMENT_CONFIG_TYPE_TEXT , . config = { . textElementConfig = textConfig } } )
} ;
openLayoutElement - > configsEnabled | = CLAY__ELEMENT_CONFIG_TYPE_TEXT ;
openLayoutElement - > layoutConfig = & CLAY_LAYOUT_DEFAULT ;
// Close the currently open element
Clay__int32_tArray_RemoveSwapback ( & Clay__openLayoutElementStack , ( int ) Clay__openLayoutElementStack . length - 1 ) ;
2024-08-23 04:05:23 +00:00
}
void Clay__InitializeEphemeralMemory ( Clay_Arena * arena ) {
// Ephemeral Memory - reset every frame
Clay__internalArena . nextAllocation = Clay__arenaResetOffset ;
2024-12-26 06:58:59 +00:00
Clay__layoutElementChildrenBuffer = Clay__int32_tArray_Allocate_Arena ( Clay__maxElementCount , arena ) ;
Clay__layoutElements = Clay_LayoutElementArray_Allocate_Arena ( Clay__maxElementCount , arena ) ;
2024-09-16 09:34:59 +00:00
Clay_warnings = Clay__WarningArray_Allocate_Arena ( 100 , arena ) ;
2024-08-23 04:05:23 +00:00
2024-12-26 06:58:59 +00:00
Clay__layoutConfigs = Clay__LayoutConfigArray_Allocate_Arena ( Clay__maxElementCount , arena ) ;
Clay__elementConfigBuffer = Clay__ElementConfigArray_Allocate_Arena ( Clay__maxElementCount , arena ) ;
Clay__elementConfigs = Clay__ElementConfigArray_Allocate_Arena ( Clay__maxElementCount , arena ) ;
Clay__rectangleElementConfigs = Clay__RectangleElementConfigArray_Allocate_Arena ( Clay__maxElementCount , arena ) ;
Clay__textElementConfigs = Clay__TextElementConfigArray_Allocate_Arena ( Clay__maxElementCount , arena ) ;
Clay__imageElementConfigs = Clay__ImageElementConfigArray_Allocate_Arena ( Clay__maxElementCount , arena ) ;
Clay__floatingElementConfigs = Clay__FloatingElementConfigArray_Allocate_Arena ( Clay__maxElementCount , arena ) ;
Clay__scrollElementConfigs = Clay__ScrollElementConfigArray_Allocate_Arena ( Clay__maxElementCount , arena ) ;
Clay__customElementConfigs = Clay__CustomElementConfigArray_Allocate_Arena ( Clay__maxElementCount , arena ) ;
Clay__borderElementConfigs = Clay__BorderElementConfigArray_Allocate_Arena ( Clay__maxElementCount , arena ) ;
Clay__layoutElementIdStrings = Clay__StringArray_Allocate_Arena ( Clay__maxElementCount , arena ) ;
2024-12-28 06:15:22 +00:00
Clay__wrappedTextLines = Clay__WrappedTextLineArray_Allocate_Arena ( Clay__maxElementCount , arena ) ;
2024-12-26 06:58:59 +00:00
Clay__layoutElementTreeNodeArray1 = Clay__LayoutElementTreeNodeArray_Allocate_Arena ( Clay__maxElementCount , arena ) ;
Clay__layoutElementTreeRoots = Clay__LayoutElementTreeRootArray_Allocate_Arena ( Clay__maxElementCount , arena ) ;
Clay__layoutElementChildren = Clay__int32_tArray_Allocate_Arena ( Clay__maxElementCount , arena ) ;
Clay__openLayoutElementStack = Clay__int32_tArray_Allocate_Arena ( Clay__maxElementCount , arena ) ;
Clay__textElementData = Clay__TextElementDataArray_Allocate_Arena ( Clay__maxElementCount , arena ) ;
Clay__imageElementPointers = Clay__LayoutElementPointerArray_Allocate_Arena ( Clay__maxElementCount , arena ) ;
Clay__renderCommands = Clay_RenderCommandArray_Allocate_Arena ( Clay__maxElementCount , arena ) ;
Clay__treeNodeVisited = Clay__BoolArray_Allocate_Arena ( Clay__maxElementCount , arena ) ;
2024-08-23 04:05:23 +00:00
Clay__treeNodeVisited . length = Clay__treeNodeVisited . capacity ; // This array is accessed directly rather than behaving as a list
2024-12-26 06:58:59 +00:00
Clay__openClipElementStack = Clay__int32_tArray_Allocate_Arena ( Clay__maxElementCount , arena ) ;
Clay__reusableElementIndexBuffer = Clay__int32_tArray_Allocate_Arena ( Clay__maxElementCount , arena ) ;
Clay__layoutElementClipElementIds = Clay__int32_tArray_Allocate_Arena ( Clay__maxElementCount , arena ) ;
Clay__dynamicStringData = Clay__CharArray_Allocate_Arena ( Clay__maxElementCount , arena ) ;
2024-08-23 04:05:23 +00:00
}
void Clay__InitializePersistentMemory ( Clay_Arena * arena ) {
2024-10-22 07:41:35 +00:00
// Persistent memory - initialized once and not reset
2024-08-26 07:05:43 +00:00
Clay__scrollContainerDatas = Clay__ScrollContainerDataInternalArray_Allocate_Arena ( 10 , arena ) ;
2024-12-26 06:58:59 +00:00
Clay__layoutElementsHashMapInternal = Clay__LayoutElementHashMapItemArray_Allocate_Arena ( Clay__maxElementCount , arena ) ;
Clay__layoutElementsHashMap = Clay__int32_tArray_Allocate_Arena ( Clay__maxElementCount , arena ) ;
Clay__measureTextHashMapInternal = Clay__MeasureTextCacheItemArray_Allocate_Arena ( Clay__maxElementCount , arena ) ;
Clay__measureTextHashMapInternalFreeList = Clay__int32_tArray_Allocate_Arena ( Clay__maxElementCount , arena ) ;
Clay__measuredWordsFreeList = Clay__int32_tArray_Allocate_Arena ( Clay__maxMeasureTextCacheWordCount , arena ) ;
Clay__measureTextHashMap = Clay__int32_tArray_Allocate_Arena ( Clay__maxElementCount , arena ) ;
Clay__measuredWords = Clay__MeasuredWordArray_Allocate_Arena ( Clay__maxMeasureTextCacheWordCount , arena ) ;
Clay__pointerOverIds = Clay__ElementIdArray_Allocate_Arena ( Clay__maxElementCount , arena ) ;
Clay__debugElementData = Clay__DebugElementDataArray_Allocate_Arena ( Clay__maxElementCount , arena ) ;
2024-08-23 04:05:23 +00:00
Clay__arenaResetOffset = arena - > nextAllocation ;
}
2024-12-30 00:09:14 +00:00
CLAY__TYPEDEF ( Clay__SizeDistributionType , enum {
2024-08-23 04:05:23 +00:00
CLAY__SIZE_DISTRIBUTION_TYPE_SCROLL_CONTAINER ,
CLAY__SIZE_DISTRIBUTION_TYPE_RESIZEABLE_CONTAINER ,
CLAY__SIZE_DISTRIBUTION_TYPE_GROW_CONTAINER ,
2024-12-30 00:09:14 +00:00
} ) ;
2024-08-23 04:05:23 +00:00
2024-10-22 07:41:35 +00:00
float Clay__DistributeSizeAmongChildren ( bool xAxis , float sizeToDistribute , Clay__int32_tArray resizableContainerBuffer , Clay__SizeDistributionType distributionType ) {
Clay__int32_tArray remainingElements = Clay__openClipElementStack ;
2024-10-05 05:55:22 +00:00
remainingElements . length = 0 ;
2024-12-30 00:09:14 +00:00
for ( uint32_t i = 0 ; i < resizableContainerBuffer . length ; + + i ) {
2024-10-22 07:41:35 +00:00
Clay__int32_tArray_Add ( & remainingElements , Clay__int32_tArray_Get ( & resizableContainerBuffer , i ) ) ;
2024-10-05 05:55:22 +00:00
}
2024-08-23 04:05:23 +00:00
while ( sizeToDistribute ! = 0 & & remainingElements . length > 0 ) {
2024-10-05 05:55:22 +00:00
float dividedSize = sizeToDistribute / ( float ) remainingElements . length ;
2024-12-30 00:09:14 +00:00
for ( uint32_t childOffset = 0 ; childOffset < remainingElements . length ; childOffset + + ) {
2024-10-22 07:41:35 +00:00
Clay_LayoutElement * childElement = Clay_LayoutElementArray_Get ( & Clay__layoutElements , Clay__int32_tArray_Get ( & remainingElements , childOffset ) ) ;
2024-08-23 04:05:23 +00:00
Clay_SizingAxis childSizing = xAxis ? childElement - > layoutConfig - > sizing . width : childElement - > layoutConfig - > sizing . height ;
float * childSize = xAxis ? & childElement - > dimensions . width : & childElement - > dimensions . height ;
float childMinSize = xAxis ? childElement - > minDimensions . width : childElement - > minDimensions . height ;
2024-10-05 05:55:22 +00:00
bool canDistribute = true ;
2024-08-23 04:05:23 +00:00
2024-12-26 22:12:59 +00:00
if ( ( sizeToDistribute < 0 & & * childSize = = childSizing . size . minMax . min ) | | ( sizeToDistribute > 0 & & * childSize = = childSizing . size . minMax . max ) ) {
2024-10-05 05:55:22 +00:00
canDistribute = false ;
2024-08-23 04:05:23 +00:00
}
2024-10-05 05:55:22 +00:00
// Currently, we don't support squishing aspect ratio images on their Y axis as it would break ratio
2024-10-22 07:41:35 +00:00
else if ( ! xAxis & & Clay__ElementHasConfig ( childElement , CLAY__ELEMENT_CONFIG_TYPE_IMAGE ) ) {
2024-10-05 05:55:22 +00:00
canDistribute = false ;
}
else {
switch ( distributionType ) {
case CLAY__SIZE_DISTRIBUTION_TYPE_RESIZEABLE_CONTAINER : break ;
case CLAY__SIZE_DISTRIBUTION_TYPE_GROW_CONTAINER : if ( childSizing . type ! = CLAY__SIZING_TYPE_GROW ) canDistribute = false ; break ;
2024-10-22 07:41:35 +00:00
case CLAY__SIZE_DISTRIBUTION_TYPE_SCROLL_CONTAINER : {
if ( Clay__ElementHasConfig ( childElement , CLAY__ELEMENT_CONFIG_TYPE_SCROLL_CONTAINER ) ) {
Clay_ScrollElementConfig * scrollConfig = Clay__FindElementConfigWithType ( childElement , CLAY__ELEMENT_CONFIG_TYPE_SCROLL_CONTAINER ) . scrollElementConfig ;
if ( ( xAxis & & ! scrollConfig - > horizontal ) | | ( ! xAxis & & ! scrollConfig - > vertical ) ) {
Clay__int32_tArray_RemoveSwapback ( & remainingElements , childOffset ) ;
childOffset - - ;
continue ;
}
}
}
2024-10-05 05:55:22 +00:00
}
2024-08-23 04:05:23 +00:00
}
2024-10-05 05:55:22 +00:00
if ( ! canDistribute ) {
2024-10-22 07:41:35 +00:00
Clay__int32_tArray_RemoveSwapback ( & remainingElements , childOffset ) ;
2024-10-05 05:55:22 +00:00
childOffset - - ;
continue ;
2024-08-23 04:05:23 +00:00
}
float oldChildSize = * childSize ;
2024-12-26 22:12:59 +00:00
* childSize = CLAY__MAX ( CLAY__MAX ( CLAY__MIN ( childSizing . size . minMax . max , * childSize + dividedSize ) , childSizing . size . minMax . min ) , childMinSize ) ;
2024-08-23 04:05:23 +00:00
float diff = * childSize - oldChildSize ;
2024-10-05 05:55:22 +00:00
if ( diff > - 0.01 & & diff < 0.01 ) {
2024-10-22 07:41:35 +00:00
Clay__int32_tArray_RemoveSwapback ( & remainingElements , childOffset ) ;
2024-10-05 05:55:22 +00:00
childOffset - - ;
continue ;
2024-08-23 04:05:23 +00:00
}
sizeToDistribute - = diff ;
}
}
2024-10-05 05:55:22 +00:00
return ( sizeToDistribute > - 0.01 & & sizeToDistribute < 0.01 ) ? 0 : sizeToDistribute ;
2024-08-23 04:05:23 +00:00
}
void Clay__SizeContainersAlongAxis ( bool xAxis ) {
2024-09-16 09:34:59 +00:00
Clay__int32_tArray bfsBuffer = Clay__layoutElementChildrenBuffer ;
2024-10-22 07:41:35 +00:00
Clay__int32_tArray resizableContainerBuffer = Clay__openLayoutElementStack ;
2024-12-30 00:09:14 +00:00
for ( uint32_t rootIndex = 0 ; rootIndex < Clay__layoutElementTreeRoots . length ; + + rootIndex ) {
2024-08-23 04:05:23 +00:00
bfsBuffer . length = 0 ;
Clay__LayoutElementTreeRoot * root = Clay__LayoutElementTreeRootArray_Get ( & Clay__layoutElementTreeRoots , rootIndex ) ;
2024-09-16 09:34:59 +00:00
Clay_LayoutElement * rootElement = Clay_LayoutElementArray_Get ( & Clay__layoutElements , ( int ) root - > layoutElementIndex ) ;
Clay__int32_tArray_Add ( & bfsBuffer , ( int32_t ) root - > layoutElementIndex ) ;
2024-08-26 07:05:43 +00:00
// Size floating containers to their parents
2024-10-22 07:41:35 +00:00
if ( Clay__ElementHasConfig ( rootElement , CLAY__ELEMENT_CONFIG_TYPE_FLOATING_CONTAINER ) ) {
Clay_FloatingElementConfig * floatingElementConfig = Clay__FindElementConfigWithType ( rootElement , CLAY__ELEMENT_CONFIG_TYPE_FLOATING_CONTAINER ) . floatingElementConfig ;
Clay_LayoutElementHashMapItem * parentItem = Clay__GetHashMapItem ( floatingElementConfig - > parentId ) ;
2024-08-26 07:05:43 +00:00
if ( parentItem ) {
Clay_LayoutElement * parentLayoutElement = parentItem - > layoutElement ;
if ( rootElement - > layoutConfig - > sizing . width . type = = CLAY__SIZING_TYPE_GROW ) {
2024-09-16 09:34:59 +00:00
rootElement - > dimensions . width = parentLayoutElement - > dimensions . width ;
2024-08-26 07:05:43 +00:00
}
if ( rootElement - > layoutConfig - > sizing . height . type = = CLAY__SIZING_TYPE_GROW ) {
2024-09-16 09:34:59 +00:00
rootElement - > dimensions . height = parentLayoutElement - > dimensions . height ;
2024-08-26 07:05:43 +00:00
}
}
}
2024-12-26 22:12:59 +00:00
rootElement - > dimensions . width = CLAY__MIN ( CLAY__MAX ( rootElement - > dimensions . width , rootElement - > layoutConfig - > sizing . width . size . minMax . min ) , rootElement - > layoutConfig - > sizing . width . size . minMax . max ) ;
rootElement - > dimensions . height = CLAY__MIN ( CLAY__MAX ( rootElement - > dimensions . height , rootElement - > layoutConfig - > sizing . height . size . minMax . min ) , rootElement - > layoutConfig - > sizing . height . size . minMax . max ) ;
2024-08-23 04:05:23 +00:00
2024-12-30 00:09:14 +00:00
for ( uint32_t i = 0 ; i < bfsBuffer . length ; + + i ) {
2024-09-16 09:34:59 +00:00
int32_t parentIndex = Clay__int32_tArray_Get ( & bfsBuffer , i ) ;
Clay_LayoutElement * parent = Clay_LayoutElementArray_Get ( & Clay__layoutElements , parentIndex ) ;
2024-08-23 04:05:23 +00:00
Clay_LayoutConfig * parentStyleConfig = parent - > layoutConfig ;
2024-10-07 05:56:13 +00:00
int growContainerCount = 0 ;
2024-08-23 04:05:23 +00:00
float parentSize = xAxis ? parent - > dimensions . width : parent - > dimensions . height ;
float parentPadding = ( float ) ( xAxis ? parent - > layoutConfig - > padding . x : parent - > layoutConfig - > padding . y ) ;
2024-10-07 05:56:13 +00:00
float innerContentSize = 0 , growContainerContentSize = 0 , totalPaddingAndChildGaps = parentPadding * 2 ;
2024-08-23 04:05:23 +00:00
bool sizingAlongAxis = ( xAxis & & parentStyleConfig - > layoutDirection = = CLAY_LEFT_TO_RIGHT ) | | ( ! xAxis & & parentStyleConfig - > layoutDirection = = CLAY_TOP_TO_BOTTOM ) ;
resizableContainerBuffer . length = 0 ;
float parentChildGap = parentStyleConfig - > childGap ;
2024-12-30 00:09:14 +00:00
for ( int childOffset = 0 ; childOffset < parent - > childrenOrTextContent . children . length ; childOffset + + ) {
int32_t childElementIndex = parent - > childrenOrTextContent . children . elements [ childOffset ] ;
2024-09-16 09:34:59 +00:00
Clay_LayoutElement * childElement = Clay_LayoutElementArray_Get ( & Clay__layoutElements , childElementIndex ) ;
2024-08-23 04:05:23 +00:00
Clay_SizingAxis childSizing = xAxis ? childElement - > layoutConfig - > sizing . width : childElement - > layoutConfig - > sizing . height ;
float childSize = xAxis ? childElement - > dimensions . width : childElement - > dimensions . height ;
2024-12-30 00:09:14 +00:00
if ( ! Clay__ElementHasConfig ( childElement , CLAY__ELEMENT_CONFIG_TYPE_TEXT ) & & childElement - > childrenOrTextContent . children . length > 0 ) {
2024-09-16 09:34:59 +00:00
Clay__int32_tArray_Add ( & bfsBuffer , childElementIndex ) ;
2024-08-23 04:05:23 +00:00
}
2024-10-22 07:41:35 +00:00
if ( childSizing . type ! = CLAY__SIZING_TYPE_PERCENT & & childSizing . type ! = CLAY__SIZING_TYPE_FIXED & & ( ! Clay__ElementHasConfig ( childElement , CLAY__ELEMENT_CONFIG_TYPE_TEXT ) | | ( Clay__FindElementConfigWithType ( childElement , CLAY__ELEMENT_CONFIG_TYPE_TEXT ) . textElementConfig - > wrapMode = = CLAY_TEXT_WRAP_WORDS ) ) ) {
Clay__int32_tArray_Add ( & resizableContainerBuffer , childElementIndex ) ;
2024-08-23 04:05:23 +00:00
}
if ( sizingAlongAxis ) {
innerContentSize + = ( childSizing . type = = CLAY__SIZING_TYPE_PERCENT ? 0 : childSize ) ;
2024-10-07 05:56:13 +00:00
if ( childSizing . type = = CLAY__SIZING_TYPE_GROW ) {
growContainerContentSize + = childSize ;
growContainerCount + + ;
}
2024-08-23 04:05:23 +00:00
if ( childOffset > 0 ) {
innerContentSize + = parentChildGap ; // For children after index 0, the childAxisOffset is the gap from the previous child
totalPaddingAndChildGaps + = parentChildGap ;
}
} else {
innerContentSize = CLAY__MAX ( childSize , innerContentSize ) ;
}
}
// Expand percentage containers to size
2024-12-30 00:09:14 +00:00
for ( int childOffset = 0 ; childOffset < parent - > childrenOrTextContent . children . length ; childOffset + + ) {
int32_t childElementIndex = parent - > childrenOrTextContent . children . elements [ childOffset ] ;
2024-09-16 09:34:59 +00:00
Clay_LayoutElement * childElement = Clay_LayoutElementArray_Get ( & Clay__layoutElements , childElementIndex ) ;
2024-08-23 04:05:23 +00:00
Clay_SizingAxis childSizing = xAxis ? childElement - > layoutConfig - > sizing . width : childElement - > layoutConfig - > sizing . height ;
float * childSize = xAxis ? & childElement - > dimensions . width : & childElement - > dimensions . height ;
if ( childSizing . type = = CLAY__SIZING_TYPE_PERCENT ) {
2024-12-26 22:12:59 +00:00
* childSize = ( parentSize - totalPaddingAndChildGaps ) * childSizing . size . percent ;
2024-08-23 04:05:23 +00:00
if ( sizingAlongAxis ) {
innerContentSize + = * childSize ;
if ( childOffset > 0 ) {
innerContentSize + = parentChildGap ; // For children after index 0, the childAxisOffset is the gap from the previous child
totalPaddingAndChildGaps + = parentChildGap ;
}
} else {
innerContentSize = CLAY__MAX ( * childSize , innerContentSize ) ;
}
}
}
if ( sizingAlongAxis ) {
float sizeToDistribute = parentSize - parentPadding * 2 - innerContentSize ;
// If the content is too large, compress the children as much as possible
if ( sizeToDistribute < 0 ) {
2024-10-07 05:56:13 +00:00
// If the parent can scroll in the axis direction in this direction, don't compress children, just leave them alone
2024-10-22 07:41:35 +00:00
if ( Clay__ElementHasConfig ( parent , CLAY__ELEMENT_CONFIG_TYPE_SCROLL_CONTAINER ) ) {
Clay_ScrollElementConfig * scrollElementConfig = Clay__FindElementConfigWithType ( parent , CLAY__ELEMENT_CONFIG_TYPE_SCROLL_CONTAINER ) . scrollElementConfig ;
if ( ( ( xAxis & & scrollElementConfig - > horizontal ) | | ( ! xAxis & & scrollElementConfig - > vertical ) ) ) {
2024-08-23 04:05:23 +00:00
continue ;
}
}
// Scrolling containers preferentially compress before others
sizeToDistribute = Clay__DistributeSizeAmongChildren ( xAxis , sizeToDistribute , resizableContainerBuffer , CLAY__SIZE_DISTRIBUTION_TYPE_SCROLL_CONTAINER ) ;
// If there is still height to make up, remove it from all containers that haven't hit their minimum size
if ( sizeToDistribute < 0 ) {
Clay__DistributeSizeAmongChildren ( xAxis , sizeToDistribute , resizableContainerBuffer , CLAY__SIZE_DISTRIBUTION_TYPE_RESIZEABLE_CONTAINER ) ;
}
// The content is too small, allow SIZING_GROW containers to expand
2024-10-07 05:56:13 +00:00
} else if ( sizeToDistribute > 0 & & growContainerCount > 0 ) {
float targetSize = ( sizeToDistribute + growContainerContentSize ) / growContainerCount ;
2024-12-30 00:09:14 +00:00
for ( uint32_t childOffset = 0 ; childOffset < resizableContainerBuffer . length ; childOffset + + ) {
2024-10-22 07:41:35 +00:00
Clay_LayoutElement * childElement = Clay_LayoutElementArray_Get ( & Clay__layoutElements , Clay__int32_tArray_Get ( & resizableContainerBuffer , childOffset ) ) ;
2024-10-07 05:56:13 +00:00
Clay_SizingAxis childSizing = xAxis ? childElement - > layoutConfig - > sizing . width : childElement - > layoutConfig - > sizing . height ;
if ( childSizing . type = = CLAY__SIZING_TYPE_GROW ) {
float * childSize = xAxis ? & childElement - > dimensions . width : & childElement - > dimensions . height ;
float * minSize = xAxis ? & childElement - > minDimensions . width : & childElement - > minDimensions . height ;
if ( targetSize < * minSize ) {
growContainerContentSize - = * minSize ;
2024-10-22 07:41:35 +00:00
Clay__int32_tArray_RemoveSwapback ( & resizableContainerBuffer , childOffset ) ;
2024-10-07 05:56:13 +00:00
growContainerCount - - ;
targetSize = ( sizeToDistribute + growContainerContentSize ) / growContainerCount ;
childOffset = - 1 ;
continue ;
}
* childSize = targetSize ;
}
}
2024-08-23 04:05:23 +00:00
}
// Sizing along the non layout axis ("off axis")
} else {
2024-12-30 00:09:14 +00:00
for ( uint32_t childOffset = 0 ; childOffset < resizableContainerBuffer . length ; childOffset + + ) {
2024-10-22 07:41:35 +00:00
Clay_LayoutElement * childElement = Clay_LayoutElementArray_Get ( & Clay__layoutElements , Clay__int32_tArray_Get ( & resizableContainerBuffer , childOffset ) ) ;
2024-08-23 04:05:23 +00:00
Clay_SizingAxis childSizing = xAxis ? childElement - > layoutConfig - > sizing . width : childElement - > layoutConfig - > sizing . height ;
float * childSize = xAxis ? & childElement - > dimensions . width : & childElement - > dimensions . height ;
2024-10-22 07:41:35 +00:00
if ( ! xAxis & & Clay__ElementHasConfig ( childElement , CLAY__ELEMENT_CONFIG_TYPE_IMAGE ) ) {
2024-08-23 04:05:23 +00:00
continue ; // Currently we don't support resizing aspect ratio images on the Y axis because it would break the ratio
}
// If we're laying out the children of a scroll panel, grow containers expand to the height of the inner content, not the outer container
float maxSize = parentSize - parentPadding * 2 ;
2024-10-22 07:41:35 +00:00
if ( Clay__ElementHasConfig ( parent , CLAY__ELEMENT_CONFIG_TYPE_SCROLL_CONTAINER ) ) {
Clay_ScrollElementConfig * scrollElementConfig = Clay__FindElementConfigWithType ( parent , CLAY__ELEMENT_CONFIG_TYPE_SCROLL_CONTAINER ) . scrollElementConfig ;
if ( ( ( xAxis & & scrollElementConfig - > horizontal ) | | ( ! xAxis & & scrollElementConfig - > vertical ) ) ) {
maxSize = CLAY__MAX ( maxSize , innerContentSize ) ;
}
2024-08-23 04:05:23 +00:00
}
if ( childSizing . type = = CLAY__SIZING_TYPE_FIT ) {
2024-12-26 22:12:59 +00:00
* childSize = CLAY__MAX ( childSizing . size . minMax . min , CLAY__MIN ( * childSize , maxSize ) ) ;
2024-08-23 04:05:23 +00:00
} else if ( childSizing . type = = CLAY__SIZING_TYPE_GROW ) {
2024-12-26 22:12:59 +00:00
* childSize = CLAY__MIN ( maxSize , childSizing . size . minMax . max ) ;
2024-08-23 04:05:23 +00:00
}
}
}
}
}
}
2024-12-01 08:05:56 +00:00
Clay_String Clay__IntToString ( int integer ) {
if ( integer = = 0 ) {
return CLAY__INIT ( Clay_String ) { . length = 1 , . chars = " 0 " } ;
}
char * chars = ( char * ) ( Clay__dynamicStringData . internalArray + Clay__dynamicStringData . length ) ;
2024-12-30 00:09:14 +00:00
size_t length = 0 ;
2024-12-01 08:05:56 +00:00
int sign = integer ;
if ( integer < 0 ) {
integer = - integer ;
}
while ( integer > 0 ) {
chars [ length + + ] = ( char ) ( integer % 10 + ' 0 ' ) ;
integer / = 10 ;
}
if ( sign < 0 ) {
chars [ length + + ] = ' - ' ;
}
// Reverse the string to get the correct order
for ( int j = 0 , k = length - 1 ; j < k ; j + + , k - - ) {
char temp = chars [ j ] ;
chars [ j ] = chars [ k ] ;
chars [ k ] = temp ;
}
Clay__dynamicStringData . length + = length ;
return CLAY__INIT ( Clay_String ) { . length = length , . chars = chars } ;
}
void Clay__AddRenderCommand ( Clay_RenderCommand renderCommand ) {
if ( Clay__renderCommands . length < Clay__renderCommands . capacity - 1 ) {
Clay_RenderCommandArray_Add ( & Clay__renderCommands , renderCommand ) ;
} else {
2024-12-26 06:58:59 +00:00
if ( ! Clay__booleanWarnings . maxRenderCommandsExceeded ) {
Clay__booleanWarnings . maxRenderCommandsExceeded = true ;
Clay__errorHandler . errorHandlerFunction ( CLAY__INIT ( Clay_ErrorData ) {
. errorType = CLAY_ERROR_TYPE_ELEMENTS_CAPACITY_EXCEEDED ,
. errorText = CLAY_STRING ( " Clay ran out of capacity while attempting to create render commands. This is usually caused by a large amount of wrapping text elements while close to the max element capacity. Try using Clay_SetMaxElementCount() with a higher value. " ) ,
. userData = Clay__errorHandler . userData } ) ;
}
2024-12-01 08:05:56 +00:00
}
}
2024-12-18 22:35:18 +00:00
bool Clay__ElementIsOffscreen ( Clay_BoundingBox * boundingBox ) {
if ( Clay__disableCulling ) {
return false ;
}
return ( boundingBox - > x > ( float ) Clay__layoutDimensions . width ) | |
( boundingBox - > y > ( float ) Clay__layoutDimensions . height ) | |
( boundingBox - > x + boundingBox - > width < 0 ) | |
( boundingBox - > y + boundingBox - > height < 0 ) ;
}
2024-12-30 00:09:14 +00:00
void Clay__CalculateFinalLayout ( void ) {
2024-08-23 04:05:23 +00:00
// Calculate sizing along the X axis
Clay__SizeContainersAlongAxis ( true ) ;
// Wrap text
2024-12-30 00:09:14 +00:00
for ( uint32_t textElementIndex = 0 ; textElementIndex < Clay__textElementData . length ; + + textElementIndex ) {
2024-10-22 07:41:35 +00:00
Clay__TextElementData * textElementData = Clay__TextElementDataArray_Get ( & Clay__textElementData , textElementIndex ) ;
2024-12-28 06:15:22 +00:00
textElementData - > wrappedLines = CLAY__INIT ( Clay__WrappedTextLineArraySlice ) { . length = 0 , . internalArray = & Clay__wrappedTextLines . internalArray [ Clay__wrappedTextLines . length ] } ;
2024-09-16 09:34:59 +00:00
Clay_LayoutElement * containerElement = Clay_LayoutElementArray_Get ( & Clay__layoutElements , ( int ) textElementData - > elementIndex ) ;
2024-10-22 07:41:35 +00:00
Clay_TextElementConfig * textConfig = Clay__FindElementConfigWithType ( containerElement , CLAY__ELEMENT_CONFIG_TYPE_TEXT ) . textElementConfig ;
Clay__MeasureTextCacheItem * measureTextCacheItem = Clay__MeasureTextCached ( & textElementData - > text , textConfig ) ;
float lineWidth = 0 ;
float lineHeight = textConfig - > lineHeight > 0 ? textConfig - > lineHeight : textElementData - > preferredDimensions . height ;
uint32_t lineLengthChars = 0 ;
uint32_t lineStartOffset = 0 ;
if ( textElementData - > preferredDimensions . width < = containerElement - > dimensions . width ) {
2024-12-28 06:15:22 +00:00
Clay__WrappedTextLineArray_Add ( & Clay__wrappedTextLines , CLAY__INIT ( Clay__WrappedTextLine ) { containerElement - > dimensions , textElementData - > text } ) ;
2024-10-22 07:41:35 +00:00
textElementData - > wrappedLines . length + + ;
2024-09-16 09:34:59 +00:00
continue ;
}
2024-12-01 08:05:56 +00:00
int32_t wordIndex = measureTextCacheItem - > measuredWordsStartIndex ;
while ( wordIndex ! = - 1 ) {
if ( Clay__wrappedTextLines . length > Clay__wrappedTextLines . capacity - 1 ) {
break ;
}
Clay__MeasuredWord * measuredWord = Clay__MeasuredWordArray_Get ( & Clay__measuredWords , wordIndex ) ;
2024-10-22 08:07:24 +00:00
// Only word on the line is too large, just render it anyway
if ( lineLengthChars = = 0 & & lineWidth + measuredWord - > width > containerElement - > dimensions . width ) {
2024-12-30 00:09:14 +00:00
Clay__WrappedTextLineArray_Add ( & Clay__wrappedTextLines , CLAY__INIT ( Clay__WrappedTextLine ) { { measuredWord - > width , lineHeight } , { . length = measuredWord - > length , . chars = & textElementData - > text . chars [ measuredWord - > startOffset ] } } ) ;
2024-10-22 08:07:24 +00:00
textElementData - > wrappedLines . length + + ;
2024-12-01 08:05:56 +00:00
wordIndex = measuredWord - > next ;
2024-10-22 08:07:24 +00:00
}
2024-10-22 07:41:35 +00:00
// measuredWord->length == 0 means a newline character
2024-10-22 08:07:24 +00:00
else if ( measuredWord - > length = = 0 | | lineWidth + measuredWord - > width > containerElement - > dimensions . width ) {
2024-12-01 08:05:56 +00:00
// Wrapped text lines list has overflowed, just render out the line
2024-12-30 00:09:14 +00:00
Clay__WrappedTextLineArray_Add ( & Clay__wrappedTextLines , CLAY__INIT ( Clay__WrappedTextLine ) { { lineWidth , lineHeight } , { . length = lineLengthChars , . chars = & textElementData - > text . chars [ lineStartOffset ] } } ) ;
2024-10-22 07:41:35 +00:00
textElementData - > wrappedLines . length + + ;
2024-12-01 08:05:56 +00:00
if ( lineLengthChars = = 0 | | measuredWord - > length = = 0 ) {
wordIndex = measuredWord - > next ;
2024-08-23 04:05:23 +00:00
}
2024-10-22 07:41:35 +00:00
lineWidth = 0 ;
lineLengthChars = 0 ;
lineStartOffset = measuredWord - > startOffset ;
2024-08-23 04:05:23 +00:00
} else {
2024-10-22 07:41:35 +00:00
lineWidth + = measuredWord - > width ;
lineLengthChars + = measuredWord - > length ;
2024-12-01 08:05:56 +00:00
wordIndex = measuredWord - > next ;
2024-08-23 04:05:23 +00:00
}
}
2024-10-22 07:41:35 +00:00
if ( lineLengthChars > 0 ) {
2024-12-30 00:09:14 +00:00
Clay__WrappedTextLineArray_Add ( & Clay__wrappedTextLines , CLAY__INIT ( Clay__WrappedTextLine ) { { lineWidth , lineHeight } , { . length = lineLengthChars , . chars = & textElementData - > text . chars [ lineStartOffset ] } } ) ;
2024-10-22 07:41:35 +00:00
textElementData - > wrappedLines . length + + ;
}
containerElement - > dimensions . height = lineHeight * textElementData - > wrappedLines . length ;
2024-08-23 04:05:23 +00:00
}
// Scale vertical image heights according to aspect ratio
2024-12-30 00:09:14 +00:00
for ( uint32_t i = 0 ; i < Clay__imageElementPointers . length ; + + i ) {
2024-10-22 07:41:35 +00:00
Clay_LayoutElement * imageElement = Clay__LayoutElementPointerArray_Get ( & Clay__imageElementPointers , i ) ;
Clay_ImageElementConfig * config = Clay__FindElementConfigWithType ( imageElement , CLAY__ELEMENT_CONFIG_TYPE_IMAGE ) . imageElementConfig ;
2024-08-23 04:05:23 +00:00
imageElement - > dimensions . height = ( config - > sourceDimensions . height / CLAY__MAX ( config - > sourceDimensions . width , 1 ) ) * imageElement - > dimensions . width ;
}
// Propagate effect of text wrapping, image aspect scaling etc. on height of parents
Clay__LayoutElementTreeNodeArray dfsBuffer = Clay__layoutElementTreeNodeArray1 ;
dfsBuffer . length = 0 ;
2024-12-30 00:09:14 +00:00
for ( uint32_t i = 0 ; i < Clay__layoutElementTreeRoots . length ; + + i ) {
2024-08-23 04:05:23 +00:00
Clay__LayoutElementTreeRoot * root = Clay__LayoutElementTreeRootArray_Get ( & Clay__layoutElementTreeRoots , i ) ;
2024-09-16 09:34:59 +00:00
Clay__treeNodeVisited . internalArray [ dfsBuffer . length ] = false ;
2024-09-28 23:53:46 +00:00
Clay__LayoutElementTreeNodeArray_Add ( & dfsBuffer , CLAY__INIT ( Clay__LayoutElementTreeNode ) { . layoutElement = Clay_LayoutElementArray_Get ( & Clay__layoutElements , ( int ) root - > layoutElementIndex ) } ) ;
2024-08-23 04:05:23 +00:00
}
while ( dfsBuffer . length > 0 ) {
Clay__LayoutElementTreeNode * currentElementTreeNode = Clay__LayoutElementTreeNodeArray_Get ( & dfsBuffer , ( int ) dfsBuffer . length - 1 ) ;
Clay_LayoutElement * currentElement = currentElementTreeNode - > layoutElement ;
if ( ! Clay__treeNodeVisited . internalArray [ dfsBuffer . length - 1 ] ) {
Clay__treeNodeVisited . internalArray [ dfsBuffer . length - 1 ] = true ;
// If the element has no children or is the container for a text element, don't bother inspecting it
2024-12-30 00:09:14 +00:00
if ( Clay__ElementHasConfig ( currentElement , CLAY__ELEMENT_CONFIG_TYPE_TEXT ) | | currentElement - > childrenOrTextContent . children . length = = 0 ) {
2024-08-23 04:05:23 +00:00
dfsBuffer . length - - ;
continue ;
}
// Add the children to the DFS buffer (needs to be pushed in reverse so that stack traversal is in correct layout order)
2024-12-30 00:09:14 +00:00
for ( int i = 0 ; i < currentElement - > childrenOrTextContent . children . length ; i + + ) {
2024-08-23 04:05:23 +00:00
Clay__treeNodeVisited . internalArray [ dfsBuffer . length ] = false ;
2024-12-30 00:09:14 +00:00
Clay__LayoutElementTreeNodeArray_Add ( & dfsBuffer , CLAY__INIT ( Clay__LayoutElementTreeNode ) { . layoutElement = Clay_LayoutElementArray_Get ( & Clay__layoutElements , currentElement - > childrenOrTextContent . children . elements [ i ] ) } ) ;
2024-08-23 04:05:23 +00:00
}
continue ;
}
dfsBuffer . length - - ;
// DFS node has been visited, this is on the way back up to the root
Clay_LayoutConfig * layoutConfig = currentElement - > layoutConfig ;
if ( layoutConfig - > sizing . height . type = = CLAY__SIZING_TYPE_PERCENT ) {
continue ;
}
if ( layoutConfig - > layoutDirection = = CLAY_LEFT_TO_RIGHT ) {
// Resize any parent containers that have grown in height along their non layout axis
2024-12-30 00:09:14 +00:00
for ( int j = 0 ; j < currentElement - > childrenOrTextContent . children . length ; + + j ) {
Clay_LayoutElement * childElement = Clay_LayoutElementArray_Get ( & Clay__layoutElements , currentElement - > childrenOrTextContent . children . elements [ j ] ) ;
2024-08-23 04:05:23 +00:00
float childHeightWithPadding = CLAY__MAX ( childElement - > dimensions . height + layoutConfig - > padding . y * 2 , currentElement - > dimensions . height ) ;
2024-12-26 22:12:59 +00:00
currentElement - > dimensions . height = CLAY__MIN ( CLAY__MAX ( childHeightWithPadding , layoutConfig - > sizing . height . size . minMax . min ) , layoutConfig - > sizing . height . size . minMax . max ) ;
2024-08-23 04:05:23 +00:00
}
} else if ( layoutConfig - > layoutDirection = = CLAY_TOP_TO_BOTTOM ) {
// Resizing along the layout axis
float contentHeight = ( float ) layoutConfig - > padding . y * 2 ;
2024-12-30 00:09:14 +00:00
for ( int j = 0 ; j < currentElement - > childrenOrTextContent . children . length ; + + j ) {
Clay_LayoutElement * childElement = Clay_LayoutElementArray_Get ( & Clay__layoutElements , currentElement - > childrenOrTextContent . children . elements [ j ] ) ;
2024-08-23 04:05:23 +00:00
contentHeight + = childElement - > dimensions . height ;
}
2024-12-30 00:09:14 +00:00
contentHeight + = ( float ) ( CLAY__MAX ( currentElement - > childrenOrTextContent . children . length - 1 , 0 ) * layoutConfig - > childGap ) ;
2024-12-26 22:12:59 +00:00
currentElement - > dimensions . height = CLAY__MIN ( CLAY__MAX ( contentHeight , layoutConfig - > sizing . height . size . minMax . min ) , layoutConfig - > sizing . height . size . minMax . max ) ;
2024-08-23 04:05:23 +00:00
}
}
// Calculate sizing along the Y axis
Clay__SizeContainersAlongAxis ( false ) ;
// Calculate final positions and generate render commands
Clay__renderCommands . length = 0 ;
dfsBuffer . length = 0 ;
2024-12-30 00:09:14 +00:00
for ( uint32_t rootIndex = 0 ; rootIndex < Clay__layoutElementTreeRoots . length ; + + rootIndex ) {
2024-08-23 04:05:23 +00:00
dfsBuffer . length = 0 ;
Clay__LayoutElementTreeRoot * root = Clay__LayoutElementTreeRootArray_Get ( & Clay__layoutElementTreeRoots , rootIndex ) ;
2024-09-16 09:34:59 +00:00
Clay_LayoutElement * rootElement = Clay_LayoutElementArray_Get ( & Clay__layoutElements , ( int ) root - > layoutElementIndex ) ;
2024-12-30 00:09:14 +00:00
Clay_Vector2 rootPosition = { 0 } ;
2024-08-23 04:05:23 +00:00
Clay_LayoutElementHashMapItem * parentHashMapItem = Clay__GetHashMapItem ( root - > parentId ) ;
// Position root floating containers
2024-10-22 07:41:35 +00:00
if ( Clay__ElementHasConfig ( rootElement , CLAY__ELEMENT_CONFIG_TYPE_FLOATING_CONTAINER ) & & parentHashMapItem ) {
Clay_FloatingElementConfig * config = Clay__FindElementConfigWithType ( rootElement , CLAY__ELEMENT_CONFIG_TYPE_FLOATING_CONTAINER ) . floatingElementConfig ;
2024-09-16 09:34:59 +00:00
Clay_Dimensions rootDimensions = rootElement - > dimensions ;
2024-08-28 04:13:28 +00:00
Clay_BoundingBox parentBoundingBox = parentHashMapItem - > boundingBox ;
2024-08-23 04:05:23 +00:00
// Set X position
2024-12-30 00:09:14 +00:00
Clay_Vector2 targetAttachPosition = { 0 } ;
2024-08-23 04:05:23 +00:00
switch ( config - > attachment . parent ) {
case CLAY_ATTACH_POINT_LEFT_TOP :
case CLAY_ATTACH_POINT_LEFT_CENTER :
case CLAY_ATTACH_POINT_LEFT_BOTTOM : targetAttachPosition . x = parentBoundingBox . x ; break ;
case CLAY_ATTACH_POINT_CENTER_TOP :
case CLAY_ATTACH_POINT_CENTER_CENTER :
case CLAY_ATTACH_POINT_CENTER_BOTTOM : targetAttachPosition . x = parentBoundingBox . x + ( parentBoundingBox . width / 2 ) ; break ;
case CLAY_ATTACH_POINT_RIGHT_TOP :
case CLAY_ATTACH_POINT_RIGHT_CENTER :
case CLAY_ATTACH_POINT_RIGHT_BOTTOM : targetAttachPosition . x = parentBoundingBox . x + parentBoundingBox . width ; break ;
}
switch ( config - > attachment . element ) {
case CLAY_ATTACH_POINT_LEFT_TOP :
case CLAY_ATTACH_POINT_LEFT_CENTER :
case CLAY_ATTACH_POINT_LEFT_BOTTOM : break ;
case CLAY_ATTACH_POINT_CENTER_TOP :
case CLAY_ATTACH_POINT_CENTER_CENTER :
case CLAY_ATTACH_POINT_CENTER_BOTTOM : targetAttachPosition . x - = ( rootDimensions . width / 2 ) ; break ;
case CLAY_ATTACH_POINT_RIGHT_TOP :
case CLAY_ATTACH_POINT_RIGHT_CENTER :
case CLAY_ATTACH_POINT_RIGHT_BOTTOM : targetAttachPosition . x - = rootDimensions . width ; break ;
}
switch ( config - > attachment . parent ) { // I know I could merge the x and y switch statements, but this is easier to read
case CLAY_ATTACH_POINT_LEFT_TOP :
case CLAY_ATTACH_POINT_RIGHT_TOP :
case CLAY_ATTACH_POINT_CENTER_TOP : targetAttachPosition . y = parentBoundingBox . y ; break ;
case CLAY_ATTACH_POINT_LEFT_CENTER :
case CLAY_ATTACH_POINT_CENTER_CENTER :
case CLAY_ATTACH_POINT_RIGHT_CENTER : targetAttachPosition . y = parentBoundingBox . y + ( parentBoundingBox . height / 2 ) ; break ;
case CLAY_ATTACH_POINT_LEFT_BOTTOM :
case CLAY_ATTACH_POINT_CENTER_BOTTOM :
case CLAY_ATTACH_POINT_RIGHT_BOTTOM : targetAttachPosition . y = parentBoundingBox . y + parentBoundingBox . height ; break ;
}
switch ( config - > attachment . element ) {
case CLAY_ATTACH_POINT_LEFT_TOP :
case CLAY_ATTACH_POINT_RIGHT_TOP :
case CLAY_ATTACH_POINT_CENTER_TOP : break ;
case CLAY_ATTACH_POINT_LEFT_CENTER :
case CLAY_ATTACH_POINT_CENTER_CENTER :
case CLAY_ATTACH_POINT_RIGHT_CENTER : targetAttachPosition . y - = ( rootDimensions . height / 2 ) ; break ;
case CLAY_ATTACH_POINT_LEFT_BOTTOM :
case CLAY_ATTACH_POINT_CENTER_BOTTOM :
case CLAY_ATTACH_POINT_RIGHT_BOTTOM : targetAttachPosition . y - = rootDimensions . height ; break ;
}
targetAttachPosition . x + = config - > offset . x ;
targetAttachPosition . y + = config - > offset . y ;
rootPosition = targetAttachPosition ;
}
if ( root - > clipElementId ) {
Clay_LayoutElementHashMapItem * clipHashMapItem = Clay__GetHashMapItem ( root - > clipElementId ) ;
if ( clipHashMapItem ) {
2024-12-18 22:35:18 +00:00
// Floating elements that are attached to scrolling contents won't be correctly positioned if external scroll handling is enabled, fix here
if ( Clay__externalScrollHandlingEnabled ) {
Clay_ScrollElementConfig * scrollConfig = Clay__FindElementConfigWithType ( clipHashMapItem - > layoutElement , CLAY__ELEMENT_CONFIG_TYPE_SCROLL_CONTAINER ) . scrollElementConfig ;
2024-12-30 00:09:14 +00:00
for ( uint32_t i = 0 ; i < Clay__scrollContainerDatas . length ; i + + ) {
2024-12-18 22:35:18 +00:00
Clay__ScrollContainerDataInternal * mapping = Clay__ScrollContainerDataInternalArray_Get ( & Clay__scrollContainerDatas , i ) ;
if ( mapping - > layoutElement = = clipHashMapItem - > layoutElement ) {
root - > pointerOffset = mapping - > scrollPosition ;
if ( scrollConfig - > horizontal ) {
rootPosition . x + = mapping - > scrollPosition . x ;
}
if ( scrollConfig - > vertical ) {
rootPosition . y + = mapping - > scrollPosition . y ;
}
break ;
}
}
}
2024-12-01 08:05:56 +00:00
Clay__AddRenderCommand ( CLAY__INIT ( Clay_RenderCommand ) {
2024-09-28 23:53:46 +00:00
. boundingBox = clipHashMapItem - > boundingBox ,
2024-12-30 00:09:14 +00:00
. config = { . scrollElementConfig = Clay__StoreScrollElementConfig ( CLAY__INIT ( Clay_ScrollElementConfig ) { 0 } ) } ,
2024-09-16 09:34:59 +00:00
. id = Clay__RehashWithNumber ( rootElement - > id , 10 ) , // TODO need a better strategy for managing derived ids
2024-08-23 04:05:23 +00:00
. commandType = CLAY_RENDER_COMMAND_TYPE_SCISSOR_START ,
} ) ;
}
}
2024-12-30 00:09:14 +00:00
Clay__LayoutElementTreeNodeArray_Add ( & dfsBuffer , CLAY__INIT ( Clay__LayoutElementTreeNode ) { . layoutElement = rootElement , . position = rootPosition , . nextChildOffset = { . x = ( float ) rootElement - > layoutConfig - > padding . x , . y = ( float ) rootElement - > layoutConfig - > padding . y } } ) ;
2024-08-23 04:05:23 +00:00
Clay__treeNodeVisited . internalArray [ 0 ] = false ;
while ( dfsBuffer . length > 0 ) {
Clay__LayoutElementTreeNode * currentElementTreeNode = Clay__LayoutElementTreeNodeArray_Get ( & dfsBuffer , ( int ) dfsBuffer . length - 1 ) ;
Clay_LayoutElement * currentElement = currentElementTreeNode - > layoutElement ;
Clay_LayoutConfig * layoutConfig = currentElement - > layoutConfig ;
Clay_Vector2 scrollOffset = { 0 } ;
// This will only be run a single time for each element in downwards DFS order
if ( ! Clay__treeNodeVisited . internalArray [ dfsBuffer . length - 1 ] ) {
Clay__treeNodeVisited . internalArray [ dfsBuffer . length - 1 ] = true ;
2024-12-30 00:09:14 +00:00
Clay_BoundingBox currentElementBoundingBox = { currentElementTreeNode - > position . x , currentElementTreeNode - > position . y , currentElement - > dimensions . width , currentElement - > dimensions . height } ;
2024-10-22 07:41:35 +00:00
if ( Clay__ElementHasConfig ( currentElement , CLAY__ELEMENT_CONFIG_TYPE_FLOATING_CONTAINER ) ) {
Clay_FloatingElementConfig * floatingElementConfig = Clay__FindElementConfigWithType ( currentElement , CLAY__ELEMENT_CONFIG_TYPE_FLOATING_CONTAINER ) . floatingElementConfig ;
2024-08-23 04:05:23 +00:00
Clay_Dimensions expand = floatingElementConfig - > expand ;
currentElementBoundingBox . x - = expand . width ;
currentElementBoundingBox . width + = expand . width * 2 ;
currentElementBoundingBox . y - = expand . height ;
currentElementBoundingBox . height + = expand . height * 2 ;
}
2024-08-26 07:05:43 +00:00
Clay__ScrollContainerDataInternal * scrollContainerData = CLAY__NULL ;
2024-08-23 04:05:23 +00:00
// Apply scroll offsets to container
2024-10-22 07:41:35 +00:00
if ( Clay__ElementHasConfig ( currentElement , CLAY__ELEMENT_CONFIG_TYPE_SCROLL_CONTAINER ) ) {
Clay_ScrollElementConfig * scrollConfig = Clay__FindElementConfigWithType ( currentElement , CLAY__ELEMENT_CONFIG_TYPE_SCROLL_CONTAINER ) . scrollElementConfig ;
2024-08-23 04:05:23 +00:00
// This linear scan could theoretically be slow under very strange conditions, but I can't imagine a real UI with more than a few 10's of scroll containers
2024-12-30 00:09:14 +00:00
for ( uint32_t i = 0 ; i < Clay__scrollContainerDatas . length ; i + + ) {
2024-08-26 07:05:43 +00:00
Clay__ScrollContainerDataInternal * mapping = Clay__ScrollContainerDataInternalArray_Get ( & Clay__scrollContainerDatas , i ) ;
2024-08-23 04:05:23 +00:00
if ( mapping - > layoutElement = = currentElement ) {
scrollContainerData = mapping ;
mapping - > boundingBox = currentElementBoundingBox ;
2024-10-22 07:41:35 +00:00
if ( scrollConfig - > horizontal ) {
2024-08-23 04:05:23 +00:00
scrollOffset . x = mapping - > scrollPosition . x ;
}
2024-10-22 07:41:35 +00:00
if ( scrollConfig - > vertical ) {
2024-08-23 04:05:23 +00:00
scrollOffset . y = mapping - > scrollPosition . y ;
}
2024-12-18 22:35:18 +00:00
if ( Clay__externalScrollHandlingEnabled ) {
2024-12-30 00:09:14 +00:00
scrollOffset = CLAY__INIT ( Clay_Vector2 ) { 0 } ;
2024-12-18 22:35:18 +00:00
}
2024-08-23 04:05:23 +00:00
break ;
}
}
}
2024-09-16 09:34:59 +00:00
Clay_LayoutElementHashMapItem * hashMapItem = Clay__GetHashMapItem ( currentElement - > id ) ;
2024-08-23 04:05:23 +00:00
if ( hashMapItem ) {
2024-10-22 07:41:35 +00:00
hashMapItem - > boundingBox = currentElementBoundingBox ;
2024-08-23 04:05:23 +00:00
}
2024-10-22 07:41:35 +00:00
int sortedConfigIndexes [ 20 ] ;
2024-12-30 00:09:14 +00:00
for ( uint32_t elementConfigIndex = 0 ; elementConfigIndex < currentElement - > elementConfigs . length ; + + elementConfigIndex ) {
2024-10-22 07:41:35 +00:00
sortedConfigIndexes [ elementConfigIndex ] = elementConfigIndex ;
}
int sortMax = currentElement - > elementConfigs . length - 1 ;
2024-12-18 22:35:18 +00:00
while ( sortMax > 0 ) { // todo dumb bubble sort
2024-10-22 07:41:35 +00:00
for ( int i = 0 ; i < sortMax ; + + i ) {
int current = sortedConfigIndexes [ i ] ;
int next = sortedConfigIndexes [ i + 1 ] ;
Clay__ElementConfigType currentType = Clay__ElementConfigArraySlice_Get ( & currentElement - > elementConfigs , current ) - > type ;
Clay__ElementConfigType nextType = Clay__ElementConfigArraySlice_Get ( & currentElement - > elementConfigs , next ) - > type ;
if ( nextType = = CLAY__ELEMENT_CONFIG_TYPE_SCROLL_CONTAINER | | currentType = = CLAY__ELEMENT_CONFIG_TYPE_BORDER_CONTAINER ) {
sortedConfigIndexes [ i ] = next ;
sortedConfigIndexes [ i + 1 ] = current ;
}
2024-08-23 04:05:23 +00:00
}
2024-10-22 07:41:35 +00:00
sortMax - - ;
}
// Create the render commands for this element
2024-12-30 00:09:14 +00:00
for ( uint32_t elementConfigIndex = 0 ; elementConfigIndex < currentElement - > elementConfigs . length ; + + elementConfigIndex ) {
2024-10-22 07:41:35 +00:00
Clay_ElementConfig * elementConfig = Clay__ElementConfigArraySlice_Get ( & currentElement - > elementConfigs , sortedConfigIndexes [ elementConfigIndex ] ) ;
2024-12-30 00:09:14 +00:00
Clay_RenderCommand renderCommand = {
2024-10-22 07:41:35 +00:00
. boundingBox = currentElementBoundingBox ,
. config = elementConfig - > config ,
. id = currentElement - > id ,
} ;
2024-12-18 22:35:18 +00:00
bool offscreen = Clay__ElementIsOffscreen ( & currentElementBoundingBox ) ;
2024-10-22 07:41:35 +00:00
// Culling - Don't bother to generate render commands for rectangles entirely outside the screen - this won't stop their children from being rendered if they overflow
bool shouldRender = ! offscreen ;
switch ( elementConfig - > type ) {
case CLAY__ELEMENT_CONFIG_TYPE_RECTANGLE : {
renderCommand . commandType = CLAY_RENDER_COMMAND_TYPE_RECTANGLE ;
break ;
}
2024-12-18 22:35:18 +00:00
case CLAY__ELEMENT_CONFIG_TYPE_BORDER_CONTAINER : {
shouldRender = false ;
break ;
}
2024-10-22 07:41:35 +00:00
case CLAY__ELEMENT_CONFIG_TYPE_FLOATING_CONTAINER : {
renderCommand . commandType = CLAY_RENDER_COMMAND_TYPE_NONE ;
shouldRender = false ;
break ;
}
case CLAY__ELEMENT_CONFIG_TYPE_SCROLL_CONTAINER : {
renderCommand . commandType = CLAY_RENDER_COMMAND_TYPE_SCISSOR_START ;
2024-12-18 22:35:18 +00:00
shouldRender = true ;
2024-10-22 07:41:35 +00:00
break ;
}
case CLAY__ELEMENT_CONFIG_TYPE_IMAGE : {
renderCommand . commandType = CLAY_RENDER_COMMAND_TYPE_IMAGE ;
break ;
}
case CLAY__ELEMENT_CONFIG_TYPE_TEXT : {
2024-12-18 22:35:18 +00:00
if ( ! shouldRender ) {
break ;
}
2024-10-22 07:41:35 +00:00
shouldRender = false ;
Clay_ElementConfigUnion configUnion = elementConfig - > config ;
Clay_TextElementConfig * textElementConfig = configUnion . textElementConfig ;
2024-12-30 00:09:14 +00:00
float naturalLineHeight = currentElement - > childrenOrTextContent . textElementData - > preferredDimensions . height ;
2024-10-22 07:41:35 +00:00
float finalLineHeight = textElementConfig - > lineHeight > 0 ? textElementConfig - > lineHeight : naturalLineHeight ;
float lineHeightOffset = ( finalLineHeight - naturalLineHeight ) / 2 ;
float yPosition = lineHeightOffset ;
2024-12-30 00:09:14 +00:00
for ( uint32_t lineIndex = 0 ; lineIndex < currentElement - > childrenOrTextContent . textElementData - > wrappedLines . length ; + + lineIndex ) {
Clay__WrappedTextLine wrappedLine = currentElement - > childrenOrTextContent . textElementData - > wrappedLines . internalArray [ lineIndex ] ; // todo range check
2024-12-28 06:15:22 +00:00
if ( wrappedLine . line . length = = 0 ) {
2024-12-04 02:45:46 +00:00
yPosition + = finalLineHeight ;
2024-10-22 07:41:35 +00:00
continue ;
}
2024-12-01 08:05:56 +00:00
Clay__AddRenderCommand ( CLAY__INIT ( Clay_RenderCommand ) {
2024-12-28 06:15:22 +00:00
. boundingBox = { currentElementBoundingBox . x , currentElementBoundingBox . y + yPosition , wrappedLine . dimensions . width , wrappedLine . dimensions . height } , // TODO width
2024-10-22 07:41:35 +00:00
. config = configUnion ,
2024-12-28 06:15:22 +00:00
. text = wrappedLine . line ,
2024-10-22 07:41:35 +00:00
. id = Clay__HashNumber ( lineIndex , currentElement - > id ) . id ,
. commandType = CLAY_RENDER_COMMAND_TYPE_TEXT ,
} ) ;
yPosition + = finalLineHeight ;
2024-12-18 22:35:18 +00:00
if ( ! Clay__disableCulling & & ( currentElementBoundingBox . y + yPosition > Clay__layoutDimensions . height ) ) {
2024-10-22 07:41:35 +00:00
break ;
}
}
break ;
}
case CLAY__ELEMENT_CONFIG_TYPE_CUSTOM : {
renderCommand . commandType = CLAY_RENDER_COMMAND_TYPE_CUSTOM ;
break ;
2024-10-05 07:57:52 +00:00
}
2024-10-22 07:41:35 +00:00
default : break ;
2024-08-23 04:05:23 +00:00
}
2024-10-22 07:41:35 +00:00
if ( shouldRender ) {
2024-12-01 08:05:56 +00:00
Clay__AddRenderCommand ( renderCommand ) ;
2024-10-22 07:41:35 +00:00
}
if ( offscreen ) {
// NOTE: You may be tempted to try an early return / continue if an element is off screen. Why bother calculating layout for its children, right?
// Unfortunately, a FLOATING_CONTAINER may be defined that attaches to a child or grandchild of this element, which is large enough to still
// be on screen, even if this element isn't. That depends on this element and it's children being laid out correctly (even if they are entirely off screen)
2024-08-23 04:05:23 +00:00
}
}
2024-10-22 07:41:35 +00:00
// Setup initial on-axis alignment
if ( ! Clay__ElementHasConfig ( currentElementTreeNode - > layoutElement , CLAY__ELEMENT_CONFIG_TYPE_TEXT ) ) {
2024-12-30 00:09:14 +00:00
Clay_Dimensions contentSize = { 0 , 0 } ;
2024-08-23 04:05:23 +00:00
if ( layoutConfig - > layoutDirection = = CLAY_LEFT_TO_RIGHT ) {
2024-12-30 00:09:14 +00:00
for ( int i = 0 ; i < currentElement - > childrenOrTextContent . children . length ; + + i ) {
Clay_LayoutElement * childElement = Clay_LayoutElementArray_Get ( & Clay__layoutElements , currentElement - > childrenOrTextContent . children . elements [ i ] ) ;
2024-08-23 04:05:23 +00:00
contentSize . width + = childElement - > dimensions . width ;
contentSize . height = CLAY__MAX ( contentSize . height , childElement - > dimensions . height ) ;
}
2024-12-30 00:09:14 +00:00
contentSize . width + = ( float ) ( CLAY__MAX ( currentElement - > childrenOrTextContent . children . length - 1 , 0 ) * layoutConfig - > childGap ) ;
2024-08-23 04:05:23 +00:00
float extraSpace = currentElement - > dimensions . width - ( float ) layoutConfig - > padding . x * 2 - contentSize . width ;
switch ( layoutConfig - > childAlignment . x ) {
case CLAY_ALIGN_X_LEFT : extraSpace = 0 ; break ;
case CLAY_ALIGN_X_CENTER : extraSpace / = 2 ; break ;
default : break ;
}
currentElementTreeNode - > nextChildOffset . x + = extraSpace ;
} else {
2024-12-30 00:09:14 +00:00
for ( int i = 0 ; i < currentElement - > childrenOrTextContent . children . length ; + + i ) {
Clay_LayoutElement * childElement = Clay_LayoutElementArray_Get ( & Clay__layoutElements , currentElement - > childrenOrTextContent . children . elements [ i ] ) ;
2024-08-23 04:05:23 +00:00
contentSize . width = CLAY__MAX ( contentSize . width , childElement - > dimensions . width ) ;
contentSize . height + = childElement - > dimensions . height ;
}
2024-12-30 00:09:14 +00:00
contentSize . height + = ( float ) ( CLAY__MAX ( currentElement - > childrenOrTextContent . children . length - 1 , 0 ) * layoutConfig - > childGap ) ;
2024-08-23 04:05:23 +00:00
float extraSpace = currentElement - > dimensions . height - ( float ) layoutConfig - > padding . y * 2 - contentSize . height ;
switch ( layoutConfig - > childAlignment . y ) {
case CLAY_ALIGN_Y_TOP : extraSpace = 0 ; break ;
case CLAY_ALIGN_Y_CENTER : extraSpace / = 2 ; break ;
default : break ;
}
currentElementTreeNode - > nextChildOffset . y + = extraSpace ;
}
if ( scrollContainerData ) {
2024-10-22 07:41:35 +00:00
scrollContainerData - > contentSize = CLAY__INIT ( Clay_Dimensions ) { contentSize . width + layoutConfig - > padding . x * 2 , contentSize . height + layoutConfig - > padding . y * 2 } ;
2024-08-23 04:05:23 +00:00
}
}
2024-10-22 07:41:35 +00:00
}
else {
2024-08-23 04:05:23 +00:00
// DFS is returning upwards backwards
2024-10-22 07:41:35 +00:00
bool closeScrollElement = false ;
if ( Clay__ElementHasConfig ( currentElement , CLAY__ELEMENT_CONFIG_TYPE_SCROLL_CONTAINER ) ) {
closeScrollElement = true ;
Clay_ScrollElementConfig * scrollConfig = Clay__FindElementConfigWithType ( currentElement , CLAY__ELEMENT_CONFIG_TYPE_SCROLL_CONTAINER ) . scrollElementConfig ;
2024-12-30 00:09:14 +00:00
for ( uint32_t i = 0 ; i < Clay__scrollContainerDatas . length ; i + + ) {
2024-10-22 07:41:35 +00:00
Clay__ScrollContainerDataInternal * mapping = Clay__ScrollContainerDataInternalArray_Get ( & Clay__scrollContainerDatas , i ) ;
if ( mapping - > layoutElement = = currentElement ) {
if ( scrollConfig - > horizontal ) { scrollOffset . x = mapping - > scrollPosition . x ; }
if ( scrollConfig - > vertical ) { scrollOffset . y = mapping - > scrollPosition . y ; }
2024-12-18 22:35:18 +00:00
if ( Clay__externalScrollHandlingEnabled ) {
2024-12-30 00:09:14 +00:00
scrollOffset = CLAY__INIT ( Clay_Vector2 ) { 0 } ;
2024-12-18 22:35:18 +00:00
}
2024-10-22 07:41:35 +00:00
break ;
}
2024-08-26 22:16:55 +00:00
}
2024-10-22 07:41:35 +00:00
}
2024-12-18 22:35:18 +00:00
2024-10-22 07:41:35 +00:00
if ( Clay__ElementHasConfig ( currentElement , CLAY__ELEMENT_CONFIG_TYPE_BORDER_CONTAINER ) ) {
Clay_LayoutElementHashMapItem * currentElementData = Clay__GetHashMapItem ( currentElement - > id ) ;
Clay_BoundingBox currentElementBoundingBox = currentElementData - > boundingBox ;
2024-12-18 22:35:18 +00:00
// Culling - Don't bother to generate render commands for rectangles entirely outside the screen - this won't stop their children from being rendered if they overflow
if ( ! Clay__ElementIsOffscreen ( & currentElementBoundingBox ) ) {
Clay_BorderElementConfig * borderConfig = Clay__FindElementConfigWithType ( currentElement , CLAY__ELEMENT_CONFIG_TYPE_BORDER_CONTAINER ) . borderElementConfig ;
2024-12-30 00:09:14 +00:00
Clay_RenderCommand renderCommand = {
2024-12-18 22:35:18 +00:00
. boundingBox = currentElementBoundingBox ,
. config = { . borderElementConfig = borderConfig } ,
. id = Clay__RehashWithNumber ( currentElement - > id , 4 ) ,
. commandType = CLAY_RENDER_COMMAND_TYPE_BORDER ,
} ;
Clay__AddRenderCommand ( renderCommand ) ;
if ( borderConfig - > betweenChildren . width > 0 & & borderConfig - > betweenChildren . color . a > 0 ) {
Clay_RectangleElementConfig * rectangleConfig = Clay__StoreRectangleElementConfig ( CLAY__INIT ( Clay_RectangleElementConfig ) { . color = borderConfig - > betweenChildren . color } ) ;
Clay_Vector2 borderOffset = { ( float ) layoutConfig - > padding . x , ( float ) layoutConfig - > padding . y } ;
if ( layoutConfig - > layoutDirection = = CLAY_LEFT_TO_RIGHT ) {
2024-12-30 00:09:14 +00:00
for ( int i = 0 ; i < currentElement - > childrenOrTextContent . children . length ; + + i ) {
Clay_LayoutElement * childElement = Clay_LayoutElementArray_Get ( & Clay__layoutElements , currentElement - > childrenOrTextContent . children . elements [ i ] ) ;
2024-12-18 22:35:18 +00:00
if ( i > 0 ) {
Clay__AddRenderCommand ( CLAY__INIT ( Clay_RenderCommand ) {
. boundingBox = { currentElementBoundingBox . x + borderOffset . x + scrollOffset . x , currentElementBoundingBox . y + scrollOffset . y , ( float ) borderConfig - > betweenChildren . width , currentElement - > dimensions . height } ,
. config = { rectangleConfig } ,
. id = Clay__RehashWithNumber ( currentElement - > id , 5 + i ) ,
. commandType = CLAY_RENDER_COMMAND_TYPE_RECTANGLE ,
} ) ;
}
borderOffset . x + = ( childElement - > dimensions . width + ( float ) layoutConfig - > childGap / 2 ) ;
2024-08-23 04:05:23 +00:00
}
2024-12-18 22:35:18 +00:00
} else {
2024-12-30 00:09:14 +00:00
for ( int i = 0 ; i < currentElement - > childrenOrTextContent . children . length ; + + i ) {
Clay_LayoutElement * childElement = Clay_LayoutElementArray_Get ( & Clay__layoutElements , currentElement - > childrenOrTextContent . children . elements [ i ] ) ;
2024-12-18 22:35:18 +00:00
if ( i > 0 ) {
Clay__AddRenderCommand ( CLAY__INIT ( Clay_RenderCommand ) {
. boundingBox = { currentElementBoundingBox . x + scrollOffset . x , currentElementBoundingBox . y + borderOffset . y + scrollOffset . y , currentElement - > dimensions . width , ( float ) borderConfig - > betweenChildren . width } ,
. config = { rectangleConfig } ,
. id = Clay__RehashWithNumber ( currentElement - > id , 5 + i ) ,
. commandType = CLAY_RENDER_COMMAND_TYPE_RECTANGLE ,
} ) ;
}
borderOffset . y + = ( childElement - > dimensions . height + ( float ) layoutConfig - > childGap / 2 ) ;
2024-08-23 04:05:23 +00:00
}
}
}
}
}
2024-10-22 07:41:35 +00:00
// This exists because the scissor needs to end _after_ borders between elements
if ( closeScrollElement ) {
2024-12-01 08:05:56 +00:00
Clay__AddRenderCommand ( CLAY__INIT ( Clay_RenderCommand ) {
2024-10-22 07:41:35 +00:00
. id = Clay__RehashWithNumber ( currentElement - > id , 11 ) ,
. commandType = CLAY_RENDER_COMMAND_TYPE_SCISSOR_END ,
} ) ;
}
2024-08-23 04:05:23 +00:00
dfsBuffer . length - - ;
continue ;
}
// Add children to the DFS buffer
2024-10-22 07:41:35 +00:00
if ( ! Clay__ElementHasConfig ( currentElement , CLAY__ELEMENT_CONFIG_TYPE_TEXT ) ) {
2024-12-30 00:09:14 +00:00
dfsBuffer . length + = currentElement - > childrenOrTextContent . children . length ;
for ( int i = 0 ; i < currentElement - > childrenOrTextContent . children . length ; + + i ) {
Clay_LayoutElement * childElement = Clay_LayoutElementArray_Get ( & Clay__layoutElements , currentElement - > childrenOrTextContent . children . elements [ i ] ) ;
2024-08-23 04:05:23 +00:00
// Alignment along non layout axis
if ( layoutConfig - > layoutDirection = = CLAY_LEFT_TO_RIGHT ) {
currentElementTreeNode - > nextChildOffset . y = currentElement - > layoutConfig - > padding . y ;
float whiteSpaceAroundChild = currentElement - > dimensions . height - ( float ) currentElement - > layoutConfig - > padding . y * 2 - childElement - > dimensions . height ;
switch ( layoutConfig - > childAlignment . y ) {
case CLAY_ALIGN_Y_TOP : break ;
case CLAY_ALIGN_Y_CENTER : currentElementTreeNode - > nextChildOffset . y + = whiteSpaceAroundChild / 2 ; break ;
case CLAY_ALIGN_Y_BOTTOM : currentElementTreeNode - > nextChildOffset . y + = whiteSpaceAroundChild ; break ;
}
} else {
currentElementTreeNode - > nextChildOffset . x = currentElement - > layoutConfig - > padding . x ;
float whiteSpaceAroundChild = currentElement - > dimensions . width - ( float ) currentElement - > layoutConfig - > padding . x * 2 - childElement - > dimensions . width ;
switch ( layoutConfig - > childAlignment . x ) {
case CLAY_ALIGN_X_LEFT : break ;
case CLAY_ALIGN_X_CENTER : currentElementTreeNode - > nextChildOffset . x + = whiteSpaceAroundChild / 2 ; break ;
case CLAY_ALIGN_X_RIGHT : currentElementTreeNode - > nextChildOffset . x + = whiteSpaceAroundChild ; break ;
}
}
2024-12-30 00:09:14 +00:00
Clay_Vector2 childPosition = {
2024-08-23 04:05:23 +00:00
currentElementTreeNode - > position . x + currentElementTreeNode - > nextChildOffset . x + scrollOffset . x ,
currentElementTreeNode - > position . y + currentElementTreeNode - > nextChildOffset . y + scrollOffset . y ,
} ;
// DFS buffer elements need to be added in reverse because stack traversal happens backwards
uint32_t newNodeIndex = dfsBuffer . length - 1 - i ;
2024-09-28 23:53:46 +00:00
dfsBuffer . internalArray [ newNodeIndex ] = CLAY__INIT ( Clay__LayoutElementTreeNode ) {
2024-08-23 04:05:23 +00:00
. layoutElement = childElement ,
2024-12-30 00:09:14 +00:00
. position = { childPosition . x , childPosition . y } ,
. nextChildOffset = { . x = ( float ) childElement - > layoutConfig - > padding . x , . y = ( float ) childElement - > layoutConfig - > padding . y } ,
2024-08-23 04:05:23 +00:00
} ;
Clay__treeNodeVisited . internalArray [ newNodeIndex ] = false ;
// Update parent offsets
if ( layoutConfig - > layoutDirection = = CLAY_LEFT_TO_RIGHT ) {
currentElementTreeNode - > nextChildOffset . x + = childElement - > dimensions . width + ( float ) layoutConfig - > childGap ;
} else {
currentElementTreeNode - > nextChildOffset . y + = childElement - > dimensions . height + ( float ) layoutConfig - > childGap ;
}
}
}
}
if ( root - > clipElementId ) {
2024-12-01 08:05:56 +00:00
Clay__AddRenderCommand ( CLAY__INIT ( Clay_RenderCommand ) { . id = Clay__RehashWithNumber ( rootElement - > id , 11 ) , . commandType = CLAY_RENDER_COMMAND_TYPE_SCISSOR_END } ) ;
2024-09-16 09:34:59 +00:00
}
}
}
2024-10-22 07:41:35 +00:00
void Clay__AttachId ( Clay_ElementId elementId ) {
2024-12-26 06:58:59 +00:00
if ( Clay__booleanWarnings . maxElementsExceeded ) {
2024-12-01 08:05:56 +00:00
return ;
}
2024-10-22 07:41:35 +00:00
Clay_LayoutElement * openLayoutElement = Clay__GetOpenLayoutElement ( ) ;
openLayoutElement - > id = elementId . id ;
Clay__AddHashMapItem ( elementId , openLayoutElement ) ;
Clay__StringArray_Add ( & Clay__layoutElementIdStrings , elementId . stringId ) ;
}
void Clay__AttachLayoutConfig ( Clay_LayoutConfig * config ) {
2024-12-26 06:58:59 +00:00
if ( Clay__booleanWarnings . maxElementsExceeded ) {
2024-12-01 08:05:56 +00:00
return ;
}
2024-10-22 07:41:35 +00:00
Clay__GetOpenLayoutElement ( ) - > layoutConfig = config ;
}
void Clay__AttachElementConfig ( Clay_ElementConfigUnion config , Clay__ElementConfigType type ) {
2024-12-26 06:58:59 +00:00
if ( Clay__booleanWarnings . maxElementsExceeded ) {
2024-12-01 08:05:56 +00:00
return ;
}
2024-10-22 07:41:35 +00:00
Clay_LayoutElement * openLayoutElement = Clay__GetOpenLayoutElement ( ) ;
openLayoutElement - > elementConfigs . length + + ;
Clay__ElementConfigArray_Add ( & Clay__elementConfigBuffer , CLAY__INIT ( Clay_ElementConfig ) { . type = type , . config = config } ) ;
}
2024-12-26 06:58:59 +00:00
Clay_LayoutConfig * Clay__StoreLayoutConfig ( Clay_LayoutConfig config ) { return Clay__booleanWarnings . maxElementsExceeded ? & CLAY_LAYOUT_DEFAULT : Clay__LayoutConfigArray_Add ( & Clay__layoutConfigs , config ) ; }
Clay_RectangleElementConfig * Clay__StoreRectangleElementConfig ( Clay_RectangleElementConfig config ) { return Clay__booleanWarnings . maxElementsExceeded ? & CLAY__RECTANGLE_ELEMENT_CONFIG_DEFAULT : Clay__RectangleElementConfigArray_Add ( & Clay__rectangleElementConfigs , config ) ; }
Clay_TextElementConfig * Clay__StoreTextElementConfig ( Clay_TextElementConfig config ) { return Clay__booleanWarnings . maxElementsExceeded ? & CLAY__TEXT_ELEMENT_CONFIG_DEFAULT : Clay__TextElementConfigArray_Add ( & Clay__textElementConfigs , config ) ; }
Clay_ImageElementConfig * Clay__StoreImageElementConfig ( Clay_ImageElementConfig config ) { return Clay__booleanWarnings . maxElementsExceeded ? & CLAY__IMAGE_ELEMENT_CONFIG_DEFAULT : Clay__ImageElementConfigArray_Add ( & Clay__imageElementConfigs , config ) ; }
Clay_FloatingElementConfig * Clay__StoreFloatingElementConfig ( Clay_FloatingElementConfig config ) { return Clay__booleanWarnings . maxElementsExceeded ? & CLAY__FLOATING_ELEMENT_CONFIG_DEFAULT : Clay__FloatingElementConfigArray_Add ( & Clay__floatingElementConfigs , config ) ; }
Clay_CustomElementConfig * Clay__StoreCustomElementConfig ( Clay_CustomElementConfig config ) { return Clay__booleanWarnings . maxElementsExceeded ? & CLAY__CUSTOM_ELEMENT_CONFIG_DEFAULT : Clay__CustomElementConfigArray_Add ( & Clay__customElementConfigs , config ) ; }
Clay_ScrollElementConfig * Clay__StoreScrollElementConfig ( Clay_ScrollElementConfig config ) { return Clay__booleanWarnings . maxElementsExceeded ? & CLAY__SCROLL_ELEMENT_CONFIG_DEFAULT : Clay__ScrollElementConfigArray_Add ( & Clay__scrollElementConfigs , config ) ; }
Clay_BorderElementConfig * Clay__StoreBorderElementConfig ( Clay_BorderElementConfig config ) { return Clay__booleanWarnings . maxElementsExceeded ? & CLAY__BORDER_ELEMENT_CONFIG_DEFAULT : Clay__BorderElementConfigArray_Add ( & Clay__borderElementConfigs , config ) ; }
2024-09-16 09:34:59 +00:00
# pragma region DebugTools
2024-12-30 00:09:14 +00:00
Clay_Color CLAY__DEBUGVIEW_COLOR_1 = { 58 , 56 , 52 , 255 } ;
Clay_Color CLAY__DEBUGVIEW_COLOR_2 = { 62 , 60 , 58 , 255 } ;
Clay_Color CLAY__DEBUGVIEW_COLOR_3 = { 141 , 133 , 135 , 255 } ;
Clay_Color CLAY__DEBUGVIEW_COLOR_4 = { 238 , 226 , 231 , 255 } ;
Clay_Color CLAY__DEBUGVIEW_COLOR_SELECTED_ROW = { 102 , 80 , 78 , 255 } ;
2024-09-16 09:34:59 +00:00
const int CLAY__DEBUGVIEW_ROW_HEIGHT = 30 ;
const int CLAY__DEBUGVIEW_OUTER_PADDING = 10 ;
const int CLAY__DEBUGVIEW_INDENT_WIDTH = 16 ;
2024-12-30 00:09:14 +00:00
Clay_TextElementConfig Clay__DebugView_TextNameConfig = { . textColor = { 238 , 226 , 231 , 255 } , . fontSize = 16 , . wrapMode = CLAY_TEXT_WRAP_NONE } ;
Clay_LayoutConfig Clay__DebugView_ScrollViewItemLayoutConfig = { 0 } ;
2024-09-16 09:34:59 +00:00
2024-12-30 00:09:14 +00:00
CLAY__TYPEDEF ( Clay__DebugElementConfigTypeLabelConfig , struct {
2024-10-22 07:41:35 +00:00
Clay_String label ;
Clay_Color color ;
2024-12-30 00:09:14 +00:00
} ) ;
2024-10-22 07:41:35 +00:00
Clay__DebugElementConfigTypeLabelConfig Clay__DebugGetElementConfigTypeLabel ( Clay__ElementConfigType type ) {
switch ( type ) {
2024-12-30 00:09:14 +00:00
case CLAY__ELEMENT_CONFIG_TYPE_RECTANGLE : return CLAY__INIT ( Clay__DebugElementConfigTypeLabelConfig ) { CLAY_STRING ( " Rectangle " ) , { 243 , 134 , 48 , 255 } } ;
case CLAY__ELEMENT_CONFIG_TYPE_TEXT : return CLAY__INIT ( Clay__DebugElementConfigTypeLabelConfig ) { CLAY_STRING ( " Text " ) , { 105 , 210 , 231 , 255 } } ;
case CLAY__ELEMENT_CONFIG_TYPE_IMAGE : return CLAY__INIT ( Clay__DebugElementConfigTypeLabelConfig ) { CLAY_STRING ( " Image " ) , { 121 , 189 , 154 , 255 } } ;
case CLAY__ELEMENT_CONFIG_TYPE_FLOATING_CONTAINER : return CLAY__INIT ( Clay__DebugElementConfigTypeLabelConfig ) { CLAY_STRING ( " Floating " ) , { 250 , 105 , 0 , 255 } } ;
case CLAY__ELEMENT_CONFIG_TYPE_SCROLL_CONTAINER : return CLAY__INIT ( Clay__DebugElementConfigTypeLabelConfig ) { CLAY_STRING ( " Scroll " ) , { 242 , 196 , 90 , 255 } } ;
case CLAY__ELEMENT_CONFIG_TYPE_BORDER_CONTAINER : return CLAY__INIT ( Clay__DebugElementConfigTypeLabelConfig ) { CLAY_STRING ( " Border " ) , { 108 , 91 , 123 , 255 } } ;
case CLAY__ELEMENT_CONFIG_TYPE_CUSTOM : return CLAY__INIT ( Clay__DebugElementConfigTypeLabelConfig ) { CLAY_STRING ( " Custom " ) , { 11 , 72 , 107 , 255 } } ;
default : break ;
2024-10-22 07:41:35 +00:00
}
2024-12-30 00:09:14 +00:00
return CLAY__INIT ( Clay__DebugElementConfigTypeLabelConfig ) { CLAY_STRING ( " Error " ) , { 0 , 0 , 0 , 255 } } ;
2024-10-22 07:41:35 +00:00
}
2024-12-30 00:09:14 +00:00
CLAY__TYPEDEF ( Clay__RenderDebugLayoutData , struct {
2024-09-16 09:34:59 +00:00
uint32_t rowCount ;
uint32_t selectedElementRowIndex ;
2024-12-30 00:09:14 +00:00
} ) ;
2024-09-16 09:34:59 +00:00
// Returns row count
Clay__RenderDebugLayoutData Clay__RenderDebugLayoutElementsList ( int32_t initialRootsLength , int32_t highlightedRowIndex ) {
Clay__int32_tArray dfsBuffer = Clay__reusableElementIndexBuffer ;
2024-09-28 23:53:46 +00:00
Clay__DebugView_ScrollViewItemLayoutConfig = CLAY__INIT ( Clay_LayoutConfig ) { . sizing = { . height = CLAY_SIZING_FIXED ( CLAY__DEBUGVIEW_ROW_HEIGHT ) } , . childGap = 6 , . childAlignment = { . y = CLAY_ALIGN_Y_CENTER } } ;
2024-12-30 00:09:14 +00:00
Clay__RenderDebugLayoutData layoutData = { 0 } ;
2024-09-16 09:34:59 +00:00
uint32_t highlightedElementId = 0 ;
for ( int rootIndex = 0 ; rootIndex < initialRootsLength ; + + rootIndex ) {
dfsBuffer . length = 0 ;
Clay__LayoutElementTreeRoot * root = Clay__LayoutElementTreeRootArray_Get ( & Clay__layoutElementTreeRoots , rootIndex ) ;
Clay__int32_tArray_Add ( & dfsBuffer , ( int32_t ) root - > layoutElementIndex ) ;
Clay__treeNodeVisited . internalArray [ 0 ] = false ;
if ( rootIndex > 0 ) {
2024-12-30 00:09:14 +00:00
CLAY ( CLAY_IDI ( " Clay__DebugView_EmptyRowOuter " , rootIndex ) , CLAY_LAYOUT ( { . sizing = { . width = CLAY_SIZING_GROW ( { 0 } ) } , . padding = { CLAY__DEBUGVIEW_INDENT_WIDTH / 2 , 0 } } ) ) {
CLAY ( CLAY_IDI ( " Clay__DebugView_EmptyRow " , rootIndex ) , CLAY_LAYOUT ( { . sizing = { . width = CLAY_SIZING_GROW ( { 0 } ) , . height = CLAY_SIZING_FIXED ( ( float ) CLAY__DEBUGVIEW_ROW_HEIGHT ) } } ) , CLAY_BORDER ( { . top = { . width = 1 , . color = CLAY__DEBUGVIEW_COLOR_3 } } ) ) { }
2024-09-28 03:43:29 +00:00
}
2024-09-16 09:34:59 +00:00
layoutData . rowCount + + ;
}
while ( dfsBuffer . length > 0 ) {
uint32_t currentElementIndex = Clay__int32_tArray_Get ( & dfsBuffer , ( int ) dfsBuffer . length - 1 ) ;
Clay_LayoutElement * currentElement = Clay_LayoutElementArray_Get ( & Clay__layoutElements , ( int ) currentElementIndex ) ;
if ( Clay__treeNodeVisited . internalArray [ dfsBuffer . length - 1 ] ) {
2024-12-30 00:09:14 +00:00
if ( ! Clay__ElementHasConfig ( currentElement , CLAY__ELEMENT_CONFIG_TYPE_TEXT ) & & currentElement - > childrenOrTextContent . children . length > 0 ) {
2024-10-22 07:41:35 +00:00
Clay__CloseElement ( ) ;
Clay__CloseElement ( ) ;
Clay__CloseElement ( ) ;
2024-09-16 09:34:59 +00:00
}
dfsBuffer . length - - ;
continue ;
}
if ( highlightedRowIndex = = layoutData . rowCount ) {
2024-10-22 20:44:39 +00:00
if ( Clay__pointerInfo . state = = CLAY_POINTER_DATA_PRESSED_THIS_FRAME ) {
2024-09-16 09:34:59 +00:00
Clay__debugSelectedElementId = currentElement - > id ;
}
highlightedElementId = currentElement - > id ;
}
2024-09-28 03:43:29 +00:00
2024-09-16 09:34:59 +00:00
Clay__treeNodeVisited . internalArray [ dfsBuffer . length - 1 ] = true ;
Clay_LayoutElementHashMapItem * currentElementData = Clay__GetHashMapItem ( currentElement - > id ) ;
2024-12-18 22:35:18 +00:00
bool offscreen = Clay__ElementIsOffscreen ( & currentElementData - > boundingBox ) ;
2024-09-16 09:34:59 +00:00
if ( Clay__debugSelectedElementId = = currentElement - > id ) {
layoutData . selectedElementRowIndex = layoutData . rowCount ;
}
2024-10-22 07:41:35 +00:00
CLAY ( CLAY_IDI ( " Clay__DebugView_ElementOuter " , currentElement - > id ) , Clay__AttachLayoutConfig ( & Clay__DebugView_ScrollViewItemLayoutConfig ) ) {
// Collapse icon / button
2024-12-30 00:09:14 +00:00
if ( ! ( Clay__ElementHasConfig ( currentElement , CLAY__ELEMENT_CONFIG_TYPE_TEXT ) | | currentElement - > childrenOrTextContent . children . length = = 0 ) ) {
2024-10-22 07:41:35 +00:00
CLAY ( CLAY_IDI ( " Clay__DebugView_CollapseElement " , currentElement - > id ) ,
CLAY_LAYOUT ( { . sizing = { CLAY_SIZING_FIXED ( 16 ) , CLAY_SIZING_FIXED ( 16 ) } , . childAlignment = { CLAY_ALIGN_X_CENTER , CLAY_ALIGN_Y_CENTER } } ) ,
CLAY_BORDER_OUTSIDE_RADIUS ( 1 , CLAY__DEBUGVIEW_COLOR_3 , 4 )
) {
CLAY_TEXT ( ( currentElementData & & currentElementData - > debugData - > collapsed ) ? CLAY_STRING ( " + " ) : CLAY_STRING ( " - " ) , CLAY_TEXT_CONFIG ( { . textColor = CLAY__DEBUGVIEW_COLOR_4 , . fontSize = 16 } ) ) ;
2024-09-28 03:43:29 +00:00
}
2024-10-22 07:41:35 +00:00
} else { // Square dot for empty containers
CLAY ( CLAY_LAYOUT ( { . sizing = { CLAY_SIZING_FIXED ( 16 ) , CLAY_SIZING_FIXED ( 16 ) } , . childAlignment = { CLAY_ALIGN_X_CENTER , CLAY_ALIGN_Y_CENTER } } ) ) {
CLAY ( CLAY_LAYOUT ( { . sizing = { CLAY_SIZING_FIXED ( 8 ) , CLAY_SIZING_FIXED ( 8 ) } } ) , CLAY_RECTANGLE ( { . color = CLAY__DEBUGVIEW_COLOR_3 , . cornerRadius = CLAY_CORNER_RADIUS ( 2 ) } ) ) { }
2024-09-28 03:43:29 +00:00
}
2024-09-16 09:34:59 +00:00
}
2024-10-22 07:41:35 +00:00
// Collisions and offscreen info
if ( currentElementData ) {
if ( currentElementData - > debugData - > collision ) {
CLAY ( CLAY_LAYOUT ( { . padding = { 8 , 2 } } ) , CLAY_BORDER_OUTSIDE_RADIUS ( 1 , ( CLAY__INIT ( Clay_Color ) { 177 , 147 , 8 , 255 } ) , 4 ) ) {
CLAY_TEXT ( CLAY_STRING ( " Duplicate ID " ) , CLAY_TEXT_CONFIG ( { . textColor = CLAY__DEBUGVIEW_COLOR_3 , . fontSize = 16 } ) ) ;
2024-09-16 09:34:59 +00:00
}
}
2024-10-22 07:41:35 +00:00
if ( offscreen ) {
CLAY ( CLAY_LAYOUT ( { . padding = { 8 , 2 } } ) , CLAY_BORDER_OUTSIDE_RADIUS ( 1 , CLAY__DEBUGVIEW_COLOR_3 , 4 ) ) {
CLAY_TEXT ( CLAY_STRING ( " Offscreen " ) , CLAY_TEXT_CONFIG ( { . textColor = CLAY__DEBUGVIEW_COLOR_3 , . fontSize = 16 } ) ) ;
}
2024-09-16 09:34:59 +00:00
}
2024-10-22 07:41:35 +00:00
}
Clay_String idString = Clay__layoutElementIdStrings . internalArray [ currentElementIndex ] ;
if ( idString . length > 0 ) {
CLAY_TEXT ( idString , offscreen ? CLAY_TEXT_CONFIG ( { . textColor = CLAY__DEBUGVIEW_COLOR_3 , . fontSize = 16 } ) : & Clay__DebugView_TextNameConfig ) ;
}
2024-12-30 00:09:14 +00:00
for ( uint32_t elementConfigIndex = 0 ; elementConfigIndex < currentElement - > elementConfigs . length ; + + elementConfigIndex ) {
2024-10-22 07:41:35 +00:00
Clay_ElementConfig * elementConfig = Clay__ElementConfigArraySlice_Get ( & currentElement - > elementConfigs , elementConfigIndex ) ;
Clay__DebugElementConfigTypeLabelConfig config = Clay__DebugGetElementConfigTypeLabel ( elementConfig - > type ) ;
Clay_Color backgroundColor = config . color ;
2024-09-16 09:34:59 +00:00
backgroundColor . a = 90 ;
2024-10-22 07:41:35 +00:00
CLAY ( CLAY_LAYOUT ( { . padding = { 8 , 2 } } ) , CLAY_RECTANGLE ( { . color = backgroundColor , . cornerRadius = CLAY_CORNER_RADIUS ( 4 ) } ) , CLAY_BORDER_OUTSIDE_RADIUS ( 1 , config . color , 4 ) ) {
CLAY_TEXT ( config . label , CLAY_TEXT_CONFIG ( { . textColor = offscreen ? CLAY__DEBUGVIEW_COLOR_3 : CLAY__DEBUGVIEW_COLOR_4 , . fontSize = 16 } ) ) ;
2024-09-28 03:43:29 +00:00
}
}
}
2024-09-16 09:34:59 +00:00
// Render the text contents below the element as a non-interactive row
2024-10-22 07:41:35 +00:00
if ( Clay__ElementHasConfig ( currentElement , CLAY__ELEMENT_CONFIG_TYPE_TEXT ) ) {
2024-09-16 09:34:59 +00:00
layoutData . rowCount + + ;
2024-12-30 00:09:14 +00:00
Clay__TextElementData * textElementData = currentElement - > childrenOrTextContent . textElementData ;
2024-10-22 07:41:35 +00:00
Clay_TextElementConfig * rawTextConfig = offscreen ? CLAY_TEXT_CONFIG ( { . textColor = CLAY__DEBUGVIEW_COLOR_3 , . fontSize = 16 } ) : & Clay__DebugView_TextNameConfig ;
2024-12-30 00:09:14 +00:00
CLAY ( CLAY_LAYOUT ( { . sizing = { . height = CLAY_SIZING_FIXED ( CLAY__DEBUGVIEW_ROW_HEIGHT ) } , . childAlignment = { . y = CLAY_ALIGN_Y_CENTER } } ) , CLAY_RECTANGLE ( { 0 } ) ) {
CLAY ( CLAY_LAYOUT ( { . sizing = { CLAY_SIZING_FIXED ( CLAY__DEBUGVIEW_INDENT_WIDTH + 16 ) , { 0 } } } ) ) { }
2024-10-22 07:41:35 +00:00
CLAY_TEXT ( CLAY_STRING ( " \" " ) , rawTextConfig ) ;
CLAY_TEXT ( textElementData - > text . length > 40 ? ( CLAY__INIT ( Clay_String ) { . length = 40 , . chars = textElementData - > text . chars } ) : textElementData - > text , rawTextConfig ) ;
if ( textElementData - > text . length > 40 ) {
CLAY_TEXT ( CLAY_STRING ( " ... " ) , rawTextConfig ) ;
2024-09-16 09:34:59 +00:00
}
2024-10-22 07:41:35 +00:00
CLAY_TEXT ( CLAY_STRING ( " \" " ) , rawTextConfig ) ;
2024-09-28 03:43:29 +00:00
}
2024-12-30 00:09:14 +00:00
} else if ( currentElement - > childrenOrTextContent . children . length > 0 ) {
2024-10-22 07:41:35 +00:00
Clay__OpenElement ( ) ;
2024-12-30 00:09:14 +00:00
CLAY_LAYOUT ( { . padding = { 8 , 0 } } ) ;
2024-10-22 07:41:35 +00:00
Clay__ElementPostConfiguration ( ) ;
Clay__OpenElement ( ) ;
CLAY_BORDER ( { . left = { . width = 1 , . color = CLAY__DEBUGVIEW_COLOR_3 } } ) ;
Clay__ElementPostConfiguration ( ) ;
2024-12-30 00:09:14 +00:00
CLAY ( CLAY_LAYOUT ( { . sizing = { CLAY_SIZING_FIXED ( CLAY__DEBUGVIEW_INDENT_WIDTH ) , { 0 } } , . childAlignment = { . x = CLAY_ALIGN_X_RIGHT } } ) ) { }
2024-10-22 07:41:35 +00:00
Clay__OpenElement ( ) ;
CLAY_LAYOUT ( { . layoutDirection = CLAY_TOP_TO_BOTTOM } ) ;
Clay__ElementPostConfiguration ( ) ;
2024-09-16 09:34:59 +00:00
}
layoutData . rowCount + + ;
2024-10-22 07:41:35 +00:00
if ( ! ( Clay__ElementHasConfig ( currentElement , CLAY__ELEMENT_CONFIG_TYPE_TEXT ) | | ( currentElementData & & currentElementData - > debugData - > collapsed ) ) ) {
2024-12-30 00:09:14 +00:00
for ( int i = currentElement - > childrenOrTextContent . children . length - 1 ; i > = 0 ; - - i ) {
Clay__int32_tArray_Add ( & dfsBuffer , currentElement - > childrenOrTextContent . children . elements [ i ] ) ;
2024-09-16 09:34:59 +00:00
Clay__treeNodeVisited . internalArray [ dfsBuffer . length - 1 ] = false ; // TODO needs to be ranged checked
}
}
}
}
2024-10-22 20:44:39 +00:00
if ( Clay__pointerInfo . state = = CLAY_POINTER_DATA_PRESSED_THIS_FRAME ) {
2024-10-22 07:41:35 +00:00
Clay_ElementId collapseButtonId = Clay__HashString ( CLAY_STRING ( " Clay__DebugView_CollapseElement " ) , 0 , 0 ) ;
2024-12-18 22:35:18 +00:00
for ( int i = ( int ) Clay__pointerOverIds . length - 1 ; i > = 0 ; i - - ) {
Clay_ElementId * elementId = Clay__ElementIdArray_Get ( & Clay__pointerOverIds , i ) ;
if ( elementId - > baseId = = collapseButtonId . baseId ) {
Clay_LayoutElementHashMapItem * highlightedItem = Clay__GetHashMapItem ( elementId - > offset ) ;
highlightedItem - > debugData - > collapsed = ! highlightedItem - > debugData - > collapsed ;
break ;
2024-09-16 09:34:59 +00:00
}
2024-08-23 04:05:23 +00:00
}
}
2024-09-16 09:34:59 +00:00
if ( highlightedElementId ) {
2024-12-30 00:09:14 +00:00
CLAY ( CLAY_ID ( " Clay__DebugView_ElementHighlight " ) , CLAY_LAYOUT ( { . sizing = { CLAY_SIZING_GROW ( { 0 } ) , CLAY_SIZING_GROW ( { 0 } ) } } ) , CLAY_FLOATING ( { . zIndex = 65535 , . parentId = highlightedElementId } ) ) {
CLAY ( CLAY_ID ( " Clay__DebugView_ElementHighlightRectangle " ) , CLAY_LAYOUT ( { . sizing = { CLAY_SIZING_GROW ( { 0 } ) , CLAY_SIZING_GROW ( { 0 } ) } } ) , CLAY_RECTANGLE ( { . color = Clay__debugViewHighlightColor } ) ) { }
2024-09-28 03:43:29 +00:00
}
2024-09-16 09:34:59 +00:00
}
return layoutData ;
2024-08-23 04:05:23 +00:00
}
2024-10-22 07:41:35 +00:00
void Clay__RenderDebugLayoutSizing ( Clay_SizingAxis sizing , Clay_TextElementConfig * infoTextConfig ) {
2024-09-16 09:34:59 +00:00
Clay_String sizingLabel = CLAY_STRING ( " GROW " ) ;
if ( sizing . type = = CLAY__SIZING_TYPE_FIT ) {
sizingLabel = CLAY_STRING ( " FIT " ) ;
} else if ( sizing . type = = CLAY__SIZING_TYPE_PERCENT ) {
sizingLabel = CLAY_STRING ( " PERCENT " ) ;
2024-10-22 07:41:35 +00:00
}
CLAY_TEXT ( sizingLabel , infoTextConfig ) ;
2024-09-16 09:34:59 +00:00
if ( sizing . type = = CLAY__SIZING_TYPE_GROW | | sizing . type = = CLAY__SIZING_TYPE_FIT ) {
2024-10-22 07:41:35 +00:00
CLAY_TEXT ( CLAY_STRING ( " ( " ) , infoTextConfig ) ;
2024-12-26 22:12:59 +00:00
if ( sizing . size . minMax . min ! = 0 ) {
2024-10-22 07:41:35 +00:00
CLAY_TEXT ( CLAY_STRING ( " min: " ) , infoTextConfig ) ;
2024-12-26 22:12:59 +00:00
CLAY_TEXT ( Clay__IntToString ( sizing . size . minMax . min ) , infoTextConfig ) ;
if ( sizing . size . minMax . max ! = CLAY__MAXFLOAT ) {
2024-10-22 07:41:35 +00:00
CLAY_TEXT ( CLAY_STRING ( " , " ) , infoTextConfig ) ;
2024-09-16 09:34:59 +00:00
}
}
2024-12-26 22:12:59 +00:00
if ( sizing . size . minMax . max ! = CLAY__MAXFLOAT ) {
2024-10-22 07:41:35 +00:00
CLAY_TEXT ( CLAY_STRING ( " max: " ) , infoTextConfig ) ;
2024-12-26 22:12:59 +00:00
CLAY_TEXT ( Clay__IntToString ( sizing . size . minMax . max ) , infoTextConfig ) ;
2024-09-16 09:34:59 +00:00
}
2024-10-22 07:41:35 +00:00
CLAY_TEXT ( CLAY_STRING ( " ) " ) , infoTextConfig ) ;
2024-09-16 09:34:59 +00:00
}
}
2024-10-22 07:41:35 +00:00
void Clay__RenderDebugViewElementConfigHeader ( Clay_String elementId , Clay__ElementConfigType type ) {
Clay__DebugElementConfigTypeLabelConfig config = Clay__DebugGetElementConfigTypeLabel ( type ) ;
Clay_Color backgroundColor = config . color ;
backgroundColor . a = 90 ;
2024-12-30 00:09:14 +00:00
CLAY ( CLAY_LAYOUT ( { . sizing = { CLAY_SIZING_GROW ( { 0 } ) , CLAY_SIZING_FIXED ( CLAY__DEBUGVIEW_ROW_HEIGHT + 8 ) } , . padding = { . x = CLAY__DEBUGVIEW_OUTER_PADDING } , . childAlignment = { . y = CLAY_ALIGN_Y_CENTER } } ) ) {
2024-10-22 07:41:35 +00:00
CLAY ( CLAY_LAYOUT ( { . padding = { 8 , 2 } } ) , CLAY_RECTANGLE ( { . color = backgroundColor , . cornerRadius = CLAY_CORNER_RADIUS ( 4 ) } ) , CLAY_BORDER_OUTSIDE_RADIUS ( 1 , config . color , 4 ) ) {
CLAY_TEXT ( config . label , CLAY_TEXT_CONFIG ( { . textColor = CLAY__DEBUGVIEW_COLOR_4 , . fontSize = 16 } ) ) ;
}
2024-12-30 00:09:14 +00:00
CLAY ( CLAY_LAYOUT ( { . sizing = { . width = CLAY_SIZING_GROW ( { 0 } ) } } ) ) { }
2024-10-22 07:41:35 +00:00
CLAY_TEXT ( elementId , CLAY_TEXT_CONFIG ( { . textColor = CLAY__DEBUGVIEW_COLOR_3 , . fontSize = 16 , . wrapMode = CLAY_TEXT_WRAP_NONE } ) ) ;
2024-09-28 03:43:29 +00:00
}
2024-09-16 09:34:59 +00:00
}
void Clay__RenderDebugViewColor ( Clay_Color color , Clay_TextElementConfig * textConfig ) {
2024-10-22 07:41:35 +00:00
CLAY ( CLAY_LAYOUT ( { . childAlignment = { . y = CLAY_ALIGN_Y_CENTER } } ) ) {
CLAY_TEXT ( CLAY_STRING ( " { r: " ) , textConfig ) ;
CLAY_TEXT ( Clay__IntToString ( color . r ) , textConfig ) ;
CLAY_TEXT ( CLAY_STRING ( " , g: " ) , textConfig ) ;
CLAY_TEXT ( Clay__IntToString ( color . g ) , textConfig ) ;
CLAY_TEXT ( CLAY_STRING ( " , b: " ) , textConfig ) ;
CLAY_TEXT ( Clay__IntToString ( color . b ) , textConfig ) ;
CLAY_TEXT ( CLAY_STRING ( " , a: " ) , textConfig ) ;
CLAY_TEXT ( Clay__IntToString ( color . a ) , textConfig ) ;
CLAY_TEXT ( CLAY_STRING ( " } " ) , textConfig ) ;
2024-12-30 00:09:14 +00:00
CLAY ( CLAY_LAYOUT ( { . sizing = { CLAY_SIZING_FIXED ( 10 ) , { 0 } } } ) ) { }
2024-10-22 07:41:35 +00:00
CLAY ( CLAY_LAYOUT ( { . sizing = { CLAY_SIZING_FIXED ( CLAY__DEBUGVIEW_ROW_HEIGHT - 8 ) , CLAY_SIZING_FIXED ( CLAY__DEBUGVIEW_ROW_HEIGHT - 8 ) } } ) , CLAY_BORDER_OUTSIDE_RADIUS ( 1 , CLAY__DEBUGVIEW_COLOR_4 , 4 ) ) {
CLAY ( CLAY_LAYOUT ( { . sizing = { CLAY_SIZING_FIXED ( CLAY__DEBUGVIEW_ROW_HEIGHT - 8 ) , CLAY_SIZING_FIXED ( CLAY__DEBUGVIEW_ROW_HEIGHT - 8 ) } } ) , CLAY_RECTANGLE ( { . color = color , . cornerRadius = CLAY_CORNER_RADIUS ( 4 ) } ) ) { }
2024-09-28 03:43:29 +00:00
}
}
2024-09-16 09:34:59 +00:00
}
void Clay__RenderDebugViewCornerRadius ( Clay_CornerRadius cornerRadius , Clay_TextElementConfig * textConfig ) {
2024-10-22 07:41:35 +00:00
CLAY ( CLAY_LAYOUT ( { . childAlignment = { . y = CLAY_ALIGN_Y_CENTER } } ) ) {
CLAY_TEXT ( CLAY_STRING ( " { topLeft: " ) , textConfig ) ;
CLAY_TEXT ( Clay__IntToString ( cornerRadius . topLeft ) , textConfig ) ;
CLAY_TEXT ( CLAY_STRING ( " , topRight: " ) , textConfig ) ;
CLAY_TEXT ( Clay__IntToString ( cornerRadius . topRight ) , textConfig ) ;
CLAY_TEXT ( CLAY_STRING ( " , bottomLeft: " ) , textConfig ) ;
CLAY_TEXT ( Clay__IntToString ( cornerRadius . bottomLeft ) , textConfig ) ;
CLAY_TEXT ( CLAY_STRING ( " , bottomRight: " ) , textConfig ) ;
CLAY_TEXT ( Clay__IntToString ( cornerRadius . bottomRight ) , textConfig ) ;
CLAY_TEXT ( CLAY_STRING ( " } " ) , textConfig ) ;
2024-09-28 03:43:29 +00:00
}
2024-09-16 09:34:59 +00:00
}
void Clay__RenderDebugViewBorder ( int index , Clay_Border border , Clay_TextElementConfig * textConfig ) {
2024-12-30 00:09:14 +00:00
( void ) index ;
2024-10-22 07:41:35 +00:00
CLAY ( CLAY_LAYOUT ( { . childAlignment = { . y = CLAY_ALIGN_Y_CENTER } } ) ) {
CLAY_TEXT ( CLAY_STRING ( " { width: " ) , textConfig ) ;
CLAY_TEXT ( Clay__IntToString ( border . width ) , textConfig ) ;
CLAY_TEXT ( CLAY_STRING ( " , color: " ) , textConfig ) ;
2024-09-16 09:34:59 +00:00
Clay__RenderDebugViewColor ( border . color , textConfig ) ;
2024-10-22 07:41:35 +00:00
CLAY_TEXT ( CLAY_STRING ( " } " ) , textConfig ) ;
}
}
2024-10-22 20:44:39 +00:00
void HandleDebugViewCloseButtonInteraction ( Clay_ElementId elementId , Clay_PointerData pointerInfo , intptr_t userData ) {
2024-12-30 00:09:14 +00:00
( void ) elementId ; ( void ) pointerInfo ; ( void ) userData ;
2024-10-22 20:44:39 +00:00
if ( pointerInfo . state = = CLAY_POINTER_DATA_PRESSED_THIS_FRAME ) {
2024-10-22 07:41:35 +00:00
Clay__debugModeEnabled = false ;
2024-09-28 03:43:29 +00:00
}
2024-09-16 09:34:59 +00:00
}
2024-12-30 00:09:14 +00:00
void Clay__RenderDebugView ( void ) {
2024-10-22 07:41:35 +00:00
Clay_ElementId closeButtonId = Clay__HashString ( CLAY_STRING ( " Clay__DebugViewTopHeaderCloseButtonOuter " ) , 0 , 0 ) ;
2024-10-22 20:44:39 +00:00
if ( Clay__pointerInfo . state = = CLAY_POINTER_DATA_PRESSED_THIS_FRAME ) {
2024-12-30 00:09:14 +00:00
for ( uint32_t i = 0 ; i < Clay__pointerOverIds . length ; + + i ) {
2024-09-16 09:34:59 +00:00
Clay_ElementId * elementId = Clay__ElementIdArray_Get ( & Clay__pointerOverIds , i ) ;
if ( elementId - > id = = closeButtonId . id ) {
Clay__debugModeEnabled = false ;
return ;
}
}
}
uint32_t initialRootsLength = Clay__layoutElementTreeRoots . length ;
uint32_t initialElementsLength = Clay__layoutElements . length ;
2024-10-22 07:41:35 +00:00
Clay_TextElementConfig * infoTextConfig = CLAY_TEXT_CONFIG ( { . textColor = CLAY__DEBUGVIEW_COLOR_4 , . fontSize = 16 , . wrapMode = CLAY_TEXT_WRAP_NONE } ) ;
Clay_TextElementConfig * infoTitleConfig = CLAY_TEXT_CONFIG ( { . textColor = CLAY__DEBUGVIEW_COLOR_3 , . fontSize = 16 , . wrapMode = CLAY_TEXT_WRAP_NONE } ) ;
Clay_ElementId scrollId = Clay__HashString ( CLAY_STRING ( " Clay__DebugViewOuterScrollPane " ) , 0 , 0 ) ;
2024-09-16 09:34:59 +00:00
float scrollYOffset = 0 ;
2024-12-30 00:09:14 +00:00
for ( uint32_t i = 0 ; i < Clay__scrollContainerDatas . length ; + + i ) {
2024-09-16 09:34:59 +00:00
Clay__ScrollContainerDataInternal * scrollContainerData = Clay__ScrollContainerDataInternalArray_Get ( & Clay__scrollContainerDatas , i ) ;
2024-10-22 07:41:35 +00:00
if ( scrollContainerData - > elementId = = scrollId . id ) {
2024-12-18 22:35:18 +00:00
if ( ! Clay__externalScrollHandlingEnabled ) {
scrollYOffset = scrollContainerData - > scrollPosition . y ;
}
2024-09-16 09:34:59 +00:00
break ;
}
}
2024-12-26 06:58:59 +00:00
int32_t highlightedRow = Clay__pointerInfo . position . y < Clay__layoutDimensions . height - 300
? ( int32_t ) ( ( Clay__pointerInfo . position . y - scrollYOffset ) / ( float ) CLAY__DEBUGVIEW_ROW_HEIGHT ) - 1
: - 1 ;
2024-09-16 09:34:59 +00:00
if ( Clay__pointerInfo . position . x < Clay__layoutDimensions . width - ( float ) Clay__debugViewWidth ) {
highlightedRow = - 1 ;
}
2024-12-30 00:09:14 +00:00
Clay__RenderDebugLayoutData layoutData = { 0 } ;
2024-10-22 07:41:35 +00:00
CLAY ( CLAY_ID ( " Clay__DebugView " ) ,
CLAY_FLOATING ( { . parentId = Clay__HashString ( CLAY_STRING ( " Clay__RootContainer " ) , 0 , 0 ) . id , . attachment = { . element = CLAY_ATTACH_POINT_LEFT_CENTER , . parent = CLAY_ATTACH_POINT_RIGHT_CENTER } } ) ,
CLAY_LAYOUT ( { . sizing = { CLAY_SIZING_FIXED ( ( float ) Clay__debugViewWidth ) , CLAY_SIZING_FIXED ( Clay__layoutDimensions . height ) } , . layoutDirection = CLAY_TOP_TO_BOTTOM } ) ,
CLAY_BORDER ( { . bottom = { . width = 1 , . color = CLAY__DEBUGVIEW_COLOR_3 } } )
) {
2024-12-30 00:09:14 +00:00
CLAY ( CLAY_LAYOUT ( { . sizing = { CLAY_SIZING_GROW ( { 0 } ) , CLAY_SIZING_FIXED ( CLAY__DEBUGVIEW_ROW_HEIGHT ) } , . padding = { CLAY__DEBUGVIEW_OUTER_PADDING , 0 } , . childAlignment = { . y = CLAY_ALIGN_Y_CENTER } } ) , CLAY_RECTANGLE ( { . color = CLAY__DEBUGVIEW_COLOR_2 } ) ) {
2024-10-22 07:41:35 +00:00
CLAY_TEXT ( CLAY_STRING ( " Clay Debug Tools " ) , infoTextConfig ) ;
2024-12-30 00:09:14 +00:00
CLAY ( CLAY_LAYOUT ( { . sizing = { CLAY_SIZING_GROW ( { 0 } ) , { 0 } } } ) ) { }
2024-10-22 07:41:35 +00:00
// Close button
CLAY ( CLAY_BORDER_OUTSIDE_RADIUS ( 1 , ( CLAY__INIT ( Clay_Color ) { 217 , 91 , 67 , 255 } ) , 4 ) ,
CLAY_LAYOUT ( { . sizing = { CLAY_SIZING_FIXED ( CLAY__DEBUGVIEW_ROW_HEIGHT - 10 ) , CLAY_SIZING_FIXED ( CLAY__DEBUGVIEW_ROW_HEIGHT - 10 ) } , . childAlignment = { CLAY_ALIGN_X_CENTER , CLAY_ALIGN_Y_CENTER } } ) ,
CLAY_RECTANGLE ( { . color = { 217 , 91 , 67 , 80 } } ) ,
Clay_OnHover ( HandleDebugViewCloseButtonInteraction , 0 )
) {
CLAY_TEXT ( CLAY_STRING ( " x " ) , CLAY_TEXT_CONFIG ( { . textColor = CLAY__DEBUGVIEW_COLOR_4 , . fontSize = 16 } ) ) ;
}
}
2024-12-30 00:09:14 +00:00
CLAY ( CLAY_LAYOUT ( { . sizing = { CLAY_SIZING_GROW ( { 0 } ) , CLAY_SIZING_FIXED ( 1 ) } } ) , CLAY_RECTANGLE ( { . color = CLAY__DEBUGVIEW_COLOR_3 } ) ) { }
CLAY ( Clay__AttachId ( scrollId ) , CLAY_LAYOUT ( { . sizing = { CLAY_SIZING_GROW ( { 0 } ) , CLAY_SIZING_GROW ( { 0 } ) } } ) , CLAY_SCROLL ( { . horizontal = true , . vertical = true } ) ) {
CLAY ( CLAY_LAYOUT ( { . sizing = { CLAY_SIZING_GROW ( { 0 } ) , CLAY_SIZING_GROW ( { 0 } ) } , . layoutDirection = CLAY_TOP_TO_BOTTOM } ) , CLAY_RECTANGLE ( { . color = ( ( initialElementsLength + initialRootsLength ) & 1 ) = = 0 ? CLAY__DEBUGVIEW_COLOR_2 : CLAY__DEBUGVIEW_COLOR_1 } ) ) {
2024-10-22 07:41:35 +00:00
Clay_ElementId panelContentsId = Clay__HashString ( CLAY_STRING ( " Clay__DebugViewPaneOuter " ) , 0 , 0 ) ;
// Element list
2024-12-30 00:09:14 +00:00
CLAY ( Clay__AttachId ( panelContentsId ) , CLAY_LAYOUT ( { . sizing = { CLAY_SIZING_GROW ( { 0 } ) , CLAY_SIZING_GROW ( { 0 } ) } } ) , CLAY_FLOATING ( { . pointerCaptureMode = CLAY_POINTER_CAPTURE_MODE_PASSTHROUGH } ) ) {
CLAY ( CLAY_LAYOUT ( { . sizing = { CLAY_SIZING_GROW ( { 0 } ) , CLAY_SIZING_GROW ( { 0 } ) } , . padding = { . x = CLAY__DEBUGVIEW_OUTER_PADDING } , . layoutDirection = CLAY_TOP_TO_BOTTOM } ) ) {
2024-10-22 07:41:35 +00:00
layoutData = Clay__RenderDebugLayoutElementsList ( ( int32_t ) initialRootsLength , highlightedRow ) ;
}
}
float contentWidth = Clay__GetHashMapItem ( panelContentsId . id ) - > layoutElement - > dimensions . width ;
2024-12-30 00:09:14 +00:00
CLAY ( CLAY_LAYOUT ( { . sizing = { CLAY_SIZING_FIXED ( contentWidth ) , { 0 } } , . layoutDirection = CLAY_TOP_TO_BOTTOM } ) ) { }
2024-10-22 07:41:35 +00:00
for ( uint32_t i = 0 ; i < layoutData . rowCount ; i + + ) {
Clay_Color rowColor = ( i & 1 ) = = 0 ? CLAY__DEBUGVIEW_COLOR_2 : CLAY__DEBUGVIEW_COLOR_1 ;
if ( i = = layoutData . selectedElementRowIndex ) {
rowColor = CLAY__DEBUGVIEW_COLOR_SELECTED_ROW ;
}
if ( i = = highlightedRow ) {
rowColor . r * = 1.25f ;
rowColor . g * = 1.25f ;
rowColor . b * = 1.25f ;
2024-09-28 03:43:29 +00:00
}
2024-12-30 00:09:14 +00:00
CLAY ( CLAY_LAYOUT ( { . sizing = { CLAY_SIZING_GROW ( { 0 } ) , CLAY_SIZING_FIXED ( CLAY__DEBUGVIEW_ROW_HEIGHT ) } , . layoutDirection = CLAY_TOP_TO_BOTTOM } ) , CLAY_RECTANGLE ( { . color = rowColor } ) ) { }
2024-09-28 03:43:29 +00:00
}
}
2024-10-22 07:41:35 +00:00
}
2024-12-30 00:09:14 +00:00
CLAY ( CLAY_LAYOUT ( { . sizing = { . width = CLAY_SIZING_GROW ( { 0 } ) , . height = CLAY_SIZING_FIXED ( 1 ) } } ) , CLAY_RECTANGLE ( { . color = CLAY__DEBUGVIEW_COLOR_3 } ) ) { }
2024-10-22 07:41:35 +00:00
if ( Clay__debugSelectedElementId ! = 0 ) {
Clay_LayoutElementHashMapItem * selectedItem = Clay__GetHashMapItem ( Clay__debugSelectedElementId ) ;
CLAY (
CLAY_SCROLL ( { . vertical = true } ) ,
2024-12-30 00:09:14 +00:00
CLAY_LAYOUT ( { . sizing = { CLAY_SIZING_GROW ( { 0 } ) , CLAY_SIZING_FIXED ( 300 ) } , . layoutDirection = CLAY_TOP_TO_BOTTOM } ) ,
2024-10-22 07:41:35 +00:00
CLAY_RECTANGLE ( { . color = CLAY__DEBUGVIEW_COLOR_2 } ) ,
CLAY_BORDER ( { . betweenChildren = { . width = 1 , . color = CLAY__DEBUGVIEW_COLOR_3 } } )
) {
2024-12-30 00:09:14 +00:00
CLAY ( CLAY_LAYOUT ( { . sizing = { CLAY_SIZING_GROW ( { 0 } ) , CLAY_SIZING_FIXED ( CLAY__DEBUGVIEW_ROW_HEIGHT + 8 ) } , . padding = { CLAY__DEBUGVIEW_OUTER_PADDING , 0 } , . childAlignment = { . y = CLAY_ALIGN_Y_CENTER } } ) ) {
2024-10-22 07:41:35 +00:00
CLAY_TEXT ( CLAY_STRING ( " Layout Config " ) , infoTextConfig ) ;
2024-12-30 00:09:14 +00:00
CLAY ( CLAY_LAYOUT ( { . sizing = { CLAY_SIZING_GROW ( { 0 } ) , { 0 } } } ) ) { }
2024-10-22 07:41:35 +00:00
if ( selectedItem - > elementId . stringId . length ! = 0 ) {
CLAY_TEXT ( selectedItem - > elementId . stringId , infoTitleConfig ) ;
if ( selectedItem - > elementId . offset ! = 0 ) {
CLAY_TEXT ( CLAY_STRING ( " ( " ) , infoTitleConfig ) ;
CLAY_TEXT ( Clay__IntToString ( selectedItem - > elementId . offset ) , infoTitleConfig ) ;
CLAY_TEXT ( CLAY_STRING ( " ) " ) , infoTitleConfig ) ;
2024-09-28 03:43:29 +00:00
}
}
2024-10-22 07:41:35 +00:00
}
// Clay_LayoutConfig debug info
CLAY ( CLAY_LAYOUT ( { . padding = { CLAY__DEBUGVIEW_OUTER_PADDING , 8 } , . childGap = 8 , . layoutDirection = CLAY_TOP_TO_BOTTOM } ) ) {
// .boundingBox
CLAY_TEXT ( CLAY_STRING ( " Bounding Box " ) , infoTitleConfig ) ;
2024-12-30 00:09:14 +00:00
CLAY ( CLAY_LAYOUT ( { 0 } ) ) {
2024-10-22 07:41:35 +00:00
CLAY_TEXT ( CLAY_STRING ( " { x: " ) , infoTextConfig ) ;
CLAY_TEXT ( Clay__IntToString ( selectedItem - > boundingBox . x ) , infoTextConfig ) ;
CLAY_TEXT ( CLAY_STRING ( " , y: " ) , infoTextConfig ) ;
CLAY_TEXT ( Clay__IntToString ( selectedItem - > boundingBox . y ) , infoTextConfig ) ;
CLAY_TEXT ( CLAY_STRING ( " , width: " ) , infoTextConfig ) ;
CLAY_TEXT ( Clay__IntToString ( selectedItem - > boundingBox . width ) , infoTextConfig ) ;
CLAY_TEXT ( CLAY_STRING ( " , height: " ) , infoTextConfig ) ;
CLAY_TEXT ( Clay__IntToString ( selectedItem - > boundingBox . height ) , infoTextConfig ) ;
CLAY_TEXT ( CLAY_STRING ( " } " ) , infoTextConfig ) ;
}
// .layoutDirection
CLAY_TEXT ( CLAY_STRING ( " Layout Direction " ) , infoTitleConfig ) ;
Clay_LayoutConfig * layoutConfig = selectedItem - > layoutElement - > layoutConfig ;
CLAY_TEXT ( layoutConfig - > layoutDirection = = CLAY_TOP_TO_BOTTOM ? CLAY_STRING ( " TOP_TO_BOTTOM " ) : CLAY_STRING ( " LEFT_TO_RIGHT " ) , infoTextConfig ) ;
// .sizing
CLAY_TEXT ( CLAY_STRING ( " Sizing " ) , infoTitleConfig ) ;
2024-12-30 00:09:14 +00:00
CLAY ( CLAY_LAYOUT ( { 0 } ) ) {
2024-10-22 07:41:35 +00:00
CLAY_TEXT ( CLAY_STRING ( " width: " ) , infoTextConfig ) ;
Clay__RenderDebugLayoutSizing ( layoutConfig - > sizing . width , infoTextConfig ) ;
}
2024-12-30 00:09:14 +00:00
CLAY ( CLAY_LAYOUT ( { 0 } ) ) {
2024-10-22 07:41:35 +00:00
CLAY_TEXT ( CLAY_STRING ( " height: " ) , infoTextConfig ) ;
Clay__RenderDebugLayoutSizing ( layoutConfig - > sizing . height , infoTextConfig ) ;
}
// .padding
CLAY_TEXT ( CLAY_STRING ( " Padding " ) , infoTitleConfig ) ;
CLAY ( CLAY_ID ( " Clay__DebugViewElementInfoPadding " ) ) {
CLAY_TEXT ( CLAY_STRING ( " { x: " ) , infoTextConfig ) ;
CLAY_TEXT ( Clay__IntToString ( layoutConfig - > padding . x ) , infoTextConfig ) ;
CLAY_TEXT ( CLAY_STRING ( " , y: " ) , infoTextConfig ) ;
CLAY_TEXT ( Clay__IntToString ( layoutConfig - > padding . y ) , infoTextConfig ) ;
CLAY_TEXT ( CLAY_STRING ( " } " ) , infoTextConfig ) ;
}
// .childGap
CLAY_TEXT ( CLAY_STRING ( " Child Gap " ) , infoTitleConfig ) ;
CLAY_TEXT ( Clay__IntToString ( layoutConfig - > childGap ) , infoTextConfig ) ;
// .childAlignment
CLAY_TEXT ( CLAY_STRING ( " Child Alignment " ) , infoTitleConfig ) ;
2024-12-30 00:09:14 +00:00
CLAY ( CLAY_LAYOUT ( { 0 } ) ) {
2024-10-22 07:41:35 +00:00
CLAY_TEXT ( CLAY_STRING ( " { x: " ) , infoTextConfig ) ;
Clay_String alignX = CLAY_STRING ( " LEFT " ) ;
if ( layoutConfig - > childAlignment . x = = CLAY_ALIGN_X_CENTER ) {
alignX = CLAY_STRING ( " CENTER " ) ;
} else if ( layoutConfig - > childAlignment . x = = CLAY_ALIGN_X_RIGHT ) {
alignX = CLAY_STRING ( " RIGHT " ) ;
2024-09-16 09:34:59 +00:00
}
2024-10-22 07:41:35 +00:00
CLAY_TEXT ( alignX , infoTextConfig ) ;
CLAY_TEXT ( CLAY_STRING ( " , y: " ) , infoTextConfig ) ;
Clay_String alignY = CLAY_STRING ( " TOP " ) ;
if ( layoutConfig - > childAlignment . y = = CLAY_ALIGN_Y_CENTER ) {
alignY = CLAY_STRING ( " CENTER " ) ;
} else if ( layoutConfig - > childAlignment . y = = CLAY_ALIGN_Y_BOTTOM ) {
alignY = CLAY_STRING ( " BOTTOM " ) ;
2024-09-16 09:34:59 +00:00
}
2024-10-22 07:41:35 +00:00
CLAY_TEXT ( alignY , infoTextConfig ) ;
CLAY_TEXT ( CLAY_STRING ( " } " ) , infoTextConfig ) ;
2024-09-16 09:34:59 +00:00
}
2024-09-28 03:43:29 +00:00
}
2024-12-30 00:09:14 +00:00
for ( uint32_t elementConfigIndex = 0 ; elementConfigIndex < selectedItem - > layoutElement - > elementConfigs . length ; + + elementConfigIndex ) {
2024-10-22 07:41:35 +00:00
Clay_ElementConfig * elementConfig = Clay__ElementConfigArraySlice_Get ( & selectedItem - > layoutElement - > elementConfigs , elementConfigIndex ) ;
Clay__RenderDebugViewElementConfigHeader ( selectedItem - > elementId . stringId , elementConfig - > type ) ;
switch ( elementConfig - > type ) {
case CLAY__ELEMENT_CONFIG_TYPE_RECTANGLE : {
Clay_RectangleElementConfig * rectangleConfig = elementConfig - > config . rectangleElementConfig ;
CLAY ( CLAY_LAYOUT ( { . padding = { CLAY__DEBUGVIEW_OUTER_PADDING , 8 } , . childGap = 8 , . layoutDirection = CLAY_TOP_TO_BOTTOM } ) ) {
// .color
CLAY_TEXT ( CLAY_STRING ( " Color " ) , infoTitleConfig ) ;
Clay__RenderDebugViewColor ( rectangleConfig - > color , infoTextConfig ) ;
// .cornerRadius
CLAY_TEXT ( CLAY_STRING ( " Corner Radius " ) , infoTitleConfig ) ;
Clay__RenderDebugViewCornerRadius ( rectangleConfig - > cornerRadius , infoTextConfig ) ;
2024-09-16 09:34:59 +00:00
}
2024-10-22 07:41:35 +00:00
break ;
2024-09-28 03:43:29 +00:00
}
2024-10-22 07:41:35 +00:00
case CLAY__ELEMENT_CONFIG_TYPE_TEXT : {
Clay_TextElementConfig * textConfig = elementConfig - > config . textElementConfig ;
CLAY ( CLAY_LAYOUT ( { . padding = { CLAY__DEBUGVIEW_OUTER_PADDING , 8 } , . childGap = 8 , . layoutDirection = CLAY_TOP_TO_BOTTOM } ) ) {
// .fontSize
CLAY_TEXT ( CLAY_STRING ( " Font Size " ) , infoTitleConfig ) ;
CLAY_TEXT ( Clay__IntToString ( textConfig - > fontSize ) , infoTextConfig ) ;
// .fontId
CLAY_TEXT ( CLAY_STRING ( " Font ID " ) , infoTitleConfig ) ;
CLAY_TEXT ( Clay__IntToString ( textConfig - > fontId ) , infoTextConfig ) ;
// .lineHeight
CLAY_TEXT ( CLAY_STRING ( " Line Height " ) , infoTitleConfig ) ;
CLAY_TEXT ( textConfig - > lineHeight = = 0 ? CLAY_STRING ( " auto " ) : Clay__IntToString ( textConfig - > lineHeight ) , infoTextConfig ) ;
// .letterSpacing
CLAY_TEXT ( CLAY_STRING ( " Letter Spacing " ) , infoTitleConfig ) ;
CLAY_TEXT ( Clay__IntToString ( textConfig - > letterSpacing ) , infoTextConfig ) ;
// .lineSpacing
CLAY_TEXT ( CLAY_STRING ( " Wrap Mode " ) , infoTitleConfig ) ;
Clay_String wrapMode = CLAY_STRING ( " WORDS " ) ;
if ( textConfig - > wrapMode = = CLAY_TEXT_WRAP_NONE ) {
wrapMode = CLAY_STRING ( " NONE " ) ;
} else if ( textConfig - > wrapMode = = CLAY_TEXT_WRAP_NEWLINES ) {
wrapMode = CLAY_STRING ( " NEWLINES " ) ;
2024-09-16 09:34:59 +00:00
}
2024-10-22 07:41:35 +00:00
CLAY_TEXT ( wrapMode , infoTextConfig ) ;
// .textColor
CLAY_TEXT ( CLAY_STRING ( " Text Color " ) , infoTitleConfig ) ;
Clay__RenderDebugViewColor ( textConfig - > textColor , infoTextConfig ) ;
2024-09-28 03:43:29 +00:00
}
2024-10-22 07:41:35 +00:00
break ;
2024-09-28 03:43:29 +00:00
}
2024-10-22 07:41:35 +00:00
case CLAY__ELEMENT_CONFIG_TYPE_IMAGE : {
Clay_ImageElementConfig * imageConfig = elementConfig - > config . imageElementConfig ;
CLAY ( CLAY_ID ( " Clay__DebugViewElementInfoImageBody " ) , CLAY_LAYOUT ( { . padding = { CLAY__DEBUGVIEW_OUTER_PADDING , 8 } , . childGap = 8 , . layoutDirection = CLAY_TOP_TO_BOTTOM } ) ) {
// .sourceDimensions
CLAY_TEXT ( CLAY_STRING ( " Source Dimensions " ) , infoTitleConfig ) ;
CLAY ( CLAY_ID ( " Clay__DebugViewElementInfoImageDimensions " ) ) {
CLAY_TEXT ( CLAY_STRING ( " { width: " ) , infoTextConfig ) ;
CLAY_TEXT ( Clay__IntToString ( imageConfig - > sourceDimensions . width ) , infoTextConfig ) ;
CLAY_TEXT ( CLAY_STRING ( " , height: " ) , infoTextConfig ) ;
CLAY_TEXT ( Clay__IntToString ( imageConfig - > sourceDimensions . height ) , infoTextConfig ) ;
CLAY_TEXT ( CLAY_STRING ( " } " ) , infoTextConfig ) ;
2024-09-28 03:43:29 +00:00
}
2024-10-22 07:41:35 +00:00
// Image Preview
CLAY_TEXT ( CLAY_STRING ( " Preview " ) , infoTitleConfig ) ;
2024-12-30 00:09:14 +00:00
CLAY ( CLAY_LAYOUT ( { . sizing = { CLAY_SIZING_GROW ( { . max = imageConfig - > sourceDimensions . width } ) , { 0 } } } ) , Clay__AttachElementConfig ( CLAY__INIT ( Clay_ElementConfigUnion ) { . imageElementConfig = imageConfig } , CLAY__ELEMENT_CONFIG_TYPE_IMAGE ) ) { }
2024-09-16 09:34:59 +00:00
}
2024-10-22 07:41:35 +00:00
break ;
}
case CLAY__ELEMENT_CONFIG_TYPE_SCROLL_CONTAINER : {
Clay_ScrollElementConfig * scrollConfig = elementConfig - > config . scrollElementConfig ;
CLAY ( CLAY_LAYOUT ( { . padding = { CLAY__DEBUGVIEW_OUTER_PADDING , 8 } , . childGap = 8 , . layoutDirection = CLAY_TOP_TO_BOTTOM } ) ) {
// .vertical
CLAY_TEXT ( CLAY_STRING ( " Vertical " ) , infoTitleConfig ) ;
CLAY_TEXT ( scrollConfig - > vertical ? CLAY_STRING ( " true " ) : CLAY_STRING ( " false " ) , infoTextConfig ) ;
// .horizontal
CLAY_TEXT ( CLAY_STRING ( " Horizontal " ) , infoTitleConfig ) ;
CLAY_TEXT ( scrollConfig - > horizontal ? CLAY_STRING ( " true " ) : CLAY_STRING ( " false " ) , infoTextConfig ) ;
2024-09-16 09:34:59 +00:00
}
2024-10-22 07:41:35 +00:00
break ;
}
case CLAY__ELEMENT_CONFIG_TYPE_FLOATING_CONTAINER : {
Clay_FloatingElementConfig * floatingConfig = elementConfig - > config . floatingElementConfig ;
CLAY ( CLAY_LAYOUT ( { . padding = { CLAY__DEBUGVIEW_OUTER_PADDING , 8 } , . childGap = 8 , . layoutDirection = CLAY_TOP_TO_BOTTOM } ) ) {
// .offset
CLAY_TEXT ( CLAY_STRING ( " Offset " ) , infoTitleConfig ) ;
2024-12-30 00:09:14 +00:00
CLAY ( CLAY_LAYOUT ( { 0 } ) ) {
2024-10-22 07:41:35 +00:00
CLAY_TEXT ( CLAY_STRING ( " { x: " ) , infoTextConfig ) ;
CLAY_TEXT ( Clay__IntToString ( floatingConfig - > offset . x ) , infoTextConfig ) ;
CLAY_TEXT ( CLAY_STRING ( " , y: " ) , infoTextConfig ) ;
CLAY_TEXT ( Clay__IntToString ( floatingConfig - > offset . y ) , infoTextConfig ) ;
CLAY_TEXT ( CLAY_STRING ( " } " ) , infoTextConfig ) ;
2024-09-28 03:43:29 +00:00
}
2024-10-22 07:41:35 +00:00
// .expand
CLAY_TEXT ( CLAY_STRING ( " Expand " ) , infoTitleConfig ) ;
2024-12-30 00:09:14 +00:00
CLAY ( CLAY_LAYOUT ( { 0 } ) ) {
2024-10-22 07:41:35 +00:00
CLAY_TEXT ( CLAY_STRING ( " { width: " ) , infoTextConfig ) ;
CLAY_TEXT ( Clay__IntToString ( floatingConfig - > expand . width ) , infoTextConfig ) ;
CLAY_TEXT ( CLAY_STRING ( " , height: " ) , infoTextConfig ) ;
CLAY_TEXT ( Clay__IntToString ( floatingConfig - > expand . height ) , infoTextConfig ) ;
CLAY_TEXT ( CLAY_STRING ( " } " ) , infoTextConfig ) ;
2024-09-28 03:43:29 +00:00
}
2024-10-22 07:41:35 +00:00
// .zIndex
CLAY_TEXT ( CLAY_STRING ( " z-index " ) , infoTitleConfig ) ;
CLAY_TEXT ( Clay__IntToString ( floatingConfig - > zIndex ) , infoTextConfig ) ;
// .parentId
CLAY_TEXT ( CLAY_STRING ( " Parent " ) , infoTitleConfig ) ;
Clay_LayoutElementHashMapItem * hashItem = Clay__GetHashMapItem ( floatingConfig - > parentId ) ;
CLAY_TEXT ( hashItem - > elementId . stringId , infoTextConfig ) ;
2024-09-16 09:34:59 +00:00
}
2024-10-22 07:41:35 +00:00
break ;
}
case CLAY__ELEMENT_CONFIG_TYPE_BORDER_CONTAINER : {
Clay_BorderElementConfig * borderConfig = elementConfig - > config . borderElementConfig ;
CLAY ( CLAY_ID ( " Clay__DebugViewElementInfoBorderBody " ) , CLAY_LAYOUT ( { . padding = { CLAY__DEBUGVIEW_OUTER_PADDING , 8 } , . childGap = 8 , . layoutDirection = CLAY_TOP_TO_BOTTOM } ) ) {
// .left
CLAY_TEXT ( CLAY_STRING ( " Left Border " ) , infoTitleConfig ) ;
Clay__RenderDebugViewBorder ( 1 , borderConfig - > left , infoTextConfig ) ;
// .right
CLAY_TEXT ( CLAY_STRING ( " Right Border " ) , infoTitleConfig ) ;
Clay__RenderDebugViewBorder ( 2 , borderConfig - > right , infoTextConfig ) ;
// .top
CLAY_TEXT ( CLAY_STRING ( " Top Border " ) , infoTitleConfig ) ;
Clay__RenderDebugViewBorder ( 3 , borderConfig - > top , infoTextConfig ) ;
// .bottom
CLAY_TEXT ( CLAY_STRING ( " Bottom Border " ) , infoTitleConfig ) ;
Clay__RenderDebugViewBorder ( 4 , borderConfig - > bottom , infoTextConfig ) ;
// .betweenChildren
CLAY_TEXT ( CLAY_STRING ( " Border Between Children " ) , infoTitleConfig ) ;
Clay__RenderDebugViewBorder ( 5 , borderConfig - > betweenChildren , infoTextConfig ) ;
// .cornerRadius
CLAY_TEXT ( CLAY_STRING ( " Corner Radius " ) , infoTitleConfig ) ;
Clay__RenderDebugViewCornerRadius ( borderConfig - > cornerRadius , infoTextConfig ) ;
2024-09-16 09:34:59 +00:00
}
2024-10-22 07:41:35 +00:00
break ;
}
2024-12-30 00:09:14 +00:00
case CLAY__ELEMENT_CONFIG_TYPE_CUSTOM :
default : break ;
2024-09-28 03:43:29 +00:00
}
}
2024-10-07 05:56:13 +00:00
}
2024-10-22 07:41:35 +00:00
} else {
2024-12-30 00:09:14 +00:00
CLAY ( CLAY_ID ( " Clay__DebugViewWarningsScrollPane " ) , CLAY_LAYOUT ( { . sizing = { CLAY_SIZING_GROW ( { 0 } ) , CLAY_SIZING_FIXED ( 300 ) } , . childGap = 6 , . layoutDirection = CLAY_TOP_TO_BOTTOM } ) , CLAY_SCROLL ( { . horizontal = true , . vertical = true } ) , CLAY_RECTANGLE ( { . color = CLAY__DEBUGVIEW_COLOR_2 } ) ) {
2024-10-22 07:41:35 +00:00
Clay_TextElementConfig * warningConfig = CLAY_TEXT_CONFIG ( { . textColor = CLAY__DEBUGVIEW_COLOR_4 , . fontSize = 16 , . wrapMode = CLAY_TEXT_WRAP_NONE } ) ;
2024-12-30 00:09:14 +00:00
CLAY ( CLAY_ID ( " Clay__DebugViewWarningItemHeader " ) , CLAY_LAYOUT ( { . sizing = { . height = CLAY_SIZING_FIXED ( CLAY__DEBUGVIEW_ROW_HEIGHT ) } , . padding = { CLAY__DEBUGVIEW_OUTER_PADDING , 0 } , . childGap = 8 , . childAlignment = { . y = CLAY_ALIGN_Y_CENTER } } ) ) {
2024-10-22 07:41:35 +00:00
CLAY_TEXT ( CLAY_STRING ( " Warnings " ) , warningConfig ) ;
}
2024-12-30 00:09:14 +00:00
CLAY ( CLAY_ID ( " Clay__DebugViewWarningsTopBorder " ) , CLAY_LAYOUT ( { . sizing = { . width = CLAY_SIZING_GROW ( { 0 } ) , . height = CLAY_SIZING_FIXED ( 1 ) } } ) , CLAY_RECTANGLE ( { . color = { 200 , 200 , 200 , 255 } } ) ) { }
2024-10-22 07:41:35 +00:00
int previousWarningsLength = ( int ) Clay_warnings . length ;
for ( int i = 0 ; i < previousWarningsLength ; i + + ) {
Clay__Warning warning = Clay_warnings . internalArray [ i ] ;
2024-12-30 00:09:14 +00:00
CLAY ( CLAY_IDI ( " Clay__DebugViewWarningItem " , i ) , CLAY_LAYOUT ( { . sizing = { . height = CLAY_SIZING_FIXED ( CLAY__DEBUGVIEW_ROW_HEIGHT ) } , . padding = { CLAY__DEBUGVIEW_OUTER_PADDING , 0 } , . childGap = 8 , . childAlignment = { . y = CLAY_ALIGN_Y_CENTER } } ) ) {
2024-10-22 07:41:35 +00:00
CLAY_TEXT ( warning . baseMessage , warningConfig ) ;
if ( warning . dynamicMessage . length > 0 ) {
CLAY_TEXT ( warning . dynamicMessage , warningConfig ) ;
2024-09-16 09:34:59 +00:00
}
2024-09-28 03:43:29 +00:00
}
}
2024-09-16 09:34:59 +00:00
}
2024-09-28 03:43:29 +00:00
}
}
2024-09-16 09:34:59 +00:00
}
# pragma endregion
2024-08-23 04:05:23 +00:00
// PUBLIC API FROM HERE ---------------------------------------
CLAY_WASM_EXPORT ( " Clay_MinMemorySize " )
2024-12-26 22:12:59 +00:00
uint32_t Clay_MinMemorySize ( void ) {
2024-12-30 00:09:14 +00:00
Clay_Arena fakeArena = { . capacity = SIZE_MAX } ;
2024-08-23 04:05:23 +00:00
Clay__InitializePersistentMemory ( & fakeArena ) ;
Clay__InitializeEphemeralMemory ( & fakeArena ) ;
return fakeArena . nextAllocation ;
}
CLAY_WASM_EXPORT ( " Clay_CreateArenaWithCapacityAndMemory " )
Clay_Arena Clay_CreateArenaWithCapacityAndMemory ( uint32_t capacity , void * offset ) {
2024-12-30 00:09:14 +00:00
Clay_Arena arena = {
2024-08-23 04:05:23 +00:00
. capacity = capacity ,
. memory = ( char * ) offset
} ;
return arena ;
}
# ifndef CLAY_WASM
void Clay_SetMeasureTextFunction ( Clay_Dimensions ( * measureTextFunction ) ( Clay_String * text , Clay_TextElementConfig * config ) ) {
Clay__MeasureText = measureTextFunction ;
}
2024-12-18 22:35:18 +00:00
void Clay_SetQueryScrollOffsetFunction ( Clay_Vector2 ( * queryScrollOffsetFunction ) ( uint32_t elementId ) ) {
Clay__QueryScrollOffset = queryScrollOffsetFunction ;
}
# endif
2024-09-16 09:34:59 +00:00
CLAY_WASM_EXPORT ( " Clay_SetLayoutDimensions " )
void Clay_SetLayoutDimensions ( Clay_Dimensions dimensions ) {
Clay__layoutDimensions = dimensions ;
}
CLAY_WASM_EXPORT ( " Clay_SetPointerState " )
void Clay_SetPointerState ( Clay_Vector2 position , bool isPointerDown ) {
2024-12-26 06:58:59 +00:00
if ( Clay__booleanWarnings . maxElementsExceeded ) {
2024-12-01 08:05:56 +00:00
return ;
}
2024-09-16 09:34:59 +00:00
Clay__pointerInfo . position = position ;
2024-08-23 04:05:23 +00:00
Clay__pointerOverIds . length = 0 ;
2024-09-16 09:34:59 +00:00
Clay__int32_tArray dfsBuffer = Clay__layoutElementChildrenBuffer ;
2024-12-20 17:36:34 +00:00
for ( int rootIndex = Clay__layoutElementTreeRoots . length - 1 ; rootIndex > = 0 ; - - rootIndex ) {
2024-08-23 04:05:23 +00:00
dfsBuffer . length = 0 ;
Clay__LayoutElementTreeRoot * root = Clay__LayoutElementTreeRootArray_Get ( & Clay__layoutElementTreeRoots , rootIndex ) ;
2024-09-16 09:34:59 +00:00
Clay__int32_tArray_Add ( & dfsBuffer , ( int32_t ) root - > layoutElementIndex ) ;
2024-08-23 04:05:23 +00:00
Clay__treeNodeVisited . internalArray [ 0 ] = false ;
2024-12-20 17:36:34 +00:00
bool found = false ;
2024-08-23 04:05:23 +00:00
while ( dfsBuffer . length > 0 ) {
if ( Clay__treeNodeVisited . internalArray [ dfsBuffer . length - 1 ] ) {
dfsBuffer . length - - ;
continue ;
}
Clay__treeNodeVisited . internalArray [ dfsBuffer . length - 1 ] = true ;
2024-09-16 09:34:59 +00:00
Clay_LayoutElement * currentElement = Clay_LayoutElementArray_Get ( & Clay__layoutElements , Clay__int32_tArray_Get ( & dfsBuffer , ( int ) dfsBuffer . length - 1 ) ) ;
2024-10-22 07:41:35 +00:00
Clay_LayoutElementHashMapItem * mapItem = Clay__GetHashMapItem ( currentElement - > id ) ; // TODO think of a way around this, maybe the fact that it's essentially a binary tree limits the cost, but the worst case is not great
2024-12-18 22:35:18 +00:00
Clay_BoundingBox elementBox = mapItem - > boundingBox ;
elementBox . x - = root - > pointerOffset . x ;
elementBox . y - = root - > pointerOffset . y ;
if ( mapItem ) {
if ( ( Clay__PointIsInsideRect ( position , elementBox ) ) ) {
if ( mapItem - > onHoverFunction ) {
mapItem - > onHoverFunction ( mapItem - > elementId , Clay__pointerInfo , mapItem - > hoverFunctionUserData ) ;
}
Clay__ElementIdArray_Add ( & Clay__pointerOverIds , mapItem - > elementId ) ;
2024-12-20 17:36:34 +00:00
found = true ;
2024-10-22 07:41:35 +00:00
}
if ( Clay__ElementHasConfig ( currentElement , CLAY__ELEMENT_CONFIG_TYPE_TEXT ) ) {
2024-08-23 04:05:23 +00:00
dfsBuffer . length - - ;
continue ;
}
2024-12-30 00:09:14 +00:00
for ( int i = currentElement - > childrenOrTextContent . children . length - 1 ; i > = 0 ; - - i ) {
Clay__int32_tArray_Add ( & dfsBuffer , currentElement - > childrenOrTextContent . children . elements [ i ] ) ;
2024-08-23 04:05:23 +00:00
Clay__treeNodeVisited . internalArray [ dfsBuffer . length - 1 ] = false ; // TODO needs to be ranged checked
}
} else {
dfsBuffer . length - - ;
}
}
2024-12-20 17:36:34 +00:00
Clay_LayoutElement * rootElement = Clay_LayoutElementArray_Get ( & Clay__layoutElements , root - > layoutElementIndex ) ;
if ( found & & Clay__ElementHasConfig ( rootElement , CLAY__ELEMENT_CONFIG_TYPE_FLOATING_CONTAINER ) & &
Clay__FindElementConfigWithType ( rootElement , CLAY__ELEMENT_CONFIG_TYPE_FLOATING_CONTAINER ) . floatingElementConfig - > pointerCaptureMode = = CLAY_POINTER_CAPTURE_MODE_CAPTURE ) {
break ;
}
2024-08-23 04:05:23 +00:00
}
2024-09-16 09:34:59 +00:00
if ( isPointerDown ) {
2024-10-22 20:44:39 +00:00
if ( Clay__pointerInfo . state = = CLAY_POINTER_DATA_PRESSED_THIS_FRAME ) {
Clay__pointerInfo . state = CLAY_POINTER_DATA_PRESSED ;
} else if ( Clay__pointerInfo . state ! = CLAY_POINTER_DATA_PRESSED ) {
Clay__pointerInfo . state = CLAY_POINTER_DATA_PRESSED_THIS_FRAME ;
2024-09-16 09:34:59 +00:00
}
} else {
2024-10-22 20:44:39 +00:00
if ( Clay__pointerInfo . state = = CLAY_POINTER_DATA_RELEASED_THIS_FRAME ) {
Clay__pointerInfo . state = CLAY_POINTER_DATA_RELEASED ;
} else if ( Clay__pointerInfo . state ! = CLAY_POINTER_DATA_RELEASED ) {
Clay__pointerInfo . state = CLAY_POINTER_DATA_RELEASED_THIS_FRAME ;
2024-09-16 09:34:59 +00:00
}
}
2024-08-23 04:05:23 +00:00
}
CLAY_WASM_EXPORT ( " Clay_Initialize " )
2024-12-26 06:58:59 +00:00
void Clay_Initialize ( Clay_Arena arena , Clay_Dimensions layoutDimensions , Clay_ErrorHandler errorHandler ) {
2024-08-23 04:05:23 +00:00
Clay__internalArena = arena ;
Clay__InitializePersistentMemory ( & Clay__internalArena ) ;
Clay__InitializeEphemeralMemory ( & Clay__internalArena ) ;
2024-12-30 00:09:14 +00:00
for ( uint32_t i = 0 ; i < Clay__layoutElementsHashMap . capacity ; + + i ) {
2024-08-23 04:05:23 +00:00
Clay__layoutElementsHashMap . internalArray [ i ] = - 1 ;
}
2024-12-30 00:09:14 +00:00
for ( uint32_t i = 0 ; i < Clay__measureTextHashMap . capacity ; + + i ) {
2024-09-25 01:30:36 +00:00
Clay__measureTextHashMap . internalArray [ i ] = 0 ;
}
2024-08-23 04:05:23 +00:00
Clay__measureTextHashMapInternal . length = 1 ; // Reserve the 0 value to mean "no next element"
2024-09-16 09:34:59 +00:00
Clay__layoutDimensions = layoutDimensions ;
2025-01-02 20:59:09 +00:00
if ( errorHandler . errorHandlerFunction ) {
Clay__errorHandler = errorHandler ;
}
2024-08-23 04:05:23 +00:00
}
CLAY_WASM_EXPORT ( " Clay_UpdateScrollContainers " )
2024-09-17 23:39:14 +00:00
void Clay_UpdateScrollContainers ( bool enableDragScrolling , Clay_Vector2 scrollDelta , float deltaTime ) {
2024-10-22 20:44:39 +00:00
bool isPointerActive = enableDragScrolling & & ( Clay__pointerInfo . state = = CLAY_POINTER_DATA_PRESSED | | Clay__pointerInfo . state = = CLAY_POINTER_DATA_PRESSED_THIS_FRAME ) ;
2024-08-23 04:05:23 +00:00
// Don't apply scroll events to ancestors of the inner element
int32_t highestPriorityElementIndex = - 1 ;
2024-08-26 07:05:43 +00:00
Clay__ScrollContainerDataInternal * highestPriorityScrollData = CLAY__NULL ;
2024-12-30 00:09:14 +00:00
for ( uint32_t i = 0 ; i < Clay__scrollContainerDatas . length ; i + + ) {
2024-08-26 07:05:43 +00:00
Clay__ScrollContainerDataInternal * scrollData = Clay__ScrollContainerDataInternalArray_Get ( & Clay__scrollContainerDatas , i ) ;
2024-08-23 04:05:23 +00:00
if ( ! scrollData - > openThisFrame ) {
2024-08-26 07:05:43 +00:00
Clay__ScrollContainerDataInternalArray_RemoveSwapback ( & Clay__scrollContainerDatas , i ) ;
2024-08-23 04:05:23 +00:00
continue ;
}
scrollData - > openThisFrame = false ;
Clay_LayoutElementHashMapItem * hashMapItem = Clay__GetHashMapItem ( scrollData - > elementId ) ;
// Element isn't rendered this frame but scroll offset has been retained
if ( ! hashMapItem ) {
2024-08-26 07:05:43 +00:00
Clay__ScrollContainerDataInternalArray_RemoveSwapback ( & Clay__scrollContainerDatas , i ) ;
2024-08-23 04:05:23 +00:00
continue ;
}
// Touch / click is released
if ( ! isPointerActive & & scrollData - > pointerScrollActive ) {
float xDiff = scrollData - > scrollPosition . x - scrollData - > scrollOrigin . x ;
if ( xDiff < - 10 | | xDiff > 10 ) {
scrollData - > scrollMomentum . x = ( scrollData - > scrollPosition . x - scrollData - > scrollOrigin . x ) / ( scrollData - > momentumTime * 25 ) ;
}
float yDiff = scrollData - > scrollPosition . y - scrollData - > scrollOrigin . y ;
if ( yDiff < - 10 | | yDiff > 10 ) {
scrollData - > scrollMomentum . y = ( scrollData - > scrollPosition . y - scrollData - > scrollOrigin . y ) / ( scrollData - > momentumTime * 25 ) ;
}
scrollData - > pointerScrollActive = false ;
2024-09-28 23:53:46 +00:00
scrollData - > pointerOrigin = CLAY__INIT ( Clay_Vector2 ) { 0 , 0 } ;
scrollData - > scrollOrigin = CLAY__INIT ( Clay_Vector2 ) { 0 , 0 } ;
2024-08-23 04:05:23 +00:00
scrollData - > momentumTime = 0 ;
}
// Apply existing momentum
scrollData - > scrollPosition . x + = scrollData - > scrollMomentum . x ;
scrollData - > scrollMomentum . x * = 0.95f ;
bool scrollOccurred = scrollDelta . x ! = 0 | | scrollDelta . y ! = 0 ;
if ( ( scrollData - > scrollMomentum . x > - 0.1f & & scrollData - > scrollMomentum . x < 0.1f ) | | scrollOccurred ) {
scrollData - > scrollMomentum . x = 0 ;
}
2024-10-04 22:59:29 +00:00
scrollData - > scrollPosition . x = CLAY__MIN ( CLAY__MAX ( scrollData - > scrollPosition . x , - ( CLAY__MAX ( scrollData - > contentSize . width - scrollData - > layoutElement - > dimensions . width , 0 ) ) ) , 0 ) ;
2024-08-23 04:05:23 +00:00
scrollData - > scrollPosition . y + = scrollData - > scrollMomentum . y ;
scrollData - > scrollMomentum . y * = 0.95f ;
if ( ( scrollData - > scrollMomentum . y > - 0.1f & & scrollData - > scrollMomentum . y < 0.1f ) | | scrollOccurred ) {
scrollData - > scrollMomentum . y = 0 ;
}
2024-10-04 22:59:29 +00:00
scrollData - > scrollPosition . y = CLAY__MIN ( CLAY__MAX ( scrollData - > scrollPosition . y , - ( CLAY__MAX ( scrollData - > contentSize . height - scrollData - > layoutElement - > dimensions . height , 0 ) ) ) , 0 ) ;
2024-08-23 04:05:23 +00:00
2024-12-30 00:09:14 +00:00
for ( uint32_t j = 0 ; j < Clay__pointerOverIds . length ; + + j ) { // TODO n & m are small here but this being n*m gives me the creeps
2024-09-16 09:34:59 +00:00
if ( scrollData - > layoutElement - > id = = Clay__ElementIdArray_Get ( & Clay__pointerOverIds , j ) - > id ) {
2024-08-23 04:05:23 +00:00
highestPriorityElementIndex = j ;
highestPriorityScrollData = scrollData ;
}
}
}
if ( highestPriorityElementIndex > - 1 & & highestPriorityScrollData ) {
Clay_LayoutElement * scrollElement = highestPriorityScrollData - > layoutElement ;
2024-10-22 07:41:35 +00:00
Clay_ScrollElementConfig * scrollConfig = Clay__FindElementConfigWithType ( scrollElement , CLAY__ELEMENT_CONFIG_TYPE_SCROLL_CONTAINER ) . scrollElementConfig ;
bool canScrollVertically = scrollConfig - > vertical & & highestPriorityScrollData - > contentSize . height > scrollElement - > dimensions . height ;
bool canScrollHorizontally = scrollConfig - > horizontal & & highestPriorityScrollData - > contentSize . width > scrollElement - > dimensions . width ;
2024-08-23 04:05:23 +00:00
// Handle wheel scroll
if ( canScrollVertically ) {
highestPriorityScrollData - > scrollPosition . y = highestPriorityScrollData - > scrollPosition . y + scrollDelta . y * 10 ;
}
if ( canScrollHorizontally ) {
highestPriorityScrollData - > scrollPosition . x = highestPriorityScrollData - > scrollPosition . x + scrollDelta . x * 10 ;
}
// Handle click / touch scroll
if ( isPointerActive ) {
2024-09-28 23:53:46 +00:00
highestPriorityScrollData - > scrollMomentum = CLAY__INIT ( Clay_Vector2 ) { 0 } ;
2024-08-23 04:05:23 +00:00
if ( ! highestPriorityScrollData - > pointerScrollActive ) {
2024-09-16 09:34:59 +00:00
highestPriorityScrollData - > pointerOrigin = Clay__pointerInfo . position ;
2024-08-23 04:05:23 +00:00
highestPriorityScrollData - > scrollOrigin = highestPriorityScrollData - > scrollPosition ;
highestPriorityScrollData - > pointerScrollActive = true ;
} else {
float scrollDeltaX = 0 , scrollDeltaY = 0 ;
if ( canScrollHorizontally ) {
float oldXScrollPosition = highestPriorityScrollData - > scrollPosition . x ;
2024-09-16 09:34:59 +00:00
highestPriorityScrollData - > scrollPosition . x = highestPriorityScrollData - > scrollOrigin . x + ( Clay__pointerInfo . position . x - highestPriorityScrollData - > pointerOrigin . x ) ;
2024-08-23 04:05:23 +00:00
highestPriorityScrollData - > scrollPosition . x = CLAY__MAX ( CLAY__MIN ( highestPriorityScrollData - > scrollPosition . x , 0 ) , - ( highestPriorityScrollData - > contentSize . width - highestPriorityScrollData - > boundingBox . width ) ) ;
scrollDeltaX = highestPriorityScrollData - > scrollPosition . x - oldXScrollPosition ;
}
if ( canScrollVertically ) {
float oldYScrollPosition = highestPriorityScrollData - > scrollPosition . y ;
2024-09-16 09:34:59 +00:00
highestPriorityScrollData - > scrollPosition . y = highestPriorityScrollData - > scrollOrigin . y + ( Clay__pointerInfo . position . y - highestPriorityScrollData - > pointerOrigin . y ) ;
2024-08-23 04:05:23 +00:00
highestPriorityScrollData - > scrollPosition . y = CLAY__MAX ( CLAY__MIN ( highestPriorityScrollData - > scrollPosition . y , 0 ) , - ( highestPriorityScrollData - > contentSize . height - highestPriorityScrollData - > boundingBox . height ) ) ;
2024-08-24 09:22:54 +00:00
scrollDeltaY = highestPriorityScrollData - > scrollPosition . y - oldYScrollPosition ;
2024-08-23 04:05:23 +00:00
}
2024-08-24 09:22:54 +00:00
if ( scrollDeltaX > - 0.1f & & scrollDeltaX < 0.1f & & scrollDeltaY > - 0.1f & & scrollDeltaY < 0.1f & & highestPriorityScrollData - > momentumTime > 0.15f ) {
2024-08-23 04:05:23 +00:00
highestPriorityScrollData - > momentumTime = 0 ;
2024-09-16 09:34:59 +00:00
highestPriorityScrollData - > pointerOrigin = Clay__pointerInfo . position ;
2024-08-23 04:05:23 +00:00
highestPriorityScrollData - > scrollOrigin = highestPriorityScrollData - > scrollPosition ;
} else {
highestPriorityScrollData - > momentumTime + = deltaTime ;
}
}
}
// Clamp any changes to scroll position to the maximum size of the contents
if ( canScrollVertically ) {
highestPriorityScrollData - > scrollPosition . y = CLAY__MAX ( CLAY__MIN ( highestPriorityScrollData - > scrollPosition . y , 0 ) , - ( highestPriorityScrollData - > contentSize . height - scrollElement - > dimensions . height ) ) ;
}
if ( canScrollHorizontally ) {
highestPriorityScrollData - > scrollPosition . x = CLAY__MAX ( CLAY__MIN ( highestPriorityScrollData - > scrollPosition . x , 0 ) , - ( highestPriorityScrollData - > contentSize . width - scrollElement - > dimensions . width ) ) ;
}
}
}
CLAY_WASM_EXPORT ( " Clay_BeginLayout " )
2024-12-26 22:12:59 +00:00
void Clay_BeginLayout ( void ) {
2024-08-23 04:05:23 +00:00
Clay__InitializeEphemeralMemory ( & Clay__internalArena ) ;
2024-09-16 09:34:59 +00:00
Clay__generation + + ;
Clay__dynamicElementIndex = 0 ;
2024-08-23 04:05:23 +00:00
// Set up the root container that covers the entire window
2024-12-30 00:09:14 +00:00
Clay_Dimensions rootDimensions = { Clay__layoutDimensions . width , Clay__layoutDimensions . height } ;
2024-09-16 09:34:59 +00:00
if ( Clay__debugModeEnabled ) {
rootDimensions . width - = ( float ) Clay__debugViewWidth ;
}
2024-12-26 06:58:59 +00:00
Clay__booleanWarnings . maxElementsExceeded = false ;
Clay__booleanWarnings . maxTextMeasureCacheExceeded = false ;
Clay__booleanWarnings . maxRenderCommandsExceeded = false ;
2024-10-22 07:41:35 +00:00
Clay__OpenElement ( ) ;
CLAY_ID ( " Clay__RootContainer " ) ;
CLAY_LAYOUT ( { . sizing = { CLAY_SIZING_FIXED ( ( rootDimensions . width ) ) , CLAY_SIZING_FIXED ( rootDimensions . height ) } } ) ;
Clay__ElementPostConfiguration ( ) ;
Clay__int32_tArray_Add ( & Clay__openLayoutElementStack , 0 ) ;
Clay__LayoutElementTreeRootArray_Add ( & Clay__layoutElementTreeRoots , CLAY__INIT ( Clay__LayoutElementTreeRoot ) { . layoutElementIndex = 0 } ) ;
2024-08-23 04:05:23 +00:00
}
2024-12-30 00:09:14 +00:00
Clay_TextElementConfig Clay__DebugView_ErrorTextConfig = { . textColor = { 255 , 0 , 0 , 255 } , . fontSize = 16 , . wrapMode = CLAY_TEXT_WRAP_NONE } ;
2024-12-01 08:05:56 +00:00
2024-08-23 04:05:23 +00:00
CLAY_WASM_EXPORT ( " Clay_EndLayout " )
2024-12-26 22:12:59 +00:00
Clay_RenderCommandArray Clay_EndLayout ( void ) {
2024-10-22 07:41:35 +00:00
Clay__CloseElement ( ) ;
2024-09-16 09:34:59 +00:00
if ( Clay__debugModeEnabled ) {
Clay__warningsEnabled = false ;
Clay__RenderDebugView ( ) ;
Clay__warningsEnabled = true ;
}
2024-12-26 06:58:59 +00:00
if ( Clay__booleanWarnings . maxElementsExceeded ) {
2024-12-30 00:09:14 +00:00
Clay__AddRenderCommand ( CLAY__INIT ( Clay_RenderCommand ) { . boundingBox = { Clay__layoutDimensions . width / 2 - 59 * 4 , Clay__layoutDimensions . height / 2 , 0 , 0 } , . config = { . textElementConfig = & Clay__DebugView_ErrorTextConfig } , . text = CLAY_STRING ( " Clay Error: Layout elements exceeded Clay__maxElementCount " ) , . commandType = CLAY_RENDER_COMMAND_TYPE_TEXT } ) ;
2024-12-01 08:05:56 +00:00
} else {
Clay__CalculateFinalLayout ( ) ;
}
2024-08-23 04:05:23 +00:00
return Clay__renderCommands ;
}
2024-10-22 07:41:35 +00:00
CLAY_WASM_EXPORT ( " Clay_GetElementId " )
Clay_ElementId Clay_GetElementId ( Clay_String idString ) {
return Clay__HashString ( idString , 0 , 0 ) ;
}
2024-10-24 08:38:02 +00:00
CLAY_WASM_EXPORT ( " Clay_GetElementIdWithIndex " )
Clay_ElementId Clay_GetElementIdWithIndex ( Clay_String idString , uint32_t index ) {
return Clay__HashString ( idString , index , 0 ) ;
}
2024-12-26 22:12:59 +00:00
bool Clay_Hovered ( void ) {
2024-12-26 06:58:59 +00:00
if ( Clay__booleanWarnings . maxElementsExceeded ) {
2024-12-01 08:05:56 +00:00
return false ;
}
2024-10-22 07:41:35 +00:00
Clay_LayoutElement * openLayoutElement = Clay__GetOpenLayoutElement ( ) ;
// If the element has no id attached at this point, we need to generate one
if ( openLayoutElement - > id = = 0 ) {
Clay__GenerateIdForAnonymousElement ( openLayoutElement ) ;
}
2024-12-30 00:09:14 +00:00
for ( uint32_t i = 0 ; i < Clay__pointerOverIds . length ; + + i ) {
2024-10-22 07:41:35 +00:00
if ( Clay__ElementIdArray_Get ( & Clay__pointerOverIds , i ) - > id = = openLayoutElement - > id ) {
return true ;
}
}
return false ;
}
2024-10-22 20:44:39 +00:00
void Clay_OnHover ( void ( * onHoverFunction ) ( Clay_ElementId elementId , Clay_PointerData pointerInfo , intptr_t userData ) , intptr_t userData ) {
2024-12-26 06:58:59 +00:00
if ( Clay__booleanWarnings . maxElementsExceeded ) {
2024-12-01 08:05:56 +00:00
return ;
}
2024-10-22 07:41:35 +00:00
Clay_LayoutElement * openLayoutElement = Clay__GetOpenLayoutElement ( ) ;
if ( openLayoutElement - > id = = 0 ) {
Clay__GenerateIdForAnonymousElement ( openLayoutElement ) ;
}
Clay_LayoutElementHashMapItem * hashMapItem = Clay__GetHashMapItem ( openLayoutElement - > id ) ;
hashMapItem - > onHoverFunction = onHoverFunction ;
hashMapItem - > hoverFunctionUserData = userData ;
}
2024-08-23 04:05:23 +00:00
CLAY_WASM_EXPORT ( " Clay_PointerOver " )
2024-09-16 09:34:59 +00:00
bool Clay_PointerOver ( Clay_ElementId elementId ) { // TODO return priority for separating multiple results
2024-12-30 00:09:14 +00:00
for ( uint32_t i = 0 ; i < Clay__pointerOverIds . length ; + + i ) {
2024-09-16 09:34:59 +00:00
if ( Clay__ElementIdArray_Get ( & Clay__pointerOverIds , i ) - > id = = elementId . id ) {
2024-08-23 04:05:23 +00:00
return true ;
}
}
return false ;
}
2024-08-26 07:05:43 +00:00
CLAY_WASM_EXPORT ( " Clay_GetScrollContainerData " )
2024-09-16 09:34:59 +00:00
Clay_ScrollContainerData Clay_GetScrollContainerData ( Clay_ElementId id ) {
2024-12-30 00:09:14 +00:00
for ( uint32_t i = 0 ; i < Clay__scrollContainerDatas . length ; + + i ) {
2024-08-26 07:05:43 +00:00
Clay__ScrollContainerDataInternal * scrollContainerData = Clay__ScrollContainerDataInternalArray_Get ( & Clay__scrollContainerDatas , i ) ;
2024-09-16 09:34:59 +00:00
if ( scrollContainerData - > elementId = = id . id ) {
2024-09-28 23:53:46 +00:00
return CLAY__INIT ( Clay_ScrollContainerData ) {
2024-08-26 07:05:43 +00:00
. scrollPosition = & scrollContainerData - > scrollPosition ,
2024-12-30 00:09:14 +00:00
. scrollContainerDimensions = { scrollContainerData - > boundingBox . width , scrollContainerData - > boundingBox . height } ,
2024-08-26 07:05:43 +00:00
. contentDimensions = scrollContainerData - > contentSize ,
2024-10-22 07:41:35 +00:00
. config = * Clay__FindElementConfigWithType ( scrollContainerData - > layoutElement , CLAY__ELEMENT_CONFIG_TYPE_SCROLL_CONTAINER ) . scrollElementConfig ,
2024-08-26 07:05:43 +00:00
. found = true
} ;
}
}
2024-12-30 00:09:14 +00:00
return CLAY__INIT ( Clay_ScrollContainerData ) { 0 } ;
2024-08-26 07:05:43 +00:00
}
2024-09-16 09:34:59 +00:00
CLAY_WASM_EXPORT ( " Clay_SetDebugModeEnabled " )
void Clay_SetDebugModeEnabled ( bool enabled ) {
Clay__debugModeEnabled = enabled ;
}
2024-12-29 23:04:48 +00:00
CLAY_WASM_EXPORT ( " Clay_IsDebugModeEnabled " )
bool Clay_IsDebugModeEnabled ( void ) {
return Clay__debugModeEnabled ;
}
2024-12-18 22:35:18 +00:00
CLAY_WASM_EXPORT ( " Clay_SetCullingEnabled " )
void Clay_SetCullingEnabled ( bool enabled ) {
Clay__disableCulling = ! enabled ;
}
CLAY_WASM_EXPORT ( " Clay_SetExternalScrollHandlingEnabled " )
void Clay_SetExternalScrollHandlingEnabled ( bool enabled ) {
Clay__externalScrollHandlingEnabled = enabled ;
}
2024-12-26 06:58:59 +00:00
CLAY_WASM_EXPORT ( " Clay_SetMaxElementCount " )
void Clay_SetMaxElementCount ( uint32_t maxElementCount ) {
Clay__maxElementCount = maxElementCount ;
}
CLAY_WASM_EXPORT ( " Clay_SetMaxMeasureTextCacheWordCount " )
void Clay_SetMaxMeasureTextCacheWordCount ( uint32_t maxMeasureTextCacheWordCount ) {
Clay__maxMeasureTextCacheWordCount = maxMeasureTextCacheWordCount ;
}
2024-12-30 00:09:14 +00:00
# endif // CLAY_IMPLEMENTATION
2024-08-23 04:05:23 +00:00
/*
LICENSE
zlib / libpng license
Copyright ( c ) 2024 Nic Barker
This software is provided ' as - is ' , without any express or implied warranty .
In no event will the authors be held liable for any damages arising from the
use of this software .
Permission is granted to anyone to use this software for any purpose ,
including commercial applications , and to alter it and redistribute it
freely , subject to the following restrictions :
1. The origin of this software must not be misrepresented ; you must not
claim that you wrote the original software . If you use this software in a
product , an acknowledgment in the product documentation would be
appreciated but is not required .
2. Altered source versions must be plainly marked as such , and must not
be misrepresented as being the original software .
3. This notice may not be removed or altered from any source
distribution .
2024-10-02 02:05:52 +00:00
*/