package clay import "core:c" import "core:strings" when ODIN_OS == .Windows { foreign import Clay "windows/clay.lib" } else when ODIN_OS == .Linux { foreign import Clay "linux/clay.a" } else when ODIN_OS == .Darwin { when ODIN_ARCH == .arm64 { foreign import Clay "macos-arm64/clay.a" } else { foreign import Clay "macos/clay.a" } } else when ODIN_ARCH == .wasm32 || ODIN_ARCH == .wasm64p32 { foreign import Clay "wasm/clay.o" } String :: struct { length: c.int, chars: [^]c.char, } Vector2 :: [2]c.float Dimensions :: struct { width: c.float, height: c.float, } Arena :: struct { label: String, nextAllocation: u64, capacity: u64, memory: [^]c.char, } BoundingBox :: struct { x: c.float, y: c.float, width: c.float, height: c.float, } Color :: [4]c.float CornerRadius :: struct { topLeft: c.float, topRight: c.float, bottomLeft: c.float, bottomRight: c.float, } BorderData :: struct { width: u32, color: Color, } ElementId :: struct { id: u32, offset: u32, baseId: u32, stringId: String, } when ODIN_OS == .Windows { EnumBackingType :: u32 } else { EnumBackingType :: u8 } RenderCommandType :: enum EnumBackingType { None, Rectangle, Border, Text, Image, ScissorStart, ScissorEnd, Custom, } RectangleElementConfig :: struct { color: Color, cornerRadius: CornerRadius, } TextWrapMode :: enum EnumBackingType { Words, Newlines, None, } TextElementConfig :: struct { textColor: Color, fontId: u16, fontSize: u16, letterSpacing: u16, lineSpacing: u16, wrapMode: TextWrapMode, } ImageElementConfig :: struct { imageData: rawptr, sourceDimensions: Dimensions, } CustomElementConfig :: struct { customData: rawptr, } BorderElementConfig :: struct { left: BorderData, right: BorderData, top: BorderData, bottom: BorderData, betweenChildren: BorderData, cornerRadius: CornerRadius, } ScrollElementConfig :: struct { horizontal: bool, vertical: bool, } FloatingAttachPointType :: enum EnumBackingType { LEFT_TOP, LEFT_CENTER, LEFT_BOTTOM, CENTER_TOP, CENTER_CENTER, CENTER_BOTTOM, RIGHT_TOP, RIGHT_CENTER, RIGHT_BOTTOM, } FloatingAttachPoints :: struct { element: FloatingAttachPointType, parent: FloatingAttachPointType, } FloatingElementConfig :: struct { offset: Vector2, expand: Dimensions, zIndex: u16, parentId: u32, attachment: FloatingAttachPoints, } ElementConfigUnion :: struct #raw_union { rectangleElementConfig: ^RectangleElementConfig, textElementConfig: ^TextElementConfig, imageElementConfig: ^ImageElementConfig, customElementConfig: ^CustomElementConfig, borderElementConfig: ^BorderElementConfig, } RenderCommand :: struct { boundingBox: BoundingBox, config: ElementConfigUnion, text: String, id: u32, commandType: RenderCommandType, } ScrollContainerData :: struct { // 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. scrollPosition: ^Vector2, scrollContainerDimensions: Dimensions, contentDimensions: Dimensions, config: ScrollElementConfig, // Indicates whether an actual scroll container matched the provided ID or if the default struct was returned. found: bool, } SizingType :: enum EnumBackingType { FIT, GROW, PERCENT, } SizingConstraintsMinMax :: struct { min: c.float, max: c.float, } SizingConstraints :: struct #raw_union { sizeMinMax: SizingConstraintsMinMax, sizePercent: c.float, } SizingAxis :: struct { // Note: `min` is used for CLAY_SIZING_PERCENT, slightly different to clay.h due to lack of C anonymous unions constraints: SizingConstraints, type: SizingType, } Sizing :: struct { width: SizingAxis, height: SizingAxis, } Padding :: struct { x: u16, y: u16, } LayoutDirection :: enum EnumBackingType { LEFT_TO_RIGHT, TOP_TO_BOTTOM, } LayoutAlignmentX :: enum EnumBackingType { LEFT, RIGHT, CENTER, } LayoutAlignmentY :: enum EnumBackingType { TOP, BOTTOM, CENTER, } ChildAlignment :: struct { x: LayoutAlignmentX, y: LayoutAlignmentY, } LayoutConfig :: struct { sizing: Sizing, padding: Padding, childGap: u16, layoutDirection: LayoutDirection, childAlignment: ChildAlignment, } ClayArray :: struct($type: typeid) { capacity: u32, length: u32, internalArray: [^]type, } @(link_prefix = "Clay_", default_calling_convention = "c") foreign Clay { MinMemorySize :: proc() -> u32 --- CreateArenaWithCapacityAndMemory :: proc(capacity: u32, offset: [^]u8) -> Arena --- SetPointerState :: proc(position: Vector2, pointerDown: bool) --- Initialize :: proc(arena: Arena, layoutDimensions: Dimensions) --- UpdateScrollContainers :: proc(isPointerActive: bool, scrollDelta: Vector2, deltaTime: c.float) --- SetLayoutDimensions :: proc(dimensions: Dimensions) --- BeginLayout :: proc() --- EndLayout :: proc() -> ClayArray(RenderCommand) --- PointerOver :: proc(id: ElementId) -> bool --- GetScrollContainerData :: proc(id: ElementId) -> ScrollContainerData --- SetMeasureTextFunction :: proc(measureTextFunction: proc "c" (text: ^String, config: ^TextElementConfig) -> Dimensions) --- RenderCommandArray_Get :: proc(array: ^ClayArray(RenderCommand), index: i32) -> ^RenderCommand --- SetDebugModeEnabled :: proc(enabled: bool) --- } @(link_prefix = "Clay_", default_calling_convention = "c", private) foreign Clay { _OpenContainerElement :: proc(id: ElementId, layoutConfig: ^LayoutConfig) --- _OpenRectangleElement :: proc(id: ElementId, layoutConfig: ^LayoutConfig, rectangleConfig: ^RectangleElementConfig) --- _OpenTextElement :: proc(id: ElementId, text: String, textConfig: ^TextElementConfig) --- _OpenImageElement :: proc(id: ElementId, layoutConfig: ^LayoutConfig, imageConfig: ^ImageElementConfig) --- _OpenScrollElement :: proc(id: ElementId, layoutConfig: ^LayoutConfig, imageConfig: ^ScrollElementConfig) --- _OpenFloatingElement :: proc(id: ElementId, layoutConfig: ^LayoutConfig, imageConfig: ^FloatingElementConfig) --- _OpenBorderElement :: proc(id: ElementId, layoutConfig: ^LayoutConfig, imageConfig: ^BorderElementConfig) --- _OpenCustomElement :: proc(id: ElementId, layoutConfig: ^LayoutConfig, imageConfig: ^CustomElementConfig) --- _CloseElementWithChildren :: proc() --- _CloseScrollElement :: proc() --- _CloseFloatingElement :: proc() --- _StoreLayoutConfig :: proc(config: LayoutConfig) -> ^LayoutConfig --- _StoreRectangleElementConfig :: proc(config: RectangleElementConfig) -> ^RectangleElementConfig --- _StoreTextElementConfig :: proc(config: TextElementConfig) -> ^TextElementConfig --- _StoreImageElementConfig :: proc(config: ImageElementConfig) -> ^ImageElementConfig --- _StoreFloatingElementConfig :: proc(config: FloatingElementConfig) -> ^FloatingElementConfig --- _StoreCustomElementConfig :: proc(config: CustomElementConfig) -> ^CustomElementConfig --- _StoreScrollElementConfig :: proc(config: ScrollElementConfig) -> ^ScrollElementConfig --- _StoreBorderElementConfig :: proc(config: BorderElementConfig) -> ^BorderElementConfig --- _HashString :: proc(toHash: String, index: u32, seed: u32) -> ElementId --- _GetOpenLayoutElementId :: proc() -> u32 --- } @(require_results, deferred_none = _CloseElementWithChildren) Container :: proc(id: ElementId, layoutConfig: ^LayoutConfig) -> bool { _OpenContainerElement(id, layoutConfig) return true } @(require_results, deferred_none = _CloseElementWithChildren) Rectangle :: proc(id: ElementId, layoutConfig: ^LayoutConfig, rectangleConfig: ^RectangleElementConfig) -> bool { _OpenRectangleElement(id, layoutConfig, rectangleConfig) return true } Text :: proc(id: ElementId, text: string, textConfig: ^TextElementConfig) -> bool { _OpenTextElement(id, MakeString(text), textConfig) return true } @(require_results, deferred_none = _CloseElementWithChildren) Image :: proc(id: ElementId, layoutConfig: ^LayoutConfig, imageConfig: ^ImageElementConfig) -> bool { _OpenImageElement(id, layoutConfig, imageConfig) return true } @(require_results, deferred_none = _CloseScrollElement) Scroll :: proc(id: ElementId, layoutConfig: ^LayoutConfig, scrollConfig: ^ScrollElementConfig) -> bool { _OpenScrollElement(id, layoutConfig, scrollConfig) return true } @(require_results, deferred_none = _CloseFloatingElement) Floating :: proc(id: ElementId, layoutConfig: ^LayoutConfig, floatingConfig: ^FloatingElementConfig) -> bool { _OpenFloatingElement(id, layoutConfig, floatingConfig) return true } @(require_results, deferred_none = _CloseElementWithChildren) Border :: proc(id: ElementId, layoutConfig: ^LayoutConfig, borderConfig: ^BorderElementConfig) -> bool { _OpenBorderElement(id, layoutConfig, borderConfig) return true } @(require_results, deferred_none = _CloseElementWithChildren) Custom :: proc(id: ElementId, layoutConfig: ^LayoutConfig, customConfig: ^CustomElementConfig) -> bool { _OpenCustomElement(id, layoutConfig, customConfig) return true } Layout :: proc(config: LayoutConfig) -> ^LayoutConfig { return _StoreLayoutConfig(config) } RectangleConfig :: proc(config: RectangleElementConfig) -> ^RectangleElementConfig { return _StoreRectangleElementConfig(config) } TextConfig :: proc(config: TextElementConfig) -> ^TextElementConfig { return _StoreTextElementConfig(config) } ImageConfig :: proc(config: ImageElementConfig) -> ^ImageElementConfig { return _StoreImageElementConfig(config) } FloatingConfig :: proc(config: FloatingElementConfig) -> ^FloatingElementConfig { return _StoreFloatingElementConfig(config) } CustomConfig :: proc(config: CustomElementConfig) -> ^CustomElementConfig { return _StoreCustomElementConfig(config) } ScrollConfig :: proc(config: ScrollElementConfig) -> ^ScrollElementConfig { return _StoreScrollElementConfig(config) } BorderConfig :: proc(config: BorderElementConfig) -> ^BorderElementConfig { return _StoreBorderElementConfig(config) } BorderConfigOutside :: proc(outsideBorders: BorderData) -> ^BorderElementConfig { return _StoreBorderElementConfig((BorderElementConfig){left = outsideBorders, right = outsideBorders, top = outsideBorders, bottom = outsideBorders}) } BorderConfigOutsideRadius :: proc(outsideBorders: BorderData, radius: f32) -> ^BorderElementConfig { return _StoreBorderElementConfig( (BorderElementConfig){left = outsideBorders, right = outsideBorders, top = outsideBorders, bottom = outsideBorders, cornerRadius = {radius, radius, radius, radius}}, ) } BorderConfigAll :: proc(allBorders: BorderData) -> ^BorderElementConfig { return _StoreBorderElementConfig((BorderElementConfig){left = allBorders, right = allBorders, top = allBorders, bottom = allBorders, betweenChildren = allBorders}) } BorderConfigAllRadius :: proc(allBorders: BorderData, radius: f32) -> ^BorderElementConfig { return _StoreBorderElementConfig( (BorderElementConfig){left = allBorders, right = allBorders, top = allBorders, bottom = allBorders, cornerRadius = {radius, radius, radius, radius}}, ) } CornerRadiusAll :: proc(radius: f32) -> CornerRadius { return CornerRadius{radius, radius, radius, radius} } SizingFit :: proc(sizeMinMax: SizingConstraintsMinMax) -> SizingAxis { return SizingAxis{type = SizingType.FIT, constraints = {sizeMinMax = sizeMinMax}} } SizingGrow :: proc(sizeMinMax: SizingConstraintsMinMax) -> SizingAxis { return SizingAxis{type = SizingType.GROW, constraints = {sizeMinMax = sizeMinMax}} } SizingFixed :: proc(size: c.float) -> SizingAxis { return SizingAxis{type = SizingType.FIT, constraints = {sizeMinMax = {size, size}}} } SizingPercent :: proc(sizePercent: c.float) -> SizingAxis { return SizingAxis{type = SizingType.PERCENT, constraints = {sizePercent = sizePercent}} } MakeString :: proc(label: string) -> String { return String{chars = raw_data(label), length = cast(c.int)len(label)} } ID :: proc(label: string, index: u32 = 0) -> ElementId { return _HashString(MakeString(label), index, 0) } IDLocal :: proc(label: string, index: u32 = 0) -> ElementId { return _HashString(MakeString(label), index, _GetOpenLayoutElementId()) }