From 7bfc8e8ca75882de434c601c4294ca1774b69278 Mon Sep 17 00:00:00 2001 From: Anstro Pleuton <121956207+anstropleuton@users.noreply.github.com> Date: Thu, 23 Jan 2025 01:51:36 -0800 Subject: [PATCH 1/3] [example] Add shaders_rounded_rectangle example (#4719) * Add shaders_rounded_rectangle example * Minor tweaks to example template (add star, more) * Combine shaders * Fix changes after review --------- Co-authored-by: Anstro Pleuton --- examples/Makefile | 1 + examples/Makefile.Web | 1 + examples/README.md | 27 +- examples/examples_template.c | 6 +- .../shaders/glsl100/rounded_rectangle.fs | 76 ++++ .../shaders/glsl120/rounded_rectangle.fs | 74 ++++ .../shaders/glsl330/rounded_rectangle.fs | 77 ++++ examples/shaders/shaders_rounded_rectangle.c | 238 +++++++++++ .../shaders/shaders_rounded_rectangle.png | Bin 0 -> 13936 bytes .../shaders_rounded_rectangle.vcxproj | 387 ++++++++++++++++++ projects/VS2022/raylib.sln | 19 + 11 files changed, 891 insertions(+), 15 deletions(-) create mode 100644 examples/shaders/resources/shaders/glsl100/rounded_rectangle.fs create mode 100644 examples/shaders/resources/shaders/glsl120/rounded_rectangle.fs create mode 100644 examples/shaders/resources/shaders/glsl330/rounded_rectangle.fs create mode 100644 examples/shaders/shaders_rounded_rectangle.c create mode 100644 examples/shaders/shaders_rounded_rectangle.png create mode 100644 projects/VS2022/examples/shaders_rounded_rectangle.vcxproj diff --git a/examples/Makefile b/examples/Makefile index 407abd793..0a2c908a3 100644 --- a/examples/Makefile +++ b/examples/Makefile @@ -634,6 +634,7 @@ SHADERS = \ shaders/shaders_palette_switch \ shaders/shaders_postprocessing \ shaders/shaders_raymarching \ + shaders/shaders_rounded_rectangle \ shaders/shaders_shadowmap \ shaders/shaders_shapes_textures \ shaders/shaders_simple_mask \ diff --git a/examples/Makefile.Web b/examples/Makefile.Web index 32d6daa57..6f060652f 100644 --- a/examples/Makefile.Web +++ b/examples/Makefile.Web @@ -514,6 +514,7 @@ SHADERS = \ shaders/shaders_palette_switch \ shaders/shaders_postprocessing \ shaders/shaders_raymarching \ + shaders/shaders_rounded_rectangle \ shaders/shaders_shadowmap \ shaders/shaders_shapes_textures \ shaders/shaders_simple_mask \ diff --git a/examples/README.md b/examples/README.md index ba14d82dd..c42a701a2 100644 --- a/examples/README.md +++ b/examples/README.md @@ -199,6 +199,7 @@ Examples using raylib shaders functionality, including shaders loading, paramete | 138 | [shaders_write_depth](shaders/shaders_write_depth.c) | shaders_write_depth | ⭐️⭐️☆☆ | 4.2 | 4.2 | [Buğra Alptekin Sarı](https://github.com/BugraAlptekinSari) | | 139 | [shaders_basic_pbr](shaders/shaders_basic_pbr.c) | shaders_basic_pbr | ⭐️⭐️⭐️⭐️ | 5.0 | 5.1-dev | [Afan OLOVCIC](https://github.com/_DevDad) | | 140 | [shaders_lightmap](shaders/shaders_lightmap.c) | shaders_lightmap | ⭐️⭐️⭐️☆ | 4.5 | 4.5 | [Jussi Viitala](https://github.com/nullstare) | +| 141 | [shaders_rounded_rectangle](shaders/shaders_rounded_rectangle.c) | shaders_rounded_rectangle | ⭐️⭐️⭐️☆ | 5.5 | 5.5 | [Anstro Pleuton](https://github.com/anstropleuton) | ### category: audio @@ -206,13 +207,13 @@ Examples using raylib audio functionality, including sound/music loading and pla | ## | example | image | difficulty
level | version
created | last version
updated | original
developer | |----|----------|--------|:-------------------:|:------------------:|:-----------------------:|:----------------------| -| 141 | [audio_module_playing](audio/audio_module_playing.c) | audio_module_playing | ⭐️☆☆☆ | 1.5 | 3.5 | [Ray](https://github.com/raysan5) | -| 142 | [audio_music_stream](audio/audio_music_stream.c) | audio_music_stream | ⭐️☆☆☆ | 1.3 | 4.2 | [Ray](https://github.com/raysan5) | -| 143 | [audio_raw_stream](audio/audio_raw_stream.c) | audio_raw_stream | ⭐️⭐️⭐️☆ | 1.6 | 4.2 | [Ray](https://github.com/raysan5) | -| 144 | [audio_sound_loading](audio/audio_sound_loading.c) | audio_sound_loading | ⭐️☆☆☆ | 1.1 | 3.5 | [Ray](https://github.com/raysan5) | -| 145 | [audio_mixed_processor](audio/audio_mixed_processor.c) | audio_mixed_processor | ⭐️⭐️⭐️⭐️ | 4.2 | 4.2 | [hkc](https://github.com/hatkidchan) | -| 146 | [audio_stream_effects](audio/audio_stream_effects.c) | audio_stream_effects | ⭐️⭐️⭐️⭐️ | 4.2 | 5.0 | [Ray](https://github.com/raysan5) | -| 147 | [audio_sound_multi](audio/audio_sound_multi.c) | audio_sound_multi | ⭐️⭐️☆☆ | 4.6 | 4.6 | [Jeffery Myers](https://github.com/JeffM2501) | +| 142 | [audio_module_playing](audio/audio_module_playing.c) | audio_module_playing | ⭐️☆☆☆ | 1.5 | 3.5 | [Ray](https://github.com/raysan5) | +| 143 | [audio_music_stream](audio/audio_music_stream.c) | audio_music_stream | ⭐️☆☆☆ | 1.3 | 4.2 | [Ray](https://github.com/raysan5) | +| 144 | [audio_raw_stream](audio/audio_raw_stream.c) | audio_raw_stream | ⭐️⭐️⭐️☆ | 1.6 | 4.2 | [Ray](https://github.com/raysan5) | +| 145 | [audio_sound_loading](audio/audio_sound_loading.c) | audio_sound_loading | ⭐️☆☆☆ | 1.1 | 3.5 | [Ray](https://github.com/raysan5) | +| 146 | [audio_mixed_processor](audio/audio_mixed_processor.c) | audio_mixed_processor | ⭐️⭐️⭐️⭐️ | 4.2 | 4.2 | [hkc](https://github.com/hatkidchan) | +| 147 | [audio_stream_effects](audio/audio_stream_effects.c) | audio_stream_effects | ⭐️⭐️⭐️⭐️ | 4.2 | 5.0 | [Ray](https://github.com/raysan5) | +| 148 | [audio_sound_multi](audio/audio_sound_multi.c) | audio_sound_multi | ⭐️⭐️☆☆ | 4.6 | 4.6 | [Jeffery Myers](https://github.com/JeffM2501) | ### category: others @@ -220,12 +221,12 @@ Examples showing raylib misc functionality that does not fit in other categories | ## | example | image | difficulty
level | version
created | last version
updated | original
developer | |----|----------|--------|:-------------------:|:------------------:|:-----------------------:|:----------------------| -| 148 | [rlgl_standalone](others/rlgl_standalone.c) | rlgl_standalone | ⭐️⭐️⭐️⭐️ | 1.6 | 4.0 | [Ray](https://github.com/raysan5) | -| 149 | [rlgl_compute_shader](others/rlgl_compute_shader.c) | rlgl_compute_shader | ⭐️⭐️⭐️⭐️ | 4.0 | 4.0 | [Teddy Astie](https://github.com/tsnake41) | -| 150 | [easings_testbed](others/easings_testbed.c) | easings_testbed | ⭐️⭐️⭐️☆ | 2.5 | 3.0 | [Juan Miguel López](https://github.com/flashback-fx) | -| 151 | [raylib_opengl_interop](others/raylib_opengl_interop.c) | raylib_opengl_interop | ⭐️⭐️⭐️⭐️ | 3.8 | 4.0 | [Stephan Soller](https://github.com/arkanis) | -| 152 | [embedded_files_loading](others/embedded_files_loading.c) | embedded_files_loading | ⭐️⭐️☆☆ | 3.0 | 3.5 | [Kristian Holmgren](https://github.com/defutura) | -| 153 | [raymath_vector_angle](others/raymath_vector_angle.c) | raymath_vector_angle | ⭐️⭐️☆☆ | 1.0 | 4.6 | [Ray](https://github.com/raysan5) | +| 149 | [rlgl_standalone](others/rlgl_standalone.c) | rlgl_standalone | ⭐️⭐️⭐️⭐️ | 1.6 | 4.0 | [Ray](https://github.com/raysan5) | +| 150 | [rlgl_compute_shader](others/rlgl_compute_shader.c) | rlgl_compute_shader | ⭐️⭐️⭐️⭐️ | 4.0 | 4.0 | [Teddy Astie](https://github.com/tsnake41) | +| 151 | [easings_testbed](others/easings_testbed.c) | easings_testbed | ⭐️⭐️⭐️☆ | 2.5 | 3.0 | [Juan Miguel López](https://github.com/flashback-fx) | +| 152 | [raylib_opengl_interop](others/raylib_opengl_interop.c) | raylib_opengl_interop | ⭐️⭐️⭐️⭐️ | 3.8 | 4.0 | [Stephan Soller](https://github.com/arkanis) | +| 153 | [embedded_files_loading](others/embedded_files_loading.c) | embedded_files_loading | ⭐️⭐️☆☆ | 3.0 | 3.5 | [Kristian Holmgren](https://github.com/defutura) | +| 154 | [raymath_vector_angle](others/raymath_vector_angle.c) | raymath_vector_angle | ⭐️⭐️☆☆ | 1.0 | 4.6 | [Ray](https://github.com/raysan5) | As always contributions are welcome, feel free to send new examples! Here is an [examples template](examples_template.c) to start with! diff --git a/examples/examples_template.c b/examples/examples_template.c index 0a44117b4..49136a4d4 100644 --- a/examples/examples_template.c +++ b/examples/examples_template.c @@ -56,7 +56,9 @@ /******************************************************************************************* * -* raylib [core] example - Basic window +* raylib [] example - +* +* Example complexity rating: [★☆??] ?/4 * * Example originally created with raylib 5.5, last time updated with raylib 5.5 * @@ -81,7 +83,7 @@ int main(void) const int screenWidth = 800; const int screenHeight = 450; - InitWindow(screenWidth, screenHeight, "raylib [core] example - basic window"); + InitWindow(screenWidth, screenHeight, "raylib [] example - "); // TODO: Load resources / Initialize variables at this point diff --git a/examples/shaders/resources/shaders/glsl100/rounded_rectangle.fs b/examples/shaders/resources/shaders/glsl100/rounded_rectangle.fs new file mode 100644 index 000000000..3c8d07978 --- /dev/null +++ b/examples/shaders/resources/shaders/glsl100/rounded_rectangle.fs @@ -0,0 +1,76 @@ +// Note: SDF by Iñigo Quilez is licensed under MIT License + +#version 100 + +precision mediump float; + +// Input vertex attributes (from vertex shader) +varying vec2 fragTexCoord; +varying vec4 fragColor; + +// Input uniform values +uniform sampler2D texture0; +uniform vec4 colDiffuse; + +uniform vec4 rectangle; // Rectangle dimensions (x, y, width, height) +uniform vec4 radius; // Corner radius (top-left, top-right, bottom-left, bottom-right) +uniform vec4 color; + +// Shadow parameters +uniform float shadowRadius; +uniform vec2 shadowOffset; +uniform float shadowScale; +uniform vec4 shadowColor; + +// Border parameters +uniform float borderThickness; +uniform vec4 borderColor; + +// Create a rounded rectangle using signed distance field +// Thanks to Iñigo Quilez (https://www.iquilezles.org/www/articles/distfunctions/distfunctions.htm) +// And thanks to inobelar (https://www.shadertoy.com/view/fsdyzB) for shader +// MIT License +float RoundedRectangleSDF(vec2 fragCoord, vec2 center, vec2 halfSize, vec4 radius) +{ + vec2 fragFromCenter = fragCoord - center; + + // Determine which corner radius to use + radius.xy = (fragFromCenter.y > 0.0) ? radius.xy : radius.zw; + radius.x = (fragFromCenter.x < 0.0) ? radius.x : radius.y; + + // Calculate signed distance field + vec2 dist = abs(fragFromCenter) - halfSize + radius.x; + return min(max(dist.x, dist.y), 0.0) + length(max(dist, 0.0)) - radius.x; +} + +void main() +{ + // Texel color fetching from texture sampler + vec4 texelColor = texture2D(texture0, fragTexCoord); + + // Requires fragment coordinate varying pixels + vec2 fragCoord = gl_FragCoord.xy; + + // Calculate signed distance field for rounded rectangle + vec2 halfSize = rectangle.zw*0.5; + vec2 center = rectangle.xy + halfSize; + float recSDF = RoundedRectangleSDF(fragCoord, center, halfSize, radius); + + // Calculate signed distance field for rectangle shadow + vec2 shadowHalfSize = halfSize*shadowScale; + vec2 shadowCenter = center + shadowOffset; + float shadowSDF = RoundedRectangleSDF(fragCoord, shadowCenter, shadowHalfSize, radius); + + // Caculate alpha factors + float recFactor = smoothstep(1.0, 0.0, recSDF); + float shadowFactor = smoothstep(shadowRadius, 0.0, shadowSDF); + float borderFactor = smoothstep(0.0, 1.0, recSDF + borderThickness)*recFactor; + + // Multiply each color by its respective alpha factor + vec4 recColor = vec4(color.rgb, color.a*recFactor); + vec4 shadowCol = vec4(shadowColor.rgb, shadowColor.a*shadowFactor); + vec4 borderCol = vec4(borderColor.rgb, borderColor.a*borderFactor); + + // Combine the colors varying the order (shadow, rectangle, border) + gl_FragColor = mix(mix(shadowCol, recColor, recColor.a), borderCol, borderCol.a); +} \ No newline at end of file diff --git a/examples/shaders/resources/shaders/glsl120/rounded_rectangle.fs b/examples/shaders/resources/shaders/glsl120/rounded_rectangle.fs new file mode 100644 index 000000000..eb96c28d3 --- /dev/null +++ b/examples/shaders/resources/shaders/glsl120/rounded_rectangle.fs @@ -0,0 +1,74 @@ +// Note: SDF by Iñigo Quilez is licensed under MIT License + +#version 120 + +// Input vertex attributes (from vertex shader) +varying vec2 fragTexCoord; +varying vec4 fragColor; + +// Input uniform values +uniform sampler2D texture0; +uniform vec4 colDiffuse; + +uniform vec4 rectangle; // Rectangle dimensions (x, y, width, height) +uniform vec4 radius; // Corner radius (top-left, top-right, bottom-left, bottom-right) +uniform vec4 color; + +// Shadow parameters +uniform float shadowRadius; +uniform vec2 shadowOffset; +uniform float shadowScale; +uniform vec4 shadowColor; + +// Border parameters +uniform float borderThickness; +uniform vec4 borderColor; + +// Create a rounded rectangle using signed distance field +// Thanks to Iñigo Quilez (https://www.iquilezles.org/www/articles/distfunctions/distfunctions.htm) +// And thanks to inobelar (https://www.shadertoy.com/view/fsdyzB) for shader +// MIT License +float RoundedRectangleSDF(vec2 fragCoord, vec2 center, vec2 halfSize, vec4 radius) +{ + vec2 fragFromCenter = fragCoord - center; + + // Determine which corner radius to use + radius.xy = (fragFromCenter.y > 0.0) ? radius.xy : radius.zw; + radius.x = (fragFromCenter.x < 0.0) ? radius.x : radius.y; + + // Calculate signed distance field + vec2 dist = abs(fragFromCenter) - halfSize + radius.x; + return min(max(dist.x, dist.y), 0.0) + length(max(dist, 0.0)) - radius.x; +} + +void main() +{ + // Texel color fetching from texture sampler + vec4 texelColor = texture2D(texture0, fragTexCoord); + + // Requires fragment coordinate varying pixels + vec2 fragCoord = gl_FragCoord.xy; + + // Calculate signed distance field for rounded rectangle + vec2 halfSize = rectangle.zw*0.5; + vec2 center = rectangle.xy + halfSize; + float recSDF = RoundedRectangleSDF(fragCoord, center, halfSize, radius); + + // Calculate signed distance field for rectangle shadow + vec2 shadowHalfSize = halfSize*shadowScale; + vec2 shadowCenter = center + shadowOffset; + float shadowSDF = RoundedRectangleSDF(fragCoord, shadowCenter, shadowHalfSize, radius); + + // Caculate alpha factors + float recFactor = smoothstep(1.0, 0.0, recSDF); + float shadowFactor = smoothstep(shadowRadius, 0.0, shadowSDF); + float borderFactor = smoothstep(0.0, 1.0, recSDF + borderThickness)*recFactor; + + // Multiply each color by its respective alpha factor + vec4 recColor = vec4(color.rgb, color.a*recFactor); + vec4 shadowCol = vec4(shadowColor.rgb, shadowColor.a*shadowFactor); + vec4 borderCol = vec4(borderColor.rgb, borderColor.a*borderFactor); + + // Combine the colors varying the order (shadow, rectangle, border) + gl_FragColor = mix(mix(shadowCol, recColor, recColor.a), borderCol, borderCol.a); +} \ No newline at end of file diff --git a/examples/shaders/resources/shaders/glsl330/rounded_rectangle.fs b/examples/shaders/resources/shaders/glsl330/rounded_rectangle.fs new file mode 100644 index 000000000..a96c31fab --- /dev/null +++ b/examples/shaders/resources/shaders/glsl330/rounded_rectangle.fs @@ -0,0 +1,77 @@ +// Note: SDF by Iñigo Quilez is licensed under MIT License + +#version 330 + +// Input vertex attributes (from vertex shader) +in vec2 fragTexCoord; +in vec4 fragColor; + +// Input uniform values +uniform sampler2D texture0; +uniform vec4 colDiffuse; + +// Output fragment color +out vec4 finalColor; + +uniform vec4 rectangle; // Rectangle dimensions (x, y, width, height) +uniform vec4 radius; // Corner radius (top-left, top-right, bottom-left, bottom-right) +uniform vec4 color; + +// Shadow parameters +uniform float shadowRadius; +uniform vec2 shadowOffset; +uniform float shadowScale; +uniform vec4 shadowColor; + +// Border parameters +uniform float borderThickness; +uniform vec4 borderColor; + +// Create a rounded rectangle using signed distance field +// Thanks to Iñigo Quilez (https://www.iquilezles.org/www/articles/distfunctions/distfunctions.htm) +// And thanks to inobelar (https://www.shadertoy.com/view/fsdyzB) for shader +// MIT License +float RoundedRectangleSDF(vec2 fragCoord, vec2 center, vec2 halfSize, vec4 radius) +{ + vec2 fragFromCenter = fragCoord - center; + + // Determine which corner radius to use + radius.xy = (fragFromCenter.y > 0.0) ? radius.xy : radius.zw; + radius.x = (fragFromCenter.x < 0.0) ? radius.x : radius.y; + + // Calculate signed distance field + vec2 dist = abs(fragFromCenter) - halfSize + radius.x; + return min(max(dist.x, dist.y), 0.0) + length(max(dist, 0.0)) - radius.x; +} + +void main() +{ + // Texel color fetching from texture sampler + vec4 texelColor = texture(texture0, fragTexCoord); + + // Requires fragment coordinate in pixels + vec2 fragCoord = gl_FragCoord.xy; + + // Calculate signed distance field for rounded rectangle + vec2 halfSize = rectangle.zw*0.5; + vec2 center = rectangle.xy + halfSize; + float recSDF = RoundedRectangleSDF(fragCoord, center, halfSize, radius); + + // Calculate signed distance field for rectangle shadow + vec2 shadowHalfSize = halfSize*shadowScale; + vec2 shadowCenter = center + shadowOffset; + float shadowSDF = RoundedRectangleSDF(fragCoord, shadowCenter, shadowHalfSize, radius); + + // Caculate alpha factors + float recFactor = smoothstep(1.0, 0.0, recSDF); + float shadowFactor = smoothstep(shadowRadius, 0.0, shadowSDF); + float borderFactor = smoothstep(0.0, 1.0, recSDF + borderThickness)*recFactor; + + // Multiply each color by its respective alpha factor + vec4 recColor = vec4(color.rgb, color.a*recFactor); + vec4 shadowCol = vec4(shadowColor.rgb, shadowColor.a*shadowFactor); + vec4 borderCol = vec4(borderColor.rgb, borderColor.a*borderFactor); + + // Combine the colors in the order (shadow, rectangle, border) + finalColor = mix(mix(shadowCol, recColor, recColor.a), borderCol, borderCol.a); +} \ No newline at end of file diff --git a/examples/shaders/shaders_rounded_rectangle.c b/examples/shaders/shaders_rounded_rectangle.c new file mode 100644 index 000000000..754110d92 --- /dev/null +++ b/examples/shaders/shaders_rounded_rectangle.c @@ -0,0 +1,238 @@ +/******************************************************************************************* +* +* raylib [shaders] example - Rounded Rectangle +* +* Example complexity rating: [★★★☆] 3/4 +* +* Example originally created with raylib 5.5, last time updated with raylib 5.5 +* +* Example contributed by Anstro Pleuton (@anstropleuton) and reviewed by Ramon Santamaria (@raysan5) +* +* Example licensed under an unmodified zlib/libpng license, which is an OSI-certified, +* BSD-like license that allows static linking with closed source software +* +* Copyright (c) 2025-2025 Anstro Pleuton (@anstropleuton) +* +********************************************************************************************/ + +#include "raylib.h" + +#if defined(PLATFORM_DESKTOP) + #define GLSL_VERSION 330 +#else // PLATFORM_ANDROID, PLATFORM_WEB + #define GLSL_VERSION 100 +#endif + +//------------------------------------------------------------------------------------ +// Declare custom Structs +//------------------------------------------------------------------------------------ + +// Rounded rectangle data +typedef struct { + Vector4 cornerRadius; // Individual corner radius (top-left, top-right, bottom-left, bottom-right) + + // Shadow variables + float shadowRadius; + Vector2 shadowOffset; + float shadowScale; + + // Border variables + float borderThickness; // Inner-border thickness + + // Shader locations + int rectangleLoc; + int radiusLoc; + int colorLoc; + int shadowRadiusLoc; + int shadowOffsetLoc; + int shadowScaleLoc; + int shadowColorLoc; + int borderThicknessLoc; + int borderColorLoc; +} RoundedRectangle; + +//------------------------------------------------------------------------------------ +// Module Functions Declaration +//------------------------------------------------------------------------------------ + +// Create a rounded rectangle and set uniform locations +static RoundedRectangle CreateRoundedRectangle(Vector4 cornerRadius, float shadowRadius, Vector2 shadowOffset, float shadowScale, float borderThickness, Shader shader); + +// Update rounded rectangle uniforms +static void UpdateRoundedRectangle(RoundedRectangle rec, Shader shader); + +//------------------------------------------------------------------------------------ +// Program main entry point +//------------------------------------------------------------------------------------ +int main(void) +{ + // Initialization + //-------------------------------------------------------------------------------------- + const int screenWidth = 800; + const int screenHeight = 450; + + const Color rectangleColor = BLUE; + const Color shadowColor = DARKBLUE; + const Color borderColor = SKYBLUE; + + InitWindow(screenWidth, screenHeight, "raylib [shaders] example - Rounded Rectangle"); + + // Load the shader + Shader shader = LoadShader(TextFormat("resources/shaders/glsl%i/base.vs", GLSL_VERSION), + TextFormat("resources/shaders/glsl%i/rounded_rectangle.fs", GLSL_VERSION)); + + // Create a rounded rectangle + RoundedRectangle roundedRectangle = CreateRoundedRectangle( + (Vector4){ 5.0f, 10.0f, 15.0f, 20.0f }, // Corner radius + 20.0f, // Shadow radius + (Vector2){ 0.0f, -5.0f }, // Shadow offset + 0.95f, // Shadow scale + 5.0f, // Border thickness + shader // Shader + ); + + // Update shader uniforms + UpdateRoundedRectangle(roundedRectangle, shader); + + SetTargetFPS(60); + //-------------------------------------------------------------------------------------- + + // Main game loop + while (!WindowShouldClose()) // Detect window close button or ESC key + { + // Draw + //---------------------------------------------------------------------------------- + BeginDrawing(); + + ClearBackground(RAYWHITE); + + // Draw rectangle box with rounded corners using shader + Rectangle rec = { 50, 70, 110, 60 }; + DrawRectangleLines(rec.x - 20, rec.y - 20, rec.width + 40, rec.height + 40, DARKGRAY); + DrawText("Rounded rectangle", rec.x - 20, rec.y - 35, 10, DARKGRAY); + + // Flip Y axis to match shader coordinate system + rec.y = screenHeight - rec.y - rec.height; + SetShaderValue(shader, roundedRectangle.rectangleLoc, (float[]){ rec.x, rec.y, rec.width, rec.height }, SHADER_UNIFORM_VEC4); + + // Only rectangle color + SetShaderValue(shader, roundedRectangle.colorLoc, (float[]) { rectangleColor.r/255.0f, rectangleColor.g/255.0f, rectangleColor.b/255.0f, rectangleColor.a/255.0f }, SHADER_UNIFORM_VEC4); + SetShaderValue(shader, roundedRectangle.shadowColorLoc, (float[]) { 0.0f, 0.0f, 0.0f, 0.0f }, SHADER_UNIFORM_VEC4); + SetShaderValue(shader, roundedRectangle.borderColorLoc, (float[]) { 0.0f, 0.0f, 0.0f, 0.0f }, SHADER_UNIFORM_VEC4); + + BeginShaderMode(shader); + DrawRectangle(0, 0, screenWidth, screenHeight, WHITE); + EndShaderMode(); + + + + // Draw rectangle shadow using shader + rec = (Rectangle){ 50, 200, 110, 60 }; + DrawRectangleLines(rec.x - 20, rec.y - 20, rec.width + 40, rec.height + 40, DARKGRAY); + DrawText("Rounded rectangle shadow", rec.x - 20, rec.y - 35, 10, DARKGRAY); + + rec.y = screenHeight - rec.y - rec.height; + SetShaderValue(shader, roundedRectangle.rectangleLoc, (float[]){ rec.x, rec.y, rec.width, rec.height }, SHADER_UNIFORM_VEC4); + + // Only shadow color + SetShaderValue(shader, roundedRectangle.colorLoc, (float[]) { 0.0f, 0.0f, 0.0f, 0.0f }, SHADER_UNIFORM_VEC4); + SetShaderValue(shader, roundedRectangle.shadowColorLoc, (float[]) { shadowColor.r/255.0f, shadowColor.g/255.0f, shadowColor.b/255.0f, shadowColor.a/255.0f }, SHADER_UNIFORM_VEC4); + SetShaderValue(shader, roundedRectangle.borderColorLoc, (float[]) { 0.0f, 0.0f, 0.0f, 0.0f }, SHADER_UNIFORM_VEC4); + + BeginShaderMode(shader); + DrawRectangle(0, 0, screenWidth, screenHeight, WHITE); + EndShaderMode(); + + + + // Draw rectangle's border using shader + rec = (Rectangle){ 50, 330, 110, 60 }; + DrawRectangleLines(rec.x - 20, rec.y - 20, rec.width + 40, rec.height + 40, DARKGRAY); + DrawText("Rounded rectangle border", rec.x - 20, rec.y - 35, 10, DARKGRAY); + + rec.y = screenHeight - rec.y - rec.height; + SetShaderValue(shader, roundedRectangle.rectangleLoc, (float[]){ rec.x, rec.y, rec.width, rec.height }, SHADER_UNIFORM_VEC4); + + // Only border color + SetShaderValue(shader, roundedRectangle.colorLoc, (float[]) { 0.0f, 0.0f, 0.0f, 0.0f }, SHADER_UNIFORM_VEC4); + SetShaderValue(shader, roundedRectangle.shadowColorLoc, (float[]) { 0.0f, 0.0f, 0.0f, 0.0f }, SHADER_UNIFORM_VEC4); + SetShaderValue(shader, roundedRectangle.borderColorLoc, (float[]) { borderColor.r/255.0f, borderColor.g/255.0f, borderColor.b/255.0f, borderColor.a/255.0f }, SHADER_UNIFORM_VEC4); + + BeginShaderMode(shader); + DrawRectangle(0, 0, screenWidth, screenHeight, WHITE); + EndShaderMode(); + + + + // Draw one more rectangle with all three colors + rec = (Rectangle){ 240, 80, 500, 300 }; + DrawRectangleLines(rec.x - 30, rec.y - 30, rec.width + 60, rec.height + 60, DARKGRAY); + DrawText("Rectangle with all three combined", rec.x - 30, rec.y - 45, 10, DARKGRAY); + + rec.y = screenHeight - rec.y - rec.height; + SetShaderValue(shader, roundedRectangle.rectangleLoc, (float[]){ rec.x, rec.y, rec.width, rec.height }, SHADER_UNIFORM_VEC4); + + // All three colors + SetShaderValue(shader, roundedRectangle.colorLoc, (float[]) { rectangleColor.r/255.0f, rectangleColor.g/255.0f, rectangleColor.b/255.0f, rectangleColor.a/255.0f }, SHADER_UNIFORM_VEC4); + SetShaderValue(shader, roundedRectangle.shadowColorLoc, (float[]) { shadowColor.r/255.0f, shadowColor.g/255.0f, shadowColor.b/255.0f, shadowColor.a/255.0f }, SHADER_UNIFORM_VEC4); + SetShaderValue(shader, roundedRectangle.borderColorLoc, (float[]) { borderColor.r/255.0f, borderColor.g/255.0f, borderColor.b/255.0f, borderColor.a/255.0f }, SHADER_UNIFORM_VEC4); + + BeginShaderMode(shader); + DrawRectangle(0, 0, screenWidth, screenHeight, WHITE); + EndShaderMode(); + + DrawText("(c) Rounded rectangle SDF by Iñigo Quilez. MIT License.", screenWidth - 300, screenHeight - 20, 10, BLACK); + + EndDrawing(); + //---------------------------------------------------------------------------------- + } + + // De-Initialization + //-------------------------------------------------------------------------------------- + UnloadShader(shader); // Unload shader + + CloseWindow(); // Close window and OpenGL context + //-------------------------------------------------------------------------------------- + + return 0; +} + +//------------------------------------------------------------------------------------ +// Module Functions Definitions +//------------------------------------------------------------------------------------ + +// Create a rounded rectangle and set uniform locations +RoundedRectangle CreateRoundedRectangle(Vector4 cornerRadius, float shadowRadius, Vector2 shadowOffset, float shadowScale, float borderThickness, Shader shader) +{ + RoundedRectangle rec; + rec.cornerRadius = cornerRadius; + rec.shadowRadius = shadowRadius; + rec.shadowOffset = shadowOffset; + rec.shadowScale = shadowScale; + rec.borderThickness = borderThickness; + + // Get shader uniform locations + rec.rectangleLoc = GetShaderLocation(shader, "rectangle"); + rec.radiusLoc = GetShaderLocation(shader, "radius"); + rec.colorLoc = GetShaderLocation(shader, "color"); + rec.shadowRadiusLoc = GetShaderLocation(shader, "shadowRadius"); + rec.shadowOffsetLoc = GetShaderLocation(shader, "shadowOffset"); + rec.shadowScaleLoc = GetShaderLocation(shader, "shadowScale"); + rec.shadowColorLoc = GetShaderLocation(shader, "shadowColor"); + rec.borderThicknessLoc = GetShaderLocation(shader, "borderThickness"); + rec.borderColorLoc = GetShaderLocation(shader, "borderColor"); + + UpdateRoundedRectangle(rec, shader); + + return rec; +} + +// Update rounded rectangle uniforms +void UpdateRoundedRectangle(RoundedRectangle rec, Shader shader) +{ + SetShaderValue(shader, rec.radiusLoc, (float[]){ rec.cornerRadius.x, rec.cornerRadius.y, rec.cornerRadius.z, rec.cornerRadius.w }, SHADER_UNIFORM_VEC4); + SetShaderValue(shader, rec.shadowRadiusLoc, &rec.shadowRadius, SHADER_UNIFORM_FLOAT); + SetShaderValue(shader, rec.shadowOffsetLoc, (float[]){ rec.shadowOffset.x, rec.shadowOffset.y }, SHADER_UNIFORM_VEC2); + SetShaderValue(shader, rec.shadowScaleLoc, &rec.shadowScale, SHADER_UNIFORM_FLOAT); + SetShaderValue(shader, rec.borderThicknessLoc, &rec.borderThickness, SHADER_UNIFORM_FLOAT); +} diff --git a/examples/shaders/shaders_rounded_rectangle.png b/examples/shaders/shaders_rounded_rectangle.png new file mode 100644 index 0000000000000000000000000000000000000000..0d8c2b5f41e887c4558c2c8c594912ca797e2036 GIT binary patch literal 13936 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 + + + + Debug.DLL + Win32 + + + Debug.DLL + x64 + + + Debug + Win32 + + + Debug + x64 + + + Release.DLL + Win32 + + + Release.DLL + x64 + + + Release + Win32 + + + Release + x64 + + + + {D3493FFE-8873-4C53-8F6C-74DEF78EA3C4} + Win32Proj + shaders_rounded_rectangle + 10.0 + shaders_rounded_rectangle + + + + Application + true + $(DefaultPlatformToolset) + Unicode + + + Application + true + $(DefaultPlatformToolset) + Unicode + + + Application + true + $(DefaultPlatformToolset) + Unicode + + + Application + true + $(DefaultPlatformToolset) + Unicode + + + Application + false + $(DefaultPlatformToolset) + true + Unicode + + + Application + false + $(DefaultPlatformToolset) + true + Unicode + + + Application + false + $(DefaultPlatformToolset) + true + Unicode + + + Application + false + $(DefaultPlatformToolset) + true + Unicode + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + true + $(SolutionDir)\build\$(ProjectName)\bin\$(Platform)\$(Configuration)\ + $(SolutionDir)\build\$(ProjectName)\obj\$(Platform)\$(Configuration)\ + + + true + $(SolutionDir)\build\$(ProjectName)\bin\$(Platform)\$(Configuration)\ + $(SolutionDir)\build\$(ProjectName)\obj\$(Platform)\$(Configuration)\ + + + true + $(SolutionDir)\build\$(ProjectName)\bin\$(Platform)\$(Configuration)\ + $(SolutionDir)\build\$(ProjectName)\obj\$(Platform)\$(Configuration)\ + + + true + $(SolutionDir)\build\$(ProjectName)\bin\$(Platform)\$(Configuration)\ + $(SolutionDir)\build\$(ProjectName)\obj\$(Platform)\$(Configuration)\ + + + false + $(SolutionDir)\build\$(ProjectName)\bin\$(Platform)\$(Configuration)\ + $(SolutionDir)\build\$(ProjectName)\obj\$(Platform)\$(Configuration)\ + + + false + $(SolutionDir)\build\$(ProjectName)\bin\$(Platform)\$(Configuration)\ + $(SolutionDir)\build\$(ProjectName)\obj\$(Platform)\$(Configuration)\ + + + false + $(SolutionDir)\build\$(ProjectName)\bin\$(Platform)\$(Configuration)\ + $(SolutionDir)\build\$(ProjectName)\obj\$(Platform)\$(Configuration)\ + + + false + $(SolutionDir)\build\$(ProjectName)\bin\$(Platform)\$(Configuration)\ + $(SolutionDir)\build\$(ProjectName)\obj\$(Platform)\$(Configuration)\ + + + $(SolutionDir)..\..\examples\shaders + WindowsLocalDebugger + + + $(SolutionDir)..\..\examples\shaders + WindowsLocalDebugger + + + $(SolutionDir)..\..\examples\shaders + WindowsLocalDebugger + + + $(SolutionDir)..\..\examples\shaders + WindowsLocalDebugger + + + $(SolutionDir)..\..\examples\shaders + WindowsLocalDebugger + + + $(SolutionDir)..\..\examples\shaders + WindowsLocalDebugger + + + $(SolutionDir)..\..\examples\shaders + WindowsLocalDebugger + + + $(SolutionDir)..\..\examples\shaders + WindowsLocalDebugger + + + + + + Level3 + Disabled + WIN32;_DEBUG;_CONSOLE;PLATFORM_DESKTOP;%(PreprocessorDefinitions) + CompileAsC + $(SolutionDir)..\..\src;%(AdditionalIncludeDirectories) + + + Console + true + $(SolutionDir)\build\raylib\bin\$(Platform)\$(Configuration)\ + raylib.lib;opengl32.lib;kernel32.lib;user32.lib;gdi32.lib;winmm.lib;winspool.lib;comdlg32.lib;advapi32.lib;shell32.lib;ole32.lib;oleaut32.lib;uuid.lib;odbc32.lib;odbccp32.lib;%(AdditionalDependencies) + + + + + + + Level3 + Disabled + WIN32;_DEBUG;_CONSOLE;PLATFORM_DESKTOP;%(PreprocessorDefinitions) + CompileAsC + $(SolutionDir)..\..\src;%(AdditionalIncludeDirectories) + /FS %(AdditionalOptions) + + + Console + true + $(SolutionDir)\build\raylib\bin\$(Platform)\$(Configuration)\ + raylib.lib;opengl32.lib;kernel32.lib;user32.lib;gdi32.lib;winmm.lib;winspool.lib;comdlg32.lib;advapi32.lib;shell32.lib;ole32.lib;oleaut32.lib;uuid.lib;odbc32.lib;odbccp32.lib;%(AdditionalDependencies) + + + + + + + Level3 + Disabled + WIN32;_DEBUG;_CONSOLE;PLATFORM_DESKTOP;%(PreprocessorDefinitions) + CompileAsC + $(SolutionDir)..\..\src;%(AdditionalIncludeDirectories) + + + Console + true + $(SolutionDir)\build\raylib\bin\$(Platform)\$(Configuration)\ + raylib.lib;opengl32.lib;kernel32.lib;user32.lib;gdi32.lib;winmm.lib;winspool.lib;comdlg32.lib;advapi32.lib;shell32.lib;ole32.lib;oleaut32.lib;uuid.lib;odbc32.lib;odbccp32.lib;%(AdditionalDependencies) + + + xcopy /y /d "$(SolutionDir)\build\raylib\bin\$(Platform)\$(Configuration)\raylib.dll" "$(SolutionDir)\build\$(ProjectName)\bin\$(Platform)\$(Configuration)" + Copy Debug DLL to output directory + + + + + + + Level3 + Disabled + WIN32;_DEBUG;_CONSOLE;PLATFORM_DESKTOP;%(PreprocessorDefinitions) + CompileAsC + $(SolutionDir)..\..\src;%(AdditionalIncludeDirectories) + + + Console + true + $(SolutionDir)\build\raylib\bin\$(Platform)\$(Configuration)\ + raylib.lib;opengl32.lib;kernel32.lib;user32.lib;gdi32.lib;winmm.lib;winspool.lib;comdlg32.lib;advapi32.lib;shell32.lib;ole32.lib;oleaut32.lib;uuid.lib;odbc32.lib;odbccp32.lib;%(AdditionalDependencies) + + + xcopy /y /d "$(SolutionDir)\build\raylib\bin\$(Platform)\$(Configuration)\raylib.dll" "$(SolutionDir)\build\$(ProjectName)\bin\$(Platform)\$(Configuration)" + Copy Debug DLL to output directory + + + + + Level3 + + + MaxSpeed + true + true + WIN32;NDEBUG;_CONSOLE;%(PreprocessorDefinitions);PLATFORM_DESKTOP + $(SolutionDir)..\..\src;%(AdditionalIncludeDirectories) + CompileAsC + true + + + Console + true + true + true + raylib.lib;opengl32.lib;kernel32.lib;user32.lib;gdi32.lib;winmm.lib;winspool.lib;comdlg32.lib;advapi32.lib;shell32.lib;ole32.lib;oleaut32.lib;uuid.lib;odbc32.lib;odbccp32.lib;%(AdditionalDependencies) + $(SolutionDir)\build\raylib\bin\$(Platform)\$(Configuration)\ + + + + + Level3 + + + MaxSpeed + true + true + WIN32;NDEBUG;_CONSOLE;%(PreprocessorDefinitions);PLATFORM_DESKTOP + $(SolutionDir)..\..\src;%(AdditionalIncludeDirectories) + CompileAsC + true + + + Console + true + true + true + raylib.lib;opengl32.lib;kernel32.lib;user32.lib;gdi32.lib;winmm.lib;winspool.lib;comdlg32.lib;advapi32.lib;shell32.lib;ole32.lib;oleaut32.lib;uuid.lib;odbc32.lib;odbccp32.lib;%(AdditionalDependencies) + $(SolutionDir)\build\raylib\bin\$(Platform)\$(Configuration)\ + + + + + Level3 + + + MaxSpeed + true + true + WIN32;NDEBUG;_CONSOLE;%(PreprocessorDefinitions);PLATFORM_DESKTOP + $(SolutionDir)..\..\src;%(AdditionalIncludeDirectories) + CompileAsC + true + + + Console + true + true + true + raylib.lib;opengl32.lib;kernel32.lib;user32.lib;gdi32.lib;winmm.lib;winspool.lib;comdlg32.lib;advapi32.lib;shell32.lib;ole32.lib;oleaut32.lib;uuid.lib;odbc32.lib;odbccp32.lib;%(AdditionalDependencies) + $(SolutionDir)\build\raylib\bin\$(Platform)\$(Configuration)\ + + + xcopy /y /d "$(SolutionDir)\build\raylib\bin\$(Platform)\$(Configuration)\raylib.dll" "$(SolutionDir)\build\$(ProjectName)\bin\$(Platform)\$(Configuration)" + + + Copy Release DLL to output directory + + + + + Level3 + + + MaxSpeed + true + true + WIN32;NDEBUG;_CONSOLE;%(PreprocessorDefinitions);PLATFORM_DESKTOP + $(SolutionDir)..\..\src;%(AdditionalIncludeDirectories) + CompileAsC + true + + + Console + true + true + true + raylib.lib;opengl32.lib;kernel32.lib;user32.lib;gdi32.lib;winmm.lib;winspool.lib;comdlg32.lib;advapi32.lib;shell32.lib;ole32.lib;oleaut32.lib;uuid.lib;odbc32.lib;odbccp32.lib;%(AdditionalDependencies) + $(SolutionDir)\build\raylib\bin\$(Platform)\$(Configuration)\ + + + xcopy /y /d "$(SolutionDir)\build\raylib\bin\$(Platform)\$(Configuration)\raylib.dll" "$(SolutionDir)\build\$(ProjectName)\bin\$(Platform)\$(Configuration)" + + + Copy Release DLL to output directory + + + + + + + + {e89d61ac-55de-4482-afd4-df7242ebc859} + + + + + + \ No newline at end of file diff --git a/projects/VS2022/raylib.sln b/projects/VS2022/raylib.sln index f4f0600e1..8f3323bc8 100644 --- a/projects/VS2022/raylib.sln +++ b/projects/VS2022/raylib.sln @@ -307,6 +307,8 @@ Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "models_bone_socket", "examp EndProject Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "shaders_vertex_displacement", "examples\shaders_vertex_displacement.vcxproj", "{CCA63A76-D9FC-4130-9F67-4D97F9770D53}" EndProject +Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "shaders_rounded_rectangle", "examples\shaders_rounded_rectangle.vcxproj", "{D3493FFE-8873-4C53-8F6C-74DEF78EA3C4}" +EndProject Global GlobalSection(SolutionConfigurationPlatforms) = preSolution Debug.DLL|x64 = Debug.DLL|x64 @@ -2615,6 +2617,22 @@ Global {CCA63A76-D9FC-4130-9F67-4D97F9770D53}.Release|x64.Build.0 = Release|x64 {CCA63A76-D9FC-4130-9F67-4D97F9770D53}.Release|x86.ActiveCfg = Release|Win32 {CCA63A76-D9FC-4130-9F67-4D97F9770D53}.Release|x86.Build.0 = Release|Win32 + {D3493FFE-8873-4C53-8F6C-74DEF78EA3C4}.Debug.DLL|x64.ActiveCfg = Debug.DLL|x64 + {D3493FFE-8873-4C53-8F6C-74DEF78EA3C4}.Debug.DLL|x64.Build.0 = Debug.DLL|x64 + {D3493FFE-8873-4C53-8F6C-74DEF78EA3C4}.Debug.DLL|x86.ActiveCfg = Debug.DLL|Win32 + {D3493FFE-8873-4C53-8F6C-74DEF78EA3C4}.Debug.DLL|x86.Build.0 = Debug.DLL|Win32 + {D3493FFE-8873-4C53-8F6C-74DEF78EA3C4}.Debug|x64.ActiveCfg = Debug|x64 + {D3493FFE-8873-4C53-8F6C-74DEF78EA3C4}.Debug|x64.Build.0 = Debug|x64 + {D3493FFE-8873-4C53-8F6C-74DEF78EA3C4}.Debug|x86.ActiveCfg = Debug|Win32 + {D3493FFE-8873-4C53-8F6C-74DEF78EA3C4}.Debug|x86.Build.0 = Debug|Win32 + {D3493FFE-8873-4C53-8F6C-74DEF78EA3C4}.Release.DLL|x64.ActiveCfg = Release.DLL|x64 + {D3493FFE-8873-4C53-8F6C-74DEF78EA3C4}.Release.DLL|x64.Build.0 = Release.DLL|x64 + {D3493FFE-8873-4C53-8F6C-74DEF78EA3C4}.Release.DLL|x86.ActiveCfg = Release.DLL|Win32 + {D3493FFE-8873-4C53-8F6C-74DEF78EA3C4}.Release.DLL|x86.Build.0 = Release.DLL|Win32 + {D3493FFE-8873-4C53-8F6C-74DEF78EA3C4}.Release|x64.ActiveCfg = Release|x64 + {D3493FFE-8873-4C53-8F6C-74DEF78EA3C4}.Release|x64.Build.0 = Release|x64 + {D3493FFE-8873-4C53-8F6C-74DEF78EA3C4}.Release|x86.ActiveCfg = Release|Win32 + {D3493FFE-8873-4C53-8F6C-74DEF78EA3C4}.Release|x86.Build.0 = Release|Win32 EndGlobalSection GlobalSection(SolutionProperties) = preSolution HideSolutionNode = FALSE @@ -2771,6 +2789,7 @@ Global {0981CA28-E4A5-4DF1-987F-A41D09131EFC} = {6C82BAAE-BDDF-457D-8FA8-7E2490B07035} {3A7FE53D-35F7-49DC-9C9A-A5204A53523F} = {AF5BEC5C-1F2B-4DA8-B12D-D09FE569237C} {CCA63A76-D9FC-4130-9F67-4D97F9770D53} = {5317807F-61D4-4E0F-B6DC-2D9F12621ED9} + {D3493FFE-8873-4C53-8F6C-74DEF78EA3C4} = {5317807F-61D4-4E0F-B6DC-2D9F12621ED9} EndGlobalSection GlobalSection(ExtensibilityGlobals) = postSolution SolutionGuid = {E926C768-6307-4423-A1EC-57E95B1FAB29} From 49d37b035f6d764d6bafdc835b5496fded3edcc0 Mon Sep 17 00:00:00 2001 From: Nikolas Date: Fri, 24 Jan 2025 10:41:32 +0100 Subject: [PATCH 2/3] [rtexture] Cubemap mipmap loading improvements (#4721) * [rtextures] Only build cubemap mipmaps when necessary * [rtextures] Assign correct mipmap count to cubemaps --- src/rtextures.c | 9 ++++++--- 1 file changed, 6 insertions(+), 3 deletions(-) diff --git a/src/rtextures.c b/src/rtextures.c index 9dfc24738..9d3f51f34 100644 --- a/src/rtextures.c +++ b/src/rtextures.c @@ -4208,8 +4208,11 @@ TextureCubemap LoadTextureCubemap(Image image, int layout) Image mipmapped = ImageCopy(image); #if defined(SUPPORT_IMAGE_MANIPULATION) - ImageMipmaps(&mipmapped); - ImageMipmaps(&faces); + if (image.mipmaps > 1) + { + ImageMipmaps(&mipmapped); + ImageMipmaps(&faces); + } #endif // NOTE: Image formatting does not work with compressed textures @@ -4226,7 +4229,7 @@ TextureCubemap LoadTextureCubemap(Image image, int layout) if (cubemap.id != 0) { cubemap.format = faces.format; - cubemap.mipmaps = 1; + cubemap.mipmaps = faces.mipmaps; } else TRACELOG(LOG_WARNING, "IMAGE: Failed to load cubemap image"); From 139de05e9d68145422b0d625c23a7d7505b8762b Mon Sep 17 00:00:00 2001 From: Asdqwe Date: Sat, 25 Jan 2025 16:49:42 -0300 Subject: [PATCH 3/3] [rcore] [SDL2] Fix gamepad event handling by adding joystick instance id tracking (#4724) * Fix gamepad SDL_JOYDEVICEADDED and SDL_JOYDEVICEREMOVED event handling for PLATFORM_DESKTOP_SDL by adding joystick instance id tracking * Fix gamepad button handling --- src/platforms/rcore_desktop_sdl.c | 77 ++++++++++++++++++++++--------- 1 file changed, 55 insertions(+), 22 deletions(-) diff --git a/src/platforms/rcore_desktop_sdl.c b/src/platforms/rcore_desktop_sdl.c index 33c182a70..f248e2614 100644 --- a/src/platforms/rcore_desktop_sdl.c +++ b/src/platforms/rcore_desktop_sdl.c @@ -82,6 +82,7 @@ typedef struct { SDL_GLContext glContext; SDL_GameController *gamepad[MAX_GAMEPADS]; + SDL_JoystickID gamepadId[MAX_GAMEPADS]; // Joystick instance ids SDL_Cursor *cursor; bool cursorRelative; } PlatformData; @@ -1658,11 +1659,12 @@ void PollInputEvents(void) // Check gamepad events case SDL_JOYDEVICEADDED: { - int jid = event.jdevice.which; + int jid = event.jdevice.which; // Joystick device index if (!CORE.Input.Gamepad.ready[jid] && (jid < MAX_GAMEPADS)) { platform.gamepad[jid] = SDL_GameControllerOpen(jid); + platform.gamepadId[jid] = SDL_JoystickInstanceID(SDL_GameControllerGetJoystick(platform.gamepad[jid])); if (platform.gamepad[jid]) { @@ -1681,14 +1683,18 @@ void PollInputEvents(void) } break; case SDL_JOYDEVICEREMOVED: { - int jid = event.jdevice.which; + int jid = event.jdevice.which; // Joystick instance id - if (jid == SDL_JoystickInstanceID(SDL_GameControllerGetJoystick(platform.gamepad[jid]))) + for (int i = 0; i < MAX_GAMEPADS; i++) { - SDL_GameControllerClose(platform.gamepad[jid]); - platform.gamepad[jid] = SDL_GameControllerOpen(0); - CORE.Input.Gamepad.ready[jid] = false; - memset(CORE.Input.Gamepad.name[jid], 0, MAX_GAMEPAD_NAME_LENGTH); + if (platform.gamepadId[i] == jid) + { + SDL_GameControllerClose(platform.gamepad[i]); + CORE.Input.Gamepad.ready[i] = false; + memset(CORE.Input.Gamepad.name[i], 0, MAX_GAMEPAD_NAME_LENGTH); + platform.gamepadId[i] = -1; + break; + } } } break; case SDL_CONTROLLERBUTTONDOWN: @@ -1721,8 +1727,15 @@ void PollInputEvents(void) if (button >= 0) { - CORE.Input.Gamepad.currentButtonState[event.jbutton.which][button] = 1; - CORE.Input.Gamepad.lastButtonPressed = button; + for (int i = 0; i < MAX_GAMEPADS; i++) + { + if (platform.gamepadId[i] == event.jbutton.which) + { + CORE.Input.Gamepad.currentButtonState[i][button] = 1; + CORE.Input.Gamepad.lastButtonPressed = button; + break; + } + } } } break; case SDL_CONTROLLERBUTTONUP: @@ -1755,8 +1768,15 @@ void PollInputEvents(void) if (button >= 0) { - CORE.Input.Gamepad.currentButtonState[event.jbutton.which][button] = 0; - if (CORE.Input.Gamepad.lastButtonPressed == button) CORE.Input.Gamepad.lastButtonPressed = 0; + for (int i = 0; i < MAX_GAMEPADS; i++) + { + if (platform.gamepadId[i] == event.jbutton.which) + { + CORE.Input.Gamepad.currentButtonState[i][button] = 0; + if (CORE.Input.Gamepad.lastButtonPressed == button) CORE.Input.Gamepad.lastButtonPressed = 0; + break; + } + } } } break; case SDL_CONTROLLERAXISMOTION: @@ -1776,18 +1796,25 @@ void PollInputEvents(void) if (axis >= 0) { - // SDL axis value range is -32768 to 32767, we normalize it to RayLib's -1.0 to 1.0f range - float value = event.jaxis.value/(float)32767; - CORE.Input.Gamepad.axisState[event.jaxis.which][axis] = value; - - // Register button state for triggers in addition to their axes - if ((axis == GAMEPAD_AXIS_LEFT_TRIGGER) || (axis == GAMEPAD_AXIS_RIGHT_TRIGGER)) + for (int i = 0; i < MAX_GAMEPADS; i++) { - int button = (axis == GAMEPAD_AXIS_LEFT_TRIGGER)? GAMEPAD_BUTTON_LEFT_TRIGGER_2 : GAMEPAD_BUTTON_RIGHT_TRIGGER_2; - int pressed = (value > 0.1f); - CORE.Input.Gamepad.currentButtonState[event.jaxis.which][button] = pressed; - if (pressed) CORE.Input.Gamepad.lastButtonPressed = button; - else if (CORE.Input.Gamepad.lastButtonPressed == button) CORE.Input.Gamepad.lastButtonPressed = 0; + if (platform.gamepadId[i] == event.jaxis.which) + { + // SDL axis value range is -32768 to 32767, we normalize it to RayLib's -1.0 to 1.0f range + float value = event.jaxis.value/(float)32767; + CORE.Input.Gamepad.axisState[i][axis] = value; + + // Register button state for triggers in addition to their axes + if ((axis == GAMEPAD_AXIS_LEFT_TRIGGER) || (axis == GAMEPAD_AXIS_RIGHT_TRIGGER)) + { + int button = (axis == GAMEPAD_AXIS_LEFT_TRIGGER)? GAMEPAD_BUTTON_LEFT_TRIGGER_2 : GAMEPAD_BUTTON_RIGHT_TRIGGER_2; + int pressed = (value > 0.1f); + CORE.Input.Gamepad.currentButtonState[i][button] = pressed; + if (pressed) CORE.Input.Gamepad.lastButtonPressed = button; + else if (CORE.Input.Gamepad.lastButtonPressed == button) CORE.Input.Gamepad.lastButtonPressed = 0; + } + break; + } } } } break; @@ -1967,9 +1994,15 @@ int InitPlatform(void) // Initialize input events system //---------------------------------------------------------------------------- // Initialize gamepads + for (int i = 0; i < MAX_GAMEPADS; i++) + { + platform.gamepadId[i] = -1; // Set all gamepad initial instance ids as invalid to not conflict with instance id zero + } + for (int i = 0; (i < SDL_NumJoysticks()) && (i < MAX_GAMEPADS); i++) { platform.gamepad[i] = SDL_GameControllerOpen(i); + platform.gamepadId[i] = SDL_JoystickInstanceID(SDL_GameControllerGetJoystick(platform.gamepad[i])); if (platform.gamepad[i]) {