/*******************************************************************************************
							 | 
						|
								*
							 | 
						|
								*   raylib [models] example - Mesh picking in 3d mode, ground plane, triangle, mesh
							 | 
						|
								*
							 | 
						|
								*   Example complexity rating: [★★★☆] 3/4
							 | 
						|
								*
							 | 
						|
								*   Example originally created with raylib 1.7, last time updated with raylib 4.0
							 | 
						|
								*
							 | 
						|
								*   Example contributed by Joel Davis (@joeld42) 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) 2017-2025 Joel Davis (@joeld42) and Ramon Santamaria (@raysan5)
							 | 
						|
								*
							 | 
						|
								********************************************************************************************/
							 | 
						|
								
							 | 
						|
								#include "raylib.h"
							 | 
						|
								#include "raymath.h"
							 | 
						|
								
							 | 
						|
								#define FLT_MAX     340282346638528859811704183484516925440.0f     // Maximum value of a float, from bit pattern 01111111011111111111111111111111
							 | 
						|
								
							 | 
						|
								//------------------------------------------------------------------------------------
							 | 
						|
								// Program main entry point
							 | 
						|
								//------------------------------------------------------------------------------------
							 | 
						|
								int main(void)
							 | 
						|
								{
							 | 
						|
								    // Initialization
							 | 
						|
								    //--------------------------------------------------------------------------------------
							 | 
						|
								    const int screenWidth = 800;
							 | 
						|
								    const int screenHeight = 450;
							 | 
						|
								
							 | 
						|
								    InitWindow(screenWidth, screenHeight, "raylib [models] example - mesh picking");
							 | 
						|
								
							 | 
						|
								    // Define the camera to look into our 3d world
							 | 
						|
								    Camera camera = { 0 };
							 | 
						|
								    camera.position = (Vector3){ 20.0f, 20.0f, 20.0f }; // Camera position
							 | 
						|
								    camera.target = (Vector3){ 0.0f, 8.0f, 0.0f };      // Camera looking at point
							 | 
						|
								    camera.up = (Vector3){ 0.0f, 1.6f, 0.0f };          // Camera up vector (rotation towards target)
							 | 
						|
								    camera.fovy = 45.0f;                                // Camera field-of-view Y
							 | 
						|
								    camera.projection = CAMERA_PERSPECTIVE;             // Camera projection type
							 | 
						|
								
							 | 
						|
								    Ray ray = { 0 };        // Picking ray
							 | 
						|
								
							 | 
						|
								    Model tower = LoadModel("resources/models/obj/turret.obj");                 // Load OBJ model
							 | 
						|
								    Texture2D texture = LoadTexture("resources/models/obj/turret_diffuse.png"); // Load model texture
							 | 
						|
								    tower.materials[0].maps[MATERIAL_MAP_DIFFUSE].texture = texture;            // Set model diffuse texture
							 | 
						|
								
							 | 
						|
								    Vector3 towerPos = { 0.0f, 0.0f, 0.0f };                        // Set model position
							 | 
						|
								    BoundingBox towerBBox = GetMeshBoundingBox(tower.meshes[0]);    // Get mesh bounding box
							 | 
						|
								
							 | 
						|
								    // Ground quad
							 | 
						|
								    Vector3 g0 = (Vector3){ -50.0f, 0.0f, -50.0f };
							 | 
						|
								    Vector3 g1 = (Vector3){ -50.0f, 0.0f,  50.0f };
							 | 
						|
								    Vector3 g2 = (Vector3){  50.0f, 0.0f,  50.0f };
							 | 
						|
								    Vector3 g3 = (Vector3){  50.0f, 0.0f, -50.0f };
							 | 
						|
								
							 | 
						|
								    // Test triangle
							 | 
						|
								    Vector3 ta = (Vector3){ -25.0f, 0.5f, 0.0f };
							 | 
						|
								    Vector3 tb = (Vector3){ -4.0f, 2.5f, 1.0f };
							 | 
						|
								    Vector3 tc = (Vector3){ -8.0f, 6.5f, 0.0f };
							 | 
						|
								
							 | 
						|
								    Vector3 bary = { 0.0f, 0.0f, 0.0f };
							 | 
						|
								
							 | 
						|
								    // Test sphere
							 | 
						|
								    Vector3 sp = (Vector3){ -30.0f, 5.0f, 5.0f };
							 | 
						|
								    float sr = 4.0f;
							 | 
						|
								
							 | 
						|
								    SetTargetFPS(60);                   // Set our game to run at 60 frames-per-second
							 | 
						|
								    //--------------------------------------------------------------------------------------
							 | 
						|
								    // Main game loop
							 | 
						|
								    while (!WindowShouldClose())        // Detect window close button or ESC key
							 | 
						|
								    {
							 | 
						|
								        // Update
							 | 
						|
								        //----------------------------------------------------------------------------------
							 | 
						|
								        if (IsCursorHidden()) UpdateCamera(&camera, CAMERA_FIRST_PERSON);          // Update camera
							 | 
						|
								
							 | 
						|
								        // Toggle camera controls
							 | 
						|
								        if (IsMouseButtonPressed(MOUSE_BUTTON_RIGHT))
							 | 
						|
								        {
							 | 
						|
								            if (IsCursorHidden()) EnableCursor();
							 | 
						|
								            else DisableCursor();
							 | 
						|
								        }
							 | 
						|
								
							 | 
						|
								        // Display information about closest hit
							 | 
						|
								        RayCollision collision = { 0 };
							 | 
						|
								        char *hitObjectName = "None";
							 | 
						|
								        collision.distance = FLT_MAX;
							 | 
						|
								        collision.hit = false;
							 | 
						|
								        Color cursorColor = WHITE;
							 | 
						|
								
							 | 
						|
								        // Get ray and test against objects
							 | 
						|
								        ray = GetScreenToWorldRay(GetMousePosition(), camera);
							 | 
						|
								
							 | 
						|
								        // Check ray collision against ground quad
							 | 
						|
								        RayCollision groundHitInfo = GetRayCollisionQuad(ray, g0, g1, g2, g3);
							 | 
						|
								
							 | 
						|
								        if ((groundHitInfo.hit) && (groundHitInfo.distance < collision.distance))
							 | 
						|
								        {
							 | 
						|
								            collision = groundHitInfo;
							 | 
						|
								            cursorColor = GREEN;
							 | 
						|
								            hitObjectName = "Ground";
							 | 
						|
								        }
							 | 
						|
								
							 | 
						|
								        // Check ray collision against test triangle
							 | 
						|
								        RayCollision triHitInfo = GetRayCollisionTriangle(ray, ta, tb, tc);
							 | 
						|
								
							 | 
						|
								        if ((triHitInfo.hit) && (triHitInfo.distance < collision.distance))
							 | 
						|
								        {
							 | 
						|
								            collision = triHitInfo;
							 | 
						|
								            cursorColor = PURPLE;
							 | 
						|
								            hitObjectName = "Triangle";
							 | 
						|
								
							 | 
						|
								            bary = Vector3Barycenter(collision.point, ta, tb, tc);
							 | 
						|
								        }
							 | 
						|
								
							 | 
						|
								        // Check ray collision against test sphere
							 | 
						|
								        RayCollision sphereHitInfo = GetRayCollisionSphere(ray, sp, sr);
							 | 
						|
								
							 | 
						|
								        if ((sphereHitInfo.hit) && (sphereHitInfo.distance < collision.distance))
							 | 
						|
								        {
							 | 
						|
								            collision = sphereHitInfo;
							 | 
						|
								            cursorColor = ORANGE;
							 | 
						|
								            hitObjectName = "Sphere";
							 | 
						|
								        }
							 | 
						|
								
							 | 
						|
								        // Check ray collision against bounding box first, before trying the full ray-mesh test
							 | 
						|
								        RayCollision boxHitInfo = GetRayCollisionBox(ray, towerBBox);
							 | 
						|
								
							 | 
						|
								        if ((boxHitInfo.hit) && (boxHitInfo.distance < collision.distance))
							 | 
						|
								        {
							 | 
						|
								            collision = boxHitInfo;
							 | 
						|
								            cursorColor = ORANGE;
							 | 
						|
								            hitObjectName = "Box";
							 | 
						|
								
							 | 
						|
								            // Check ray collision against model meshes
							 | 
						|
								            RayCollision meshHitInfo = { 0 };
							 | 
						|
								            for (int m = 0; m < tower.meshCount; m++)
							 | 
						|
								            {
							 | 
						|
								                // NOTE: We consider the model.transform for the collision check but 
							 | 
						|
								                // it can be checked against any transform Matrix, used when checking against same
							 | 
						|
								                // model drawn multiple times with multiple transforms
							 | 
						|
								                meshHitInfo = GetRayCollisionMesh(ray, tower.meshes[m], tower.transform);
							 | 
						|
								                if (meshHitInfo.hit)
							 | 
						|
								                {
							 | 
						|
								                    // Save the closest hit mesh
							 | 
						|
								                    if ((!collision.hit) || (collision.distance > meshHitInfo.distance)) collision = meshHitInfo;
							 | 
						|
								                    
							 | 
						|
								                    break;  // Stop once one mesh collision is detected, the colliding mesh is m
							 | 
						|
								                }
							 | 
						|
								            }
							 | 
						|
								
							 | 
						|
								            if (meshHitInfo.hit)
							 | 
						|
								            {
							 | 
						|
								                collision = meshHitInfo;
							 | 
						|
								                cursorColor = ORANGE;
							 | 
						|
								                hitObjectName = "Mesh";
							 | 
						|
								            }
							 | 
						|
								        }
							 | 
						|
								        //----------------------------------------------------------------------------------
							 | 
						|
								
							 | 
						|
								        // Draw
							 | 
						|
								        //----------------------------------------------------------------------------------
							 | 
						|
								        BeginDrawing();
							 | 
						|
								
							 | 
						|
								            ClearBackground(RAYWHITE);
							 | 
						|
								
							 | 
						|
								            BeginMode3D(camera);
							 | 
						|
								
							 | 
						|
								                // Draw the tower
							 | 
						|
								                // WARNING: If scale is different than 1.0f,
							 | 
						|
								                // not considered by GetRayCollisionModel()
							 | 
						|
								                DrawModel(tower, towerPos, 1.0f, WHITE);
							 | 
						|
								
							 | 
						|
								                // Draw the test triangle
							 | 
						|
								                DrawLine3D(ta, tb, PURPLE);
							 | 
						|
								                DrawLine3D(tb, tc, PURPLE);
							 | 
						|
								                DrawLine3D(tc, ta, PURPLE);
							 | 
						|
								
							 | 
						|
								                // Draw the test sphere
							 | 
						|
								                DrawSphereWires(sp, sr, 8, 8, PURPLE);
							 | 
						|
								
							 | 
						|
								                // Draw the mesh bbox if we hit it
							 | 
						|
								                if (boxHitInfo.hit) DrawBoundingBox(towerBBox, LIME);
							 | 
						|
								
							 | 
						|
								                // If we hit something, draw the cursor at the hit point
							 | 
						|
								                if (collision.hit)
							 | 
						|
								                {
							 | 
						|
								                    DrawCube(collision.point, 0.3f, 0.3f, 0.3f, cursorColor);
							 | 
						|
								                    DrawCubeWires(collision.point, 0.3f, 0.3f, 0.3f, RED);
							 | 
						|
								
							 | 
						|
								                    Vector3 normalEnd;
							 | 
						|
								                    normalEnd.x = collision.point.x + collision.normal.x;
							 | 
						|
								                    normalEnd.y = collision.point.y + collision.normal.y;
							 | 
						|
								                    normalEnd.z = collision.point.z + collision.normal.z;
							 | 
						|
								
							 | 
						|
								                    DrawLine3D(collision.point, normalEnd, RED);
							 | 
						|
								                }
							 | 
						|
								
							 | 
						|
								                DrawRay(ray, MAROON);
							 | 
						|
								
							 | 
						|
								                DrawGrid(10, 10.0f);
							 | 
						|
								
							 | 
						|
								            EndMode3D();
							 | 
						|
								
							 | 
						|
								            // Draw some debug GUI text
							 | 
						|
								            DrawText(TextFormat("Hit Object: %s", hitObjectName), 10, 50, 10, BLACK);
							 | 
						|
								
							 | 
						|
								            if (collision.hit)
							 | 
						|
								            {
							 | 
						|
								                int ypos = 70;
							 | 
						|
								
							 | 
						|
								                DrawText(TextFormat("Distance: %3.2f", collision.distance), 10, ypos, 10, BLACK);
							 | 
						|
								
							 | 
						|
								                DrawText(TextFormat("Hit Pos: %3.2f %3.2f %3.2f",
							 | 
						|
								                                    collision.point.x,
							 | 
						|
								                                    collision.point.y,
							 | 
						|
								                                    collision.point.z), 10, ypos + 15, 10, BLACK);
							 | 
						|
								
							 | 
						|
								                DrawText(TextFormat("Hit Norm: %3.2f %3.2f %3.2f",
							 | 
						|
								                                    collision.normal.x,
							 | 
						|
								                                    collision.normal.y,
							 | 
						|
								                                    collision.normal.z), 10, ypos + 30, 10, BLACK);
							 | 
						|
								
							 | 
						|
								                if (triHitInfo.hit && TextIsEqual(hitObjectName, "Triangle"))
							 | 
						|
								                    DrawText(TextFormat("Barycenter: %3.2f %3.2f %3.2f",  bary.x, bary.y, bary.z), 10, ypos + 45, 10, BLACK);
							 | 
						|
								            }
							 | 
						|
								
							 | 
						|
								            DrawText("Right click mouse to toggle camera controls", 10, 430, 10, GRAY);
							 | 
						|
								
							 | 
						|
								            DrawText("(c) Turret 3D model by Alberto Cano", screenWidth - 200, screenHeight - 20, 10, GRAY);
							 | 
						|
								
							 | 
						|
								            DrawFPS(10, 10);
							 | 
						|
								
							 | 
						|
								        EndDrawing();
							 | 
						|
								        //----------------------------------------------------------------------------------
							 | 
						|
								    }
							 | 
						|
								
							 | 
						|
								    // De-Initialization
							 | 
						|
								    //--------------------------------------------------------------------------------------
							 | 
						|
								    UnloadModel(tower);         // Unload model
							 | 
						|
								    UnloadTexture(texture);     // Unload texture
							 | 
						|
								
							 | 
						|
								    CloseWindow();              // Close window and OpenGL context
							 | 
						|
								    //--------------------------------------------------------------------------------------
							 | 
						|
								
							 | 
						|
								    return 0;
							 | 
						|
								}
							 |