2024-08-23 04:05:23 +00:00
# include "raylib.h"
# include "raymath.h"
# include "stdint.h"
# include "string.h"
# include "stdio.h"
# include "stdlib.h"
# define CLAY_RECTANGLE_TO_RAYLIB_RECTANGLE(rectangle) (Rectangle) { .x = rectangle.x, .y = rectangle.y, .width = rectangle.width, .height = rectangle.height }
# define CLAY_COLOR_TO_RAYLIB_COLOR(color) (Color) { .r = (unsigned char)roundf(color.r), .g = (unsigned char)roundf(color.g), .b = (unsigned char)roundf(color.b), .a = (unsigned char)roundf(color.a) }
Camera Raylib_camera ;
typedef enum
{
CUSTOM_LAYOUT_ELEMENT_TYPE_3D_MODEL
} CustomLayoutElementType ;
typedef struct
{
Model model ;
float scale ;
Vector3 position ;
Matrix rotation ;
} CustomLayoutElement_3DModel ;
typedef struct
{
CustomLayoutElementType type ;
union {
CustomLayoutElement_3DModel model ;
2025-01-29 04:14:01 +00:00
} customData ;
2024-08-23 04:05:23 +00:00
} CustomLayoutElement ;
// Get a ray trace from the screen position (i.e mouse) within a specific section of the screen
Ray GetScreenToWorldPointWithZDistance ( Vector2 position , Camera camera , int screenWidth , int screenHeight , float zDistance )
{
Ray ray = { 0 } ;
// Calculate normalized device coordinates
// NOTE: y value is negative
float x = ( 2.0f * position . x ) / ( float ) screenWidth - 1.0f ;
float y = 1.0f - ( 2.0f * position . y ) / ( float ) screenHeight ;
float z = 1.0f ;
// Store values in a vector
Vector3 deviceCoords = { x , y , z } ;
// Calculate view matrix from camera look at
Matrix matView = MatrixLookAt ( camera . position , camera . target , camera . up ) ;
Matrix matProj = MatrixIdentity ( ) ;
if ( camera . projection = = CAMERA_PERSPECTIVE )
{
// Calculate projection matrix from perspective
matProj = MatrixPerspective ( camera . fovy * DEG2RAD , ( ( double ) screenWidth / ( double ) screenHeight ) , 0.01f , zDistance ) ;
}
else if ( camera . projection = = CAMERA_ORTHOGRAPHIC )
{
double aspect = ( double ) screenWidth / ( double ) screenHeight ;
double top = camera . fovy / 2.0 ;
double right = top * aspect ;
// Calculate projection matrix from orthographic
matProj = MatrixOrtho ( - right , right , - top , top , 0.01 , 1000.0 ) ;
}
// Unproject far/near points
Vector3 nearPoint = Vector3Unproject ( ( Vector3 ) { deviceCoords . x , deviceCoords . y , 0.0f } , matProj , matView ) ;
Vector3 farPoint = Vector3Unproject ( ( Vector3 ) { deviceCoords . x , deviceCoords . y , 1.0f } , matProj , matView ) ;
// Calculate normalized direction vector
Vector3 direction = Vector3Normalize ( Vector3Subtract ( farPoint , nearPoint ) ) ;
ray . position = farPoint ;
// Apply calculated vectors to ray
ray . direction = direction ;
return ray ;
}
2025-02-04 04:00:19 +00:00
static inline Clay_Dimensions Raylib_MeasureText ( Clay_StringSlice text , Clay_TextElementConfig * config , void * userData ) {
2024-08-23 04:05:23 +00:00
// Measure string size for Font
Clay_Dimensions textSize = { 0 } ;
float maxTextWidth = 0.0f ;
float lineTextWidth = 0 ;
float textHeight = config - > fontSize ;
2025-02-04 04:00:19 +00:00
Font * fonts = ( Font * ) userData ;
Font fontToUse = fonts [ config - > fontId ] ;
2025-01-21 08:26:57 +00:00
// Font failed to load, likely the fonts are in the wrong place relative to the execution dir
if ( ! fontToUse . glyphs ) return textSize ;
2024-09-16 09:34:59 +00:00
float scaleFactor = config - > fontSize / ( float ) fontToUse . baseSize ;
2024-08-23 04:05:23 +00:00
2025-01-19 22:27:22 +00:00
for ( int i = 0 ; i < text . length ; + + i )
2024-08-23 04:05:23 +00:00
{
2025-01-19 22:27:22 +00:00
if ( text . chars [ i ] = = ' \n ' ) {
2024-08-23 04:05:23 +00:00
maxTextWidth = fmax ( maxTextWidth , lineTextWidth ) ;
lineTextWidth = 0 ;
continue ;
}
2025-01-19 22:27:22 +00:00
int index = text . chars [ i ] - 32 ;
2024-08-23 04:05:23 +00:00
if ( fontToUse . glyphs [ index ] . advanceX ! = 0 ) lineTextWidth + = fontToUse . glyphs [ index ] . advanceX ;
else lineTextWidth + = ( fontToUse . recs [ index ] . width + fontToUse . glyphs [ index ] . offsetX ) ;
}
maxTextWidth = fmax ( maxTextWidth , lineTextWidth ) ;
2024-09-16 09:34:59 +00:00
textSize . width = maxTextWidth * scaleFactor ;
2024-08-23 04:05:23 +00:00
textSize . height = textHeight ;
return textSize ;
}
2024-12-19 08:34:20 +00:00
void Clay_Raylib_Initialize ( int width , int height , const char * title , unsigned int flags ) {
2024-08-23 04:05:23 +00:00
SetConfigFlags ( flags ) ;
2024-12-19 08:34:20 +00:00
InitWindow ( width , height , title ) ;
2024-08-23 04:05:23 +00:00
// EnableEventWaiting();
}
2025-02-04 04:00:19 +00:00
void Clay_Raylib_Render ( Clay_RenderCommandArray renderCommands , Font * fonts )
2024-08-23 04:05:23 +00:00
{
for ( int j = 0 ; j < renderCommands . length ; j + + )
{
Clay_RenderCommand * renderCommand = Clay_RenderCommandArray_Get ( & renderCommands , j ) ;
2024-08-28 04:13:28 +00:00
Clay_BoundingBox boundingBox = renderCommand - > boundingBox ;
2024-08-23 04:05:23 +00:00
switch ( renderCommand - > commandType )
{
case CLAY_RENDER_COMMAND_TYPE_TEXT : {
// Raylib uses standard C strings so isn't compatible with cheap slices, we need to clone the string to append null terminator
2025-02-04 04:00:19 +00:00
Clay_TextRenderData * textData = & renderCommand - > renderData . text ;
char * cloned = ( char * ) malloc ( textData - > stringContents . length + 1 ) ;
memcpy ( cloned , textData - > stringContents . chars , textData - > stringContents . length ) ;
cloned [ textData - > stringContents . length ] = ' \0 ' ;
Font fontToUse = fonts [ textData - > fontId ] ;
DrawTextEx ( fontToUse , cloned , ( Vector2 ) { boundingBox . x , boundingBox . y } , ( float ) textData - > fontSize , ( float ) textData - > letterSpacing , CLAY_COLOR_TO_RAYLIB_COLOR ( textData - > textColor ) ) ;
2024-08-23 04:05:23 +00:00
free ( cloned ) ;
break ;
}
case CLAY_RENDER_COMMAND_TYPE_IMAGE : {
2025-02-04 04:00:19 +00:00
Texture2D imageTexture = * ( Texture2D * ) renderCommand - > renderData . image . imageData ;
Clay_Color tintColor = renderCommand - > renderData . image . backgroundColor ;
if ( tintColor . r = = 0 & & tintColor . g = = 0 | | tintColor . b = = 0 | | tintColor . a = = 0 ) {
tintColor = ( Clay_Color ) { 255 , 255 , 255 , 255 } ;
}
2024-08-23 04:05:23 +00:00
DrawTextureEx (
2025-02-04 04:00:19 +00:00
imageTexture ,
( Vector2 ) { boundingBox . x , boundingBox . y } ,
0 ,
boundingBox . width / ( float ) imageTexture . width ,
CLAY_COLOR_TO_RAYLIB_COLOR ( tintColor ) ) ;
2024-08-23 04:05:23 +00:00
break ;
}
case CLAY_RENDER_COMMAND_TYPE_SCISSOR_START : {
BeginScissorMode ( ( int ) roundf ( boundingBox . x ) , ( int ) roundf ( boundingBox . y ) , ( int ) roundf ( boundingBox . width ) , ( int ) roundf ( boundingBox . height ) ) ;
break ;
}
case CLAY_RENDER_COMMAND_TYPE_SCISSOR_END : {
EndScissorMode ( ) ;
break ;
}
case CLAY_RENDER_COMMAND_TYPE_RECTANGLE : {
2025-02-04 04:00:19 +00:00
Clay_RectangleRenderData * config = & renderCommand - > renderData . rectangle ;
2024-08-26 22:16:55 +00:00
if ( config - > cornerRadius . topLeft > 0 ) {
2024-09-16 09:34:59 +00:00
float radius = ( config - > cornerRadius . topLeft * 2 ) / ( float ) ( ( boundingBox . width > boundingBox . height ) ? boundingBox . height : boundingBox . width ) ;
2025-02-04 04:00:19 +00:00
DrawRectangleRounded ( ( Rectangle ) { boundingBox . x , boundingBox . y , boundingBox . width , boundingBox . height } , radius , 8 , CLAY_COLOR_TO_RAYLIB_COLOR ( config - > backgroundColor ) ) ;
2024-08-26 22:16:55 +00:00
} else {
2025-02-04 04:00:19 +00:00
DrawRectangle ( boundingBox . x , boundingBox . y , boundingBox . width , boundingBox . height , CLAY_COLOR_TO_RAYLIB_COLOR ( config - > backgroundColor ) ) ;
2024-08-26 22:16:55 +00:00
}
2024-08-23 04:05:23 +00:00
break ;
}
case CLAY_RENDER_COMMAND_TYPE_BORDER : {
2025-02-04 04:00:19 +00:00
Clay_BorderRenderData * config = & renderCommand - > renderData . border ;
2024-08-23 04:05:23 +00:00
// Left border
2025-02-04 04:00:19 +00:00
if ( config - > width . left > 0 ) {
DrawRectangle ( ( int ) roundf ( boundingBox . x ) , ( int ) roundf ( boundingBox . y + config - > cornerRadius . topLeft ) , ( int ) config - > width . left , ( int ) roundf ( boundingBox . height - config - > cornerRadius . topLeft - config - > cornerRadius . bottomLeft ) , CLAY_COLOR_TO_RAYLIB_COLOR ( config - > color ) ) ;
2024-08-23 04:05:23 +00:00
}
// Right border
2025-02-04 04:00:19 +00:00
if ( config - > width . right > 0 ) {
DrawRectangle ( ( int ) roundf ( boundingBox . x + boundingBox . width - config - > width . right ) , ( int ) roundf ( boundingBox . y + config - > cornerRadius . topRight ) , ( int ) config - > width . right , ( int ) roundf ( boundingBox . height - config - > cornerRadius . topRight - config - > cornerRadius . bottomRight ) , CLAY_COLOR_TO_RAYLIB_COLOR ( config - > color ) ) ;
2024-08-23 04:05:23 +00:00
}
// Top border
2025-02-04 04:00:19 +00:00
if ( config - > width . top > 0 ) {
DrawRectangle ( ( int ) roundf ( boundingBox . x + config - > cornerRadius . topLeft ) , ( int ) roundf ( boundingBox . y ) , ( int ) roundf ( boundingBox . width - config - > cornerRadius . topLeft - config - > cornerRadius . topRight ) , ( int ) config - > width . top , CLAY_COLOR_TO_RAYLIB_COLOR ( config - > color ) ) ;
2024-08-23 04:05:23 +00:00
}
// Bottom border
2025-02-04 04:00:19 +00:00
if ( config - > width . bottom > 0 ) {
DrawRectangle ( ( int ) roundf ( boundingBox . x + config - > cornerRadius . bottomLeft ) , ( int ) roundf ( boundingBox . y + boundingBox . height - config - > width . bottom ) , ( int ) roundf ( boundingBox . width - config - > cornerRadius . bottomLeft - config - > cornerRadius . bottomRight ) , ( int ) config - > width . bottom , CLAY_COLOR_TO_RAYLIB_COLOR ( config - > color ) ) ;
2024-08-23 04:05:23 +00:00
}
if ( config - > cornerRadius . topLeft > 0 ) {
2025-02-04 04:00:19 +00:00
DrawRing ( ( Vector2 ) { roundf ( boundingBox . x + config - > cornerRadius . topLeft ) , roundf ( boundingBox . y + config - > cornerRadius . topLeft ) } , roundf ( config - > cornerRadius . topLeft - config - > width . top ) , config - > cornerRadius . topLeft , 180 , 270 , 10 , CLAY_COLOR_TO_RAYLIB_COLOR ( config - > color ) ) ;
2024-08-23 04:05:23 +00:00
}
if ( config - > cornerRadius . topRight > 0 ) {
2025-02-04 04:00:19 +00:00
DrawRing ( ( Vector2 ) { roundf ( boundingBox . x + boundingBox . width - config - > cornerRadius . topRight ) , roundf ( boundingBox . y + config - > cornerRadius . topRight ) } , roundf ( config - > cornerRadius . topRight - config - > width . top ) , config - > cornerRadius . topRight , 270 , 360 , 10 , CLAY_COLOR_TO_RAYLIB_COLOR ( config - > color ) ) ;
2024-08-23 04:05:23 +00:00
}
if ( config - > cornerRadius . bottomLeft > 0 ) {
2025-02-04 04:00:19 +00:00
DrawRing ( ( Vector2 ) { roundf ( boundingBox . x + config - > cornerRadius . bottomLeft ) , roundf ( boundingBox . y + boundingBox . height - config - > cornerRadius . bottomLeft ) } , roundf ( config - > cornerRadius . bottomLeft - config - > width . top ) , config - > cornerRadius . bottomLeft , 90 , 180 , 10 , CLAY_COLOR_TO_RAYLIB_COLOR ( config - > color ) ) ;
2024-08-23 04:05:23 +00:00
}
if ( config - > cornerRadius . bottomRight > 0 ) {
2025-02-04 04:00:19 +00:00
DrawRing ( ( Vector2 ) { roundf ( boundingBox . x + boundingBox . width - config - > cornerRadius . bottomRight ) , roundf ( boundingBox . y + boundingBox . height - config - > cornerRadius . bottomRight ) } , roundf ( config - > cornerRadius . bottomRight - config - > width . bottom ) , config - > cornerRadius . bottomRight , 0.1 , 90 , 10 , CLAY_COLOR_TO_RAYLIB_COLOR ( config - > color ) ) ;
2024-08-23 04:05:23 +00:00
}
break ;
}
case CLAY_RENDER_COMMAND_TYPE_CUSTOM : {
2025-02-04 04:00:19 +00:00
Clay_CustomRenderData * config = & renderCommand - > renderData . custom ;
CustomLayoutElement * customElement = ( CustomLayoutElement * ) config - > customData ;
2024-08-23 04:05:23 +00:00
if ( ! customElement ) continue ;
switch ( customElement - > type ) {
case CUSTOM_LAYOUT_ELEMENT_TYPE_3D_MODEL : {
2024-08-28 04:13:28 +00:00
Clay_BoundingBox rootBox = renderCommands . internalArray [ 0 ] . boundingBox ;
2024-08-23 04:05:23 +00:00
float scaleValue = CLAY__MIN ( CLAY__MIN ( 1 , 768 / rootBox . height ) * CLAY__MAX ( 1 , rootBox . width / 1024 ) , 1.5f ) ;
Ray positionRay = GetScreenToWorldPointWithZDistance ( ( Vector2 ) { renderCommand - > boundingBox . x + renderCommand - > boundingBox . width / 2 , renderCommand - > boundingBox . y + ( renderCommand - > boundingBox . height / 2 ) + 20 } , Raylib_camera , ( int ) roundf ( rootBox . width ) , ( int ) roundf ( rootBox . height ) , 140 ) ;
BeginMode3D ( Raylib_camera ) ;
2025-01-29 04:14:01 +00:00
DrawModel ( customElement - > customData . model . model , positionRay . position , customElement - > customData . model . scale * scaleValue , WHITE ) ; // Draw 3d model with texture
2024-08-23 04:05:23 +00:00
EndMode3D ( ) ;
break ;
}
default : break ;
}
break ;
}
default : {
printf ( " Error: unhandled render command. " ) ;
exit ( 1 ) ;
}
}
}
2024-12-19 08:34:20 +00:00
}