2
0
mirror of https://github.com/raylib-cs/raylib-cs synced 2025-04-03 11:09:40 -04:00

Safe UpdateMeshBuffer Method (#262)

This commit is contained in:
Nicky McDonald 2024-08-23 02:54:38 -04:00 committed by GitHub
parent 51925dd42d
commit 377a32a9db
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
3 changed files with 171 additions and 0 deletions

View File

@ -0,0 +1,127 @@
using System.Numerics;
using static Raylib_cs.Raylib;
namespace Examples.Models;
public class DynamicMesh
{
public unsafe static int Main()
{
// Initialization
//--------------------------------------------------------------------------------------
const int screenWidth = 800;
const int screenHeight = 450;
InitWindow(screenWidth, screenHeight, "raylib [models] example - dynamic mesh");
// Define the camera to look into our 3d world
Camera3D camera = new();
camera.Position = Vector3.One * 1.5f;
camera.Target = camera.Position + new Vector3(1f, -0.25f, 1f);
camera.Up = Vector3.UnitY;
camera.FovY = 60.0f;
camera.Projection = CameraProjection.Perspective;
// Generate a dynamic mesh using utils to allocate/access mesh attribute data
const int triangleRows = 48;
const int vertexRows = triangleRows + 1;
Mesh dynamicMesh = new(vertexRows * vertexRows, triangleRows * triangleRows * 2);
dynamicMesh.AllocVertices();
dynamicMesh.AllocTexCoords();
dynamicMesh.AllocIndices();
Span<Vector3> vertices = dynamicMesh.VerticesAs<Vector3>();
Span<Vector2> texcoords = dynamicMesh.TexCoordsAs<Vector2>();
Span<ushort> indices = dynamicMesh.IndicesAs<ushort>();
for (int z = 0, i = 0; z < triangleRows; z++)
{
for (int x = 0; x < triangleRows; x++, i += 6)
{
indices[i + 0] = (ushort)(x + (z * vertexRows));
indices[i + 1] = (ushort)(indices[i] + vertexRows);
indices[i + 2] = (ushort)(indices[i] + 1);
indices[i + 3] = (ushort)(indices[i] + 1);
indices[i + 4] = (ushort)(indices[i] + vertexRows);
indices[i + 5] = (ushort)(indices[i] + vertexRows + 1);
}
}
UploadMesh(ref dynamicMesh, true);
// Allocate the texture
Image image = GenImageColor(triangleRows, triangleRows, Color.Blank);
Texture2D texture = LoadTextureFromImage(image);
Color[] pixels = new Color[texture.Width * texture.Height];
UnloadImage(image);
// Load the material
Material material = LoadMaterialDefault();
SetMaterialTexture(ref material, MaterialMapIndex.Diffuse, texture);
SetTargetFPS(60);
//--------------------------------------------------------------------------------------
// Main game loop
while (!WindowShouldClose())
{
// Update
//----------------------------------------------------------------------------------
float time = (float)GetTime();
Random random = new(42);
for (int z = 0, i = 0; z < vertexRows; z++)
{
for (int x = 0; x < vertexRows; x++, i++)
{
float noiseX = SmoothNoise(time + random.Next(10000));
float noiseZ = SmoothNoise(time + random.Next(10000));
vertices[i].X = x + noiseX - .5f;
vertices[i].Y = (noiseX + noiseZ) / 2;
vertices[i].Z = z + noiseZ - .5f;
texcoords[i].X = (x - noiseZ) / triangleRows;
texcoords[i].Y = (z - noiseX) / triangleRows;
}
}
UpdateMeshBuffer<Vector3>(dynamicMesh, Mesh.VboIdIndexVertices, vertices, 0);
UpdateMeshBuffer<Vector2>(dynamicMesh, Mesh.VboIdIndexTexCoords, texcoords, 0);
for (int y = 0, i = 0; y < texture.Height; y++)
{
for (int x = 0; x < texture.Width; x++, i++)
{
pixels[i] = new(32, 178, 170, 255);
pixels[i] = ColorBrightness(pixels[i], (SmoothNoise(time + random.Next(10000)) / 8) - (1 / 16f));
pixels[i] = ColorAlpha(pixels[i], (triangleRows - new Vector2(x, y).Length()) / triangleRows);
}
}
UpdateTexture(texture, pixels);
//----------------------------------------------------------------------------------
// Draw
//----------------------------------------------------------------------------------
BeginDrawing();
ClearBackground(Color.RayWhite);
BeginMode3D(camera);
DrawMesh(dynamicMesh, material, Matrix4x4.Identity);
EndMode3D();
EndDrawing();
//----------------------------------------------------------------------------------
}
// De-Initialization
//--------------------------------------------------------------------------------------
UnloadMaterial(material);
// Raylib.UnloadTexture(texture); <- No need to unload the texture. UnloadMaterial(Material) already unloaded it for us
UnloadMesh(dynamicMesh);
CloseWindow();
//--------------------------------------------------------------------------------------
return 0;
}
private static float SmoothNoise(float value)
{
return ((MathF.Sin(value) + MathF.Cos(value * MathF.E)) / 4) + .5f;
}
}

View File

@ -216,5 +216,40 @@ public unsafe partial struct Mesh
/// </summary>
public uint* VboId = default;
/// <summary>
/// Default <see cref="VboId"/> index for <see cref="Vertices"/>
/// </summary>
public const int VboIdIndexVertices = 0;
/// <summary>
/// Default <see cref="VboId"/> index for <see cref="TexCoords"/>
/// </summary>
public const int VboIdIndexTexCoords = 1;
/// <summary>
/// Default <see cref="VboId"/> index for <see cref="Normals"/>
/// </summary>
public const int VboIdIndexNormals = 2;
/// <summary>
/// Default <see cref="VboId"/> index for <see cref="Colors"/>
/// </summary>
public const int VboIdIndexColors = 3;
/// <summary>
/// Default <see cref="VboId"/> index for <see cref="Tangents"/>
/// </summary>
public const int VboIdIndexTangents = 4;
/// <summary>
/// Default <see cref="VboId"/> index for <see cref="TexCoords2"/>
/// </summary>
public const int VboIdIndexTexCoords2 = 5;
/// <summary>
/// Default <see cref="VboId"/> index for <see cref="Indices"/>
/// </summary>
public const int VboIdIndexIndices = 6;
#endregion
}

View File

@ -911,6 +911,15 @@ public static unsafe partial class Raylib
}
}
/// <summary> Update mesh vertex data in GPU for a specific buffer index </summary>
public static void UpdateMeshBuffer<T>(Mesh mesh, int index, ReadOnlySpan<T> data, int offset) where T : unmanaged
{
fixed (void* dataPtr = data)
{
UpdateMeshBuffer(mesh, index, dataPtr, data.Length * sizeof(T), offset);
}
}
/// <summary>Set texture for a material map type (MAP_DIFFUSE, MAP_SPECULAR...)</summary>
public static void SetMaterialTexture(ref Material material, MaterialMapIndex mapType, Texture2D texture)
{