forked from mirror/DotRecast
336 lines
13 KiB
C#
336 lines
13 KiB
C#
using System;
|
|
using System.Collections.Generic;
|
|
using System.IO;
|
|
using Silk.NET.OpenGL;
|
|
using DotRecast.Core;
|
|
|
|
namespace DotRecast.Recast.Demo.Draw;
|
|
|
|
public class ModernOpenGLDraw : OpenGLDraw
|
|
{
|
|
private GL _gl;
|
|
private uint program;
|
|
private int uniformTexture;
|
|
private int uniformProjectionMatrix;
|
|
private uint vbo;
|
|
private uint ebo;
|
|
private uint vao;
|
|
private DebugDrawPrimitives currentPrim;
|
|
private float fogStart;
|
|
private float fogEnd;
|
|
private bool fogEnabled;
|
|
private int uniformViewMatrix;
|
|
private readonly List<OpenGLVertex> vertices = new();
|
|
private GLCheckerTexture _texture;
|
|
private float[] _viewMatrix;
|
|
private float[] _projectionMatrix;
|
|
private int uniformUseTexture;
|
|
private int uniformFog;
|
|
private int uniformFogStart;
|
|
private int uniformFogEnd;
|
|
|
|
public ModernOpenGLDraw(GL gl)
|
|
{
|
|
_gl = gl;
|
|
}
|
|
|
|
public unsafe void init()
|
|
{
|
|
string NK_SHADER_VERSION = PlatformID.MacOSX == Environment.OSVersion.Platform ? "#version 150\n" : "#version 300 es\n";
|
|
string vertex_shader = NK_SHADER_VERSION + "uniform mat4 ProjMtx;\n" //
|
|
+ "uniform mat4 ViewMtx;\n" //
|
|
+ "in vec3 Position;\n" //
|
|
+ "in vec2 TexCoord;\n" //
|
|
+ "in vec4 Color;\n" //
|
|
+ "out vec2 Frag_UV;\n" //
|
|
+ "out vec4 Frag_Color;\n" //
|
|
+ "out float Frag_Depth;\n" //
|
|
+ "void main() {\n" //
|
|
+ " Frag_UV = TexCoord;\n" //
|
|
+ " Frag_Color = Color;\n" //
|
|
+ " vec4 VSPosition = ViewMtx * vec4(Position, 1);\n" //
|
|
+ " Frag_Depth = -VSPosition.z;\n" //
|
|
+ " gl_Position = ProjMtx * VSPosition;\n" //
|
|
+ "}\n";
|
|
string fragment_shader = NK_SHADER_VERSION + "precision mediump float;\n" //
|
|
+ "uniform sampler2D Texture;\n" //
|
|
+ "uniform float UseTexture;\n" //
|
|
+ "uniform float EnableFog;\n" //
|
|
+ "uniform float FogStart;\n" //
|
|
+ "uniform float FogEnd;\n" //
|
|
+ "const vec4 FogColor = vec4(0.3f, 0.3f, 0.32f, 1.0f);\n" //
|
|
+ "in vec2 Frag_UV;\n" //
|
|
+ "in vec4 Frag_Color;\n" //
|
|
+ "in float Frag_Depth;\n" //
|
|
+ "out vec4 Out_Color;\n" //
|
|
+ "void main(){\n" //
|
|
+ " Out_Color = mix(FogColor, Frag_Color * mix(vec4(1), texture(Texture, Frag_UV.st), UseTexture), 1.0 - EnableFog * clamp( (Frag_Depth - FogStart) / (FogEnd - FogStart), 0.0, 1.0) );\n" //
|
|
+ "}\n";
|
|
|
|
program = _gl.CreateProgram();
|
|
uint vert_shdr = _gl.CreateShader(GLEnum.VertexShader);
|
|
_gl.ShaderSource(vert_shdr, vertex_shader);
|
|
_gl.CompileShader(vert_shdr);
|
|
_gl.GetShader(vert_shdr, GLEnum.CompileStatus, out var status);
|
|
if (status != (int)GLEnum.True)
|
|
{
|
|
throw new InvalidOperationException();
|
|
}
|
|
|
|
uint frag_shdr = _gl.CreateShader(GLEnum.FragmentShader);
|
|
_gl.ShaderSource(frag_shdr, fragment_shader);
|
|
_gl.CompileShader(frag_shdr);
|
|
_gl.GetShader(frag_shdr, GLEnum.CompileStatus, out status);
|
|
if (status != (int)GLEnum.True)
|
|
{
|
|
throw new InvalidOperationException();
|
|
}
|
|
|
|
_gl.AttachShader(program, vert_shdr);
|
|
_gl.AttachShader(program, frag_shdr);
|
|
_gl.LinkProgram(program);
|
|
_gl.GetProgram(program, GLEnum.LinkStatus, out status);
|
|
if (status != (int)GLEnum.True)
|
|
{
|
|
throw new InvalidOperationException();
|
|
}
|
|
|
|
_gl.DetachShader(program, vert_shdr);
|
|
_gl.DetachShader(program, frag_shdr);
|
|
_gl.DeleteShader(vert_shdr);
|
|
_gl.DeleteShader(frag_shdr);
|
|
|
|
uniformTexture = _gl.GetUniformLocation(program, "Texture");
|
|
uniformUseTexture = _gl.GetUniformLocation(program, "UseTexture");
|
|
uniformFog = _gl.GetUniformLocation(program, "EnableFog");
|
|
uniformFogStart = _gl.GetUniformLocation(program, "FogStart");
|
|
uniformFogEnd = _gl.GetUniformLocation(program, "FogEnd");
|
|
uniformProjectionMatrix = _gl.GetUniformLocation(program, "ProjMtx");
|
|
uniformViewMatrix = _gl.GetUniformLocation(program, "ViewMtx");
|
|
|
|
uint attrib_pos = (uint)_gl.GetAttribLocation(program, "Position");
|
|
uint attrib_uv = (uint)_gl.GetAttribLocation(program, "TexCoord");
|
|
uint attrib_col = (uint)_gl.GetAttribLocation(program, "Color");
|
|
|
|
// buffer setup
|
|
vao = _gl.GenVertexArray();
|
|
vbo = _gl.GenBuffer();
|
|
ebo = _gl.GenBuffer();
|
|
|
|
_gl.BindVertexArray(vao);
|
|
_gl.BindBuffer(GLEnum.ArrayBuffer, vbo);
|
|
_gl.BindBuffer(GLEnum.ElementArrayBuffer, ebo);
|
|
|
|
|
|
_gl.EnableVertexAttribArray(attrib_pos);
|
|
_gl.EnableVertexAttribArray(attrib_uv);
|
|
_gl.EnableVertexAttribArray(attrib_col);
|
|
|
|
// _gl.VertexAttribP3(attrib_pos, GLEnum.Float, false, 24);
|
|
// _gl.VertexAttribP2(attrib_uv, GLEnum.Float, false, 24);
|
|
// _gl.VertexAttribP4(attrib_col, GLEnum.UnsignedByte, true, 24);
|
|
IntPtr pointer1 = 0;
|
|
IntPtr pointer2 = 12;
|
|
IntPtr pointer3 = 20;
|
|
_gl.VertexAttribPointer(attrib_pos, 3, VertexAttribPointerType.Float, false, 24, pointer1.ToPointer());
|
|
_gl.VertexAttribPointer(attrib_uv, 2, VertexAttribPointerType.Float, false, 24, pointer2.ToPointer());
|
|
_gl.VertexAttribPointer(attrib_col, 4, VertexAttribPointerType.UnsignedByte, true, 24, pointer3.ToPointer());
|
|
|
|
// _gl.VertexAttribPointer(attrib_pos, 3, GLEnum.Float, false, 24, (void*)0);
|
|
// _gl.VertexAttribPointer(attrib_uv, 2, GLEnum.Float, false, 24, (void*)12);
|
|
// _gl.VertexAttribPointer(attrib_col, 4, GLEnum.UnsignedByte, true, 24, (void*)20);
|
|
|
|
|
|
// _gl.VertexAttribP3(attrib_pos, GLEnum.Float, false, 0);
|
|
// _gl.VertexAttribP2(attrib_uv, GLEnum.Float, false, 12);
|
|
// _gl.VertexAttribP4(attrib_col, GLEnum.UnsignedByte, true, 20);
|
|
|
|
_gl.BindTexture(GLEnum.Texture2D, 0);
|
|
|
|
_gl.BindVertexArray(0);
|
|
_gl.BindBuffer(GLEnum.ArrayBuffer, 0);
|
|
_gl.BindBuffer(GLEnum.ElementArrayBuffer, 0);
|
|
}
|
|
|
|
public void clear()
|
|
{
|
|
_gl.ClearColor(0.3f, 0.3f, 0.32f, 1.0f);
|
|
_gl.Clear((uint)GLEnum.ColorBufferBit | (uint)GLEnum.DepthBufferBit);
|
|
_gl.Enable(GLEnum.Blend);
|
|
_gl.BlendFunc(GLEnum.SrcAlpha, GLEnum.OneMinusSrcAlpha);
|
|
_gl.Disable(GLEnum.Texture2D);
|
|
_gl.Enable(GLEnum.DepthTest);
|
|
_gl.Enable(GLEnum.CullFace);
|
|
}
|
|
|
|
public void begin(DebugDrawPrimitives prim, float size)
|
|
{
|
|
currentPrim = prim;
|
|
vertices.Clear();
|
|
_gl.LineWidth(size);
|
|
_gl.PointSize(size);
|
|
}
|
|
|
|
public unsafe void end()
|
|
{
|
|
if (0 >= vertices.Count)
|
|
{
|
|
return;
|
|
}
|
|
|
|
_gl.UseProgram(program);
|
|
_gl.Uniform1(uniformTexture, 0);
|
|
_gl.UniformMatrix4(uniformViewMatrix, 1, false, _viewMatrix);
|
|
_gl.UniformMatrix4(uniformProjectionMatrix, 1, false, _projectionMatrix);
|
|
_gl.Uniform1(uniformFogStart, fogStart);
|
|
_gl.Uniform1(uniformFogEnd, fogEnd);
|
|
_gl.Uniform1(uniformFog, fogEnabled ? 1.0f : 0.0f);
|
|
_gl.BindVertexArray(vao);
|
|
_gl.BindBuffer(GLEnum.ArrayBuffer, vbo);
|
|
_gl.BindBuffer(GLEnum.ElementArrayBuffer, ebo);
|
|
// glBufferData(GL_ARRAY_BUFFER, MAX_VERTEX_BUFFER, GL_STREAM_DRAW);
|
|
// glBufferData(GL_ELEMENT_ARRAY_BUFFER, MAX_ELEMENT_BUFFER, GL_STREAM_DRAW);
|
|
|
|
uint vboSize = (uint)vertices.Count * 24;
|
|
uint eboSize = currentPrim == DebugDrawPrimitives.QUADS ? (uint)vertices.Count * 6 : (uint)vertices.Count * 4;
|
|
|
|
_gl.BufferData(GLEnum.ArrayBuffer, vboSize, null, GLEnum.StreamDraw);
|
|
_gl.BufferData(GLEnum.ElementArrayBuffer, eboSize, null, GLEnum.StreamDraw);
|
|
// load draw vertices & elements directly into vertex + element buffer
|
|
|
|
{
|
|
byte* pVerts = (byte*)_gl.MapBuffer(GLEnum.ArrayBuffer, GLEnum.WriteOnly);
|
|
byte* pElems = (byte*)_gl.MapBuffer(GLEnum.ElementArrayBuffer, GLEnum.WriteOnly);
|
|
|
|
using var unmanagedVerts = new UnmanagedMemoryStream(pVerts, vboSize, vboSize, FileAccess.Write);
|
|
using var unmanagedElems = new UnmanagedMemoryStream(pElems, eboSize, eboSize, FileAccess.Write);
|
|
|
|
using var verts = new BinaryWriter(unmanagedVerts);
|
|
using var elems = new BinaryWriter(unmanagedElems);
|
|
|
|
vertices.forEach(v => v.store(verts));
|
|
if (currentPrim == DebugDrawPrimitives.QUADS)
|
|
{
|
|
for (int i = 0; i < vertices.Count; i += 4)
|
|
{
|
|
// elems.Write(BitConverter.GetBytes(i));
|
|
// elems.Write(BitConverter.GetBytes(i + 1));
|
|
// elems.Write(BitConverter.GetBytes(i + 2));
|
|
// elems.Write(BitConverter.GetBytes(i));
|
|
// elems.Write(BitConverter.GetBytes(i + 2));
|
|
// elems.Write(BitConverter.GetBytes(i + 3));
|
|
elems.Write(i);
|
|
elems.Write(i + 1);
|
|
elems.Write(i + 2);
|
|
elems.Write(i);
|
|
elems.Write(i + 2);
|
|
elems.Write(i + 3);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
for (int i = 0; i < vertices.Count; i++)
|
|
{
|
|
//elems.Write(BitConverter.GetBytes(i));
|
|
elems.Write(i);
|
|
}
|
|
}
|
|
|
|
verts.Flush();
|
|
elems.Flush();
|
|
|
|
_gl.UnmapBuffer(GLEnum.ElementArrayBuffer);
|
|
_gl.UnmapBuffer(GLEnum.ArrayBuffer);
|
|
}
|
|
if (_texture != null)
|
|
{
|
|
_texture.bind();
|
|
_gl.Uniform1(uniformUseTexture, 1.0f);
|
|
}
|
|
else
|
|
{
|
|
_gl.Uniform1(uniformUseTexture, 0.0f);
|
|
}
|
|
|
|
switch (currentPrim)
|
|
{
|
|
case DebugDrawPrimitives.POINTS:
|
|
_gl.DrawElements(GLEnum.Points, (uint)vertices.Count, GLEnum.UnsignedInt, (void*)0);
|
|
break;
|
|
case DebugDrawPrimitives.LINES:
|
|
_gl.DrawElements(GLEnum.Lines, (uint)vertices.Count, GLEnum.UnsignedInt, (void*)0);
|
|
break;
|
|
case DebugDrawPrimitives.TRIS:
|
|
_gl.DrawElements(GLEnum.Triangles, (uint)vertices.Count, GLEnum.UnsignedInt, (void*)0);
|
|
break;
|
|
case DebugDrawPrimitives.QUADS:
|
|
_gl.DrawElements(GLEnum.Triangles, (uint)(vertices.Count * 6 / 4), GLEnum.UnsignedInt, (void*)0);
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
_gl.UseProgram(0);
|
|
_gl.BindVertexArray(0);
|
|
_gl.BindBuffer(GLEnum.ArrayBuffer, 0);
|
|
_gl.BindBuffer(GLEnum.ElementArrayBuffer, 0);
|
|
vertices.Clear();
|
|
_gl.LineWidth(1.0f);
|
|
_gl.PointSize(1.0f);
|
|
}
|
|
|
|
public void vertex(float x, float y, float z, int color)
|
|
{
|
|
vertices.Add(new OpenGLVertex(x, y, z, color));
|
|
}
|
|
|
|
public void vertex(float[] pos, int color)
|
|
{
|
|
vertices.Add(new OpenGLVertex(pos, color));
|
|
}
|
|
|
|
public void vertex(float[] pos, int color, float[] uv)
|
|
{
|
|
vertices.Add(new OpenGLVertex(pos, uv, color));
|
|
}
|
|
|
|
public void vertex(float x, float y, float z, int color, float u, float v)
|
|
{
|
|
vertices.Add(new OpenGLVertex(x, y, z, u, v, color));
|
|
}
|
|
|
|
public void depthMask(bool state)
|
|
{
|
|
_gl.DepthMask(state);
|
|
}
|
|
|
|
public void texture(GLCheckerTexture g_tex, bool state)
|
|
{
|
|
_texture = state ? g_tex : null;
|
|
if (_texture != null)
|
|
{
|
|
_texture.bind();
|
|
}
|
|
}
|
|
|
|
public void projectionMatrix(float[] projectionMatrix)
|
|
{
|
|
this._projectionMatrix = projectionMatrix;
|
|
}
|
|
|
|
public void viewMatrix(float[] viewMatrix)
|
|
{
|
|
this._viewMatrix = viewMatrix;
|
|
}
|
|
|
|
public void fog(float start, float end)
|
|
{
|
|
fogStart = start;
|
|
fogEnd = end;
|
|
}
|
|
|
|
public void fog(bool state)
|
|
{
|
|
fogEnabled = state;
|
|
}
|
|
} |