/* Copyright (c) 2009-2010 Mikko Mononen memon@inside.org recast4j copyright (c) 2015-2019 Piotr Piastucki piotr@jtilia.org DotRecast Copyright (c) 2023 Choi Ikpil ikpil@naver.com This software is provided 'as-is', without any express or implied warranty. In no event will the authors be held liable for any damages arising from the use of this software. Permission is granted to anyone to use this software for any purpose, including commercial applications, and to alter it and redistribute it freely, subject to the following restrictions: 1. The origin of this software must not be misrepresented; you must not claim that you wrote the original software. If you use this software in a product, an acknowledgment in the product documentation would be appreciated but is not required. 2. Altered source versions must be plainly marked as such, and must not be misrepresented as being the original software. 3. This notice may not be removed or altered from any source distribution. */ using System; using System.Collections.Generic; using System.Collections.Immutable; using System.IO; using System.Linq; using System.Numerics; using Serilog; using Silk.NET.Input; using Silk.NET.Maths; using Silk.NET.OpenGL; using Silk.NET.OpenGL.Extensions.ImGui; using Silk.NET.Windowing; using ImGuiNET; using DotRecast.Core; using DotRecast.Detour; using DotRecast.Detour.Extras.Unity.Astar; using DotRecast.Detour.Io; using DotRecast.Recast.Demo.Builder; using DotRecast.Recast.Demo.Draw; using DotRecast.Recast.Demo.Geom; using DotRecast.Recast.Demo.Tools; using DotRecast.Recast.Demo.UI; using static DotRecast.Core.RecastMath; using Window = Silk.NET.Windowing.Window; namespace DotRecast.Recast.Demo; public class RecastDemo { private static readonly ILogger Logger = Log.ForContext(); private IWindow window; private GL _gl; private IInputContext _input; private ImGuiController _imgui; private RecastDemoCanvas _canvas; private int width = 1000; private int height = 900; private readonly string title = "DotRecast Demo"; //private readonly RecastDebugDraw dd; private NavMeshRenderer renderer; private bool building = false; private float timeAcc = 0; private float camr = 1000; private readonly SoloNavMeshBuilder soloNavMeshBuilder = new SoloNavMeshBuilder(); private readonly TileNavMeshBuilder tileNavMeshBuilder = new TileNavMeshBuilder(); private Sample sample; private bool processHitTest = false; private bool processHitTestShift; private int modState; private readonly float[] mousePos = new float[2]; private bool _mouseOverMenu; private bool pan; private bool movedDuringPan; private bool rotate; private bool movedDuringRotate; private float scrollZoom; private readonly float[] origMousePos = new float[2]; private readonly float[] origCameraEulers = new float[2]; private Vector3f origCameraPos = new Vector3f(); private readonly float[] cameraEulers = { 45, -45 }; private Vector3f cameraPos = Vector3f.Of(0, 0, 0); private Vector3f rayStart = new Vector3f(); private Vector3f rayEnd = new Vector3f(); private float[] projectionMatrix = new float[16]; private float[] modelviewMatrix = new float[16]; private float _moveFront; private float _moveLeft; private float _moveBack; private float _moveRight; private float _moveUp; private float _moveDown; private float _moveAccel; private int[] viewport; private bool markerPositionSet; private Vector3f markerPosition = new Vector3f(); private ToolsView toolsUI; private RcSettingsView settingsUI; private RcLogView logUI; private long prevFrameTime; private RecastDebugDraw dd; public RecastDemo() { } public void Run() { window = CreateWindow(); window.Run(); } public void OnMouseScrolled(IMouse mice, ScrollWheel scrollWheel) { if (scrollWheel.Y < 0) { // wheel down if (!_mouseOverMenu) { scrollZoom += 1.0f; } } else { if (!_mouseOverMenu) { scrollZoom -= 1.0f; } } float[] modelviewMatrix = dd.ViewMatrix(cameraPos, cameraEulers); cameraPos.x += scrollZoom * 2.0f * modelviewMatrix[2]; cameraPos.y += scrollZoom * 2.0f * modelviewMatrix[6]; cameraPos.z += scrollZoom * 2.0f * modelviewMatrix[10]; scrollZoom = 0; } public void OnMouseMoved(IMouse mouse, Vector2 position) { mousePos[0] = (float)position.X; mousePos[1] = (float)position.Y; int dx = (int)(mousePos[0] - origMousePos[0]); int dy = (int)(mousePos[1] - origMousePos[1]); if (rotate) { cameraEulers[0] = origCameraEulers[0] + dy * 0.25f; cameraEulers[1] = origCameraEulers[1] + dx * 0.25f; if (dx * dx + dy * dy > 3 * 3) { movedDuringRotate = true; } } if (pan) { float[] modelviewMatrix = dd.ViewMatrix(cameraPos, cameraEulers); cameraPos.x = origCameraPos.x; cameraPos.y = origCameraPos.y; cameraPos.z = origCameraPos.z; cameraPos.x -= 0.1f * dx * modelviewMatrix[0]; cameraPos.y -= 0.1f * dx * modelviewMatrix[4]; cameraPos.z -= 0.1f * dx * modelviewMatrix[8]; cameraPos.x += 0.1f * dy * modelviewMatrix[1]; cameraPos.y += 0.1f * dy * modelviewMatrix[5]; cameraPos.z += 0.1f * dy * modelviewMatrix[9]; if (dx * dx + dy * dy > 3 * 3) { movedDuringPan = true; } } } public void OnMouseUpAndDown(IMouse mouse, MouseButton button, bool down) { modState = 0; if (down) { if (button == MouseButton.Right) { if (!_mouseOverMenu) { // Rotate view rotate = true; movedDuringRotate = false; origMousePos[0] = mousePos[0]; origMousePos[1] = mousePos[1]; origCameraEulers[0] = cameraEulers[0]; origCameraEulers[1] = cameraEulers[1]; } } else if (button == MouseButton.Middle) { if (!_mouseOverMenu) { // Pan view pan = true; movedDuringPan = false; origMousePos[0] = mousePos[0]; origMousePos[1] = mousePos[1]; origCameraPos.x = cameraPos.x; origCameraPos.y = cameraPos.y; origCameraPos.z = cameraPos.z; } } } else { // Handle mouse clicks here. if (button == MouseButton.Right) { rotate = false; if (!_mouseOverMenu) { if (!movedDuringRotate) { processHitTest = true; processHitTestShift = true; } } } else if (button == MouseButton.Left) { if (!_mouseOverMenu) { processHitTest = true; //processHitTestShift = (mods & Keys.GLFW_MOD_SHIFT) != 0 ? true : false; //processHitTestShift = (mods & Keys.) != 0 ? true : false; } } else if (button == MouseButton.Middle) { pan = false; } } } private IWindow CreateWindow() { var monitor = Window.Platforms.First().GetMainMonitor(); var resolution = monitor.VideoMode.Resolution.Value; float aspect = 16.0f / 9.0f; width = Math.Min(resolution.X, (int)(resolution.Y * aspect)) - 100; height = resolution.Y - 100; viewport = new int[] { 0, 0, width, height }; var options = WindowOptions.Default; options.Title = title; options.Size = new Vector2D(width, height); options.Position = new Vector2D((resolution.X - width) / 2, (resolution.Y - height) / 2); options.VSync = true; options.ShouldSwapAutomatically = false; options.PreferredDepthBufferBits = 24; window = Window.Create(options); if (window == null) { throw new Exception("Failed to create the GLFW window"); } window.Closing += OnWindowClosing; window.Load += OnWindowOnLoad; window.Resize += OnWindowResize; window.FramebufferResize += OnWindowFramebufferSizeChanged; window.Update += OnWindowUpdate; window.Render += OnWindowRender; // // -- move somewhere else: // glfw.SetWindowPos(window, (mode->Width - width) / 2, (mode->Height - height) / 2); // // GlfwSetWindowMonitor(window.GetWindow(), monitor, 0, 0, mode.Width(), mode.Height(), mode.RefreshRate()); // glfw.ShowWindow(window); // glfw.MakeContextCurrent(window); //} //glfw.SwapInterval(1); return window; } private DemoInputGeomProvider LoadInputMesh(byte[] stream) { DemoInputGeomProvider geom = DemoObjImporter.Load(stream); sample = new Sample(geom, ImmutableArray.Empty, null, settingsUI, dd); toolsUI.SetEnabled(true); return geom; } private void LoadNavMesh(FileStream file, string filename) { NavMesh mesh = null; if (filename.EndsWith(".zip") || filename.EndsWith(".bytes")) { UnityAStarPathfindingImporter importer = new UnityAStarPathfindingImporter(); mesh = importer.Load(file)[0]; } else if (filename.EndsWith(".bin") || filename.EndsWith(".navmesh")) { MeshSetReader reader = new MeshSetReader(); using (var fis = new BinaryReader(file)) { mesh = reader.Read(fis, 6); } } if (mesh != null) { //sample = new Sample(null, ImmutableArray.Empty, mesh, settingsUI, dd); toolsUI.SetEnabled(true); } } private void OnWindowClosing() { } private void OnWindowResize(Vector2D size) { width = size.X; height = size.Y; } private void OnWindowFramebufferSizeChanged(Vector2D size) { _gl.Viewport(size); viewport = new int[] { 0, 0, width, height }; } private void OnWindowOnLoad() { _input = window.CreateInput(); // mouse input foreach (var mice in _input.Mice) { mice.Scroll += OnMouseScrolled; mice.MouseDown += (m, b) => OnMouseUpAndDown(m, b, true); mice.MouseUp += (m, b) => OnMouseUpAndDown(m, b, false); mice.MouseMove += OnMouseMoved; } _gl = window.CreateOpenGL(); dd = new RecastDebugDraw(_gl); renderer = new NavMeshRenderer(dd); dd.Init(camr); _imgui = new ImGuiController(_gl, window, _input); settingsUI = new RcSettingsView(); toolsUI = new ToolsView( new TestNavmeshTool(), new OffMeshConnectionTool(), new ConvexVolumeTool(), new CrowdTool(), new JumpLinkBuilderTool(), new DynamicUpdateTool() ); logUI = new RcLogView(); _canvas = new RecastDemoCanvas(window, settingsUI, toolsUI, logUI); var vendor = _gl.GetStringS(GLEnum.Vendor); var version = _gl.GetStringS(GLEnum.Version); var renderGl = _gl.GetStringS(GLEnum.Renderer); var glslString = _gl.GetStringS(GLEnum.ShadingLanguageVersion); Logger.Debug(vendor); Logger.Debug(version); Logger.Debug(renderGl); Logger.Debug(glslString); DemoInputGeomProvider geom = LoadInputMesh(Loader.ToBytes("nav_test.obj")); sample = new Sample(geom, ImmutableArray.Empty, null, settingsUI, dd); } private void UpdateKeyboard(float dt) { // keyboard input foreach (var keyboard in _input.Keyboards) { var tempMoveFront = keyboard.IsKeyPressed(Key.W) || keyboard.IsKeyPressed(Key.Up) ? 1.0f : -1f; var tempMoveLeft = keyboard.IsKeyPressed(Key.A) || keyboard.IsKeyPressed(Key.Left) ? 1.0f : -1f; var tempMoveBack = keyboard.IsKeyPressed(Key.S) || keyboard.IsKeyPressed(Key.Down) ? 1.0f : -1f; var tempMoveRight = keyboard.IsKeyPressed(Key.D) || keyboard.IsKeyPressed(Key.Right) ? 1.0f : -1f; var tempMoveUp = keyboard.IsKeyPressed(Key.Q) || keyboard.IsKeyPressed(Key.PageUp) ? 1.0f : -1f; var tempMoveDown = keyboard.IsKeyPressed(Key.E) || keyboard.IsKeyPressed(Key.PageDown) ? 1.0f : -1f; var tempMoveAccel = keyboard.IsKeyPressed(Key.ShiftLeft) || keyboard.IsKeyPressed(Key.ShiftRight) ? 1.0f : -1f; modState = keyboard.IsKeyPressed(Key.ControlLeft) || keyboard.IsKeyPressed(Key.ShiftRight) ? 1 : 0; _moveFront = Clamp(_moveFront + tempMoveFront * dt * 4.0f, 0, 2.0f); _moveLeft = Clamp(_moveLeft + tempMoveLeft * dt * 4.0f, 0, 2.0f); _moveBack = Clamp(_moveBack + tempMoveBack * dt * 4.0f, 0, 2.0f); _moveRight = Clamp(_moveRight + tempMoveRight * dt * 4.0f, 0, 2.0f); _moveUp = Clamp(_moveUp + tempMoveUp * dt * 4.0f, 0, 2.0f); _moveDown = Clamp(_moveDown + tempMoveDown * dt * 4.0f, 0, 2.0f); _moveAccel = Clamp(_moveAccel + tempMoveAccel * dt * 4.0f, 0, 2.0f); } } private void OnWindowUpdate(double dt) { /* * try (MemoryStack stack = StackPush()) { int[] w = stack.MallocInt(1); int[] h = * stack.MallocInt(1); GlfwGetWindowSize(win, w, h); width = w.x; height = h.x; } */ if (sample.GetInputGeom() != null) { Vector3f bmin = sample.GetInputGeom().GetMeshBoundsMin(); Vector3f bmax = sample.GetInputGeom().GetMeshBoundsMax(); int[] voxels = Recast.CalcGridSize(bmin, bmax, settingsUI.GetCellSize()); settingsUI.SetVoxels(voxels); settingsUI.SetTiles(tileNavMeshBuilder.GetTiles(sample.GetInputGeom(), settingsUI.GetCellSize(), settingsUI.GetTileSize())); settingsUI.SetMaxTiles(tileNavMeshBuilder.GetMaxTiles(sample.GetInputGeom(), settingsUI.GetCellSize(), settingsUI.GetTileSize())); settingsUI.SetMaxPolys(tileNavMeshBuilder.GetMaxPolysPerTile(sample.GetInputGeom(), settingsUI.GetCellSize(), settingsUI.GetTileSize())); } UpdateKeyboard((float)dt); // camera move float keySpeed = 22.0f; if (0 < _moveAccel) { keySpeed *= _moveAccel * 2.0f; } double movex = (_moveRight - _moveLeft) * keySpeed * dt; double movey = (_moveBack - _moveFront) * keySpeed * dt + scrollZoom * 2.0f; scrollZoom = 0; cameraPos.x += (float)(movex * modelviewMatrix[0]); cameraPos.y += (float)(movex * modelviewMatrix[4]); cameraPos.z += (float)(movex * modelviewMatrix[8]); cameraPos.x += (float)(movey * modelviewMatrix[2]); cameraPos.y += (float)(movey * modelviewMatrix[6]); cameraPos.z += (float)(movey * modelviewMatrix[10]); cameraPos.y += (float)((_moveUp - _moveDown) * keySpeed * dt); long time = FrequencyWatch.Ticks; prevFrameTime = time; // Update sample simulation. float SIM_RATE = 20; float DELTA_TIME = 1.0f / SIM_RATE; timeAcc = Clamp((float)(timeAcc + dt), -1.0f, 1.0f); int simIter = 0; while (timeAcc > DELTA_TIME) { timeAcc -= DELTA_TIME; if (simIter < 5 && sample != null) { toolsUI.HandleUpdate(DELTA_TIME); } simIter++; } if (settingsUI.IsMeshInputTrigerred()) { var bytes = Loader.ToBytes(settingsUI.GetMeshInputFilePath()); sample.Update(LoadInputMesh(bytes), null, null); } else if (settingsUI.IsNavMeshInputTrigerred()) { // try (MemoryStack stack = StackPush()) { // PointerBuffer aFilterPatterns = stack.MallocPointer(4); // aFilterPatterns.Put(stack.UTF8("*.bin")); // aFilterPatterns.Put(stack.UTF8("*.zip")); // aFilterPatterns.Put(stack.UTF8("*.bytes")); // aFilterPatterns.Put(stack.UTF8("*.navmesh")); // aFilterPatterns.Flip(); // string filename = TinyFileDialogs.Tinyfd_openFileDialog("Open Nav Mesh File", "", aFilterPatterns, // "Nav Mesh File", false); // if (filename != null) { // File file = new File(filename); // if (file.Exists()) { // try { // LoadNavMesh(file, filename); // geom = null; // } catch (Exception e) { // Console.WriteLine(e); // } // } // } // } } if (settingsUI.IsBuildTriggered() && sample.GetInputGeom() != null) { if (!building) { float m_cellSize = settingsUI.GetCellSize(); float m_cellHeight = settingsUI.GetCellHeight(); float m_agentHeight = settingsUI.GetAgentHeight(); float m_agentRadius = settingsUI.GetAgentRadius(); float m_agentMaxClimb = settingsUI.GetAgentMaxClimb(); float m_agentMaxSlope = settingsUI.GetAgentMaxSlope(); int m_regionMinSize = settingsUI.GetMinRegionSize(); int m_regionMergeSize = settingsUI.GetMergedRegionSize(); float m_edgeMaxLen = settingsUI.GetEdgeMaxLen(); float m_edgeMaxError = settingsUI.GetEdgeMaxError(); int m_vertsPerPoly = settingsUI.GetVertsPerPoly(); float m_detailSampleDist = settingsUI.GetDetailSampleDist(); float m_detailSampleMaxError = settingsUI.GetDetailSampleMaxError(); int m_tileSize = settingsUI.GetTileSize(); long t = FrequencyWatch.Ticks; Logger.Information($"build"); Tuple, NavMesh> buildResult; if (settingsUI.IsTiled()) { buildResult = tileNavMeshBuilder.Build(sample.GetInputGeom(), settingsUI.GetPartitioning(), m_cellSize, m_cellHeight, m_agentHeight, m_agentRadius, m_agentMaxClimb, m_agentMaxSlope, m_regionMinSize, m_regionMergeSize, m_edgeMaxLen, m_edgeMaxError, m_vertsPerPoly, m_detailSampleDist, m_detailSampleMaxError, settingsUI.IsFilterLowHangingObstacles(), settingsUI.IsFilterLedgeSpans(), settingsUI.IsFilterWalkableLowHeightSpans(), m_tileSize); } else { buildResult = soloNavMeshBuilder.Build(sample.GetInputGeom(), settingsUI.GetPartitioning(), m_cellSize, m_cellHeight, m_agentHeight, m_agentRadius, m_agentMaxClimb, m_agentMaxSlope, m_regionMinSize, m_regionMergeSize, m_edgeMaxLen, m_edgeMaxError, m_vertsPerPoly, m_detailSampleDist, m_detailSampleMaxError, settingsUI.IsFilterLowHangingObstacles(), settingsUI.IsFilterLedgeSpans(), settingsUI.IsFilterWalkableLowHeightSpans()); } sample.Update(sample.GetInputGeom(), buildResult.Item1, buildResult.Item2); sample.SetChanged(false); settingsUI.SetBuildTime((FrequencyWatch.Ticks - t) / TimeSpan.TicksPerMillisecond); //settingsUI.SetBuildTelemetry(buildResult.Item1.Select(x => x.GetTelemetry()).ToList()); toolsUI.SetSample(sample); Logger.Information($"build times"); Logger.Information($"-----------------------------------------"); var telemetries = buildResult.Item1 .Select(x => x.GetTelemetry()) .SelectMany(x => x.ToList()) .GroupBy(x => x.Item1) .ToImmutableSortedDictionary(x => x.Key, x => x.Sum(y => y.Item2)); foreach (var (key, millis) in telemetries) { Logger.Information($"{key}: {millis} ms"); } } } else { building = false; } if (!_mouseOverMenu) { GLU.GlhUnProjectf(mousePos[0], viewport[3] - 1 - mousePos[1], 0.0f, modelviewMatrix, projectionMatrix, viewport, ref rayStart); GLU.GlhUnProjectf(mousePos[0], viewport[3] - 1 - mousePos[1], 1.0f, modelviewMatrix, projectionMatrix, viewport, ref rayEnd); // Hit test mesh. DemoInputGeomProvider inputGeom = sample.GetInputGeom(); if (processHitTest && sample != null) { float? hit = null; if (inputGeom != null) { hit = inputGeom.RaycastMesh(rayStart, rayEnd); } if (!hit.HasValue && sample.GetNavMesh() != null) { hit = NavMeshRaycast.Raycast(sample.GetNavMesh(), rayStart, rayEnd); } if (!hit.HasValue && sample.GetRecastResults() != null) { hit = PolyMeshRaycast.Raycast(sample.GetRecastResults(), rayStart, rayEnd); } float[] rayDir = new float[] { rayEnd.x - rayStart.x, rayEnd.y - rayStart.y, rayEnd.z - rayStart.z }; Tool rayTool = toolsUI.GetTool(); VNormalize(rayDir); if (rayTool != null) { rayTool.HandleClickRay(rayStart, rayDir, processHitTestShift); } if (hit.HasValue) { float hitTime = hit.Value; if (0 != modState) { // Marker markerPositionSet = true; markerPosition.x = rayStart.x + (rayEnd.x - rayStart.x) * hitTime; markerPosition.y = rayStart.y + (rayEnd.y - rayStart.y) * hitTime; markerPosition.z = rayStart.z + (rayEnd.z - rayStart.z) * hitTime; } else { Vector3f pos = new Vector3f(); pos.x = rayStart.x + (rayEnd.x - rayStart.x) * hitTime; pos.y = rayStart.y + (rayEnd.y - rayStart.y) * hitTime; pos.z = rayStart.z + (rayEnd.z - rayStart.z) * hitTime; if (rayTool != null) { rayTool.HandleClick(rayStart, pos, processHitTestShift); } } } else { if (0 != modState) { // Marker markerPositionSet = false; } } } processHitTest = false; } if (sample.IsChanged()) { Vector3f? bminN = null; Vector3f? bmaxN = null; if (sample.GetInputGeom() != null) { bminN = sample.GetInputGeom().GetMeshBoundsMin(); bmaxN = sample.GetInputGeom().GetMeshBoundsMax(); } else if (sample.GetNavMesh() != null) { Vector3f[] bounds = NavMeshUtils.GetNavMeshBounds(sample.GetNavMesh()); bminN = bounds[0]; bmaxN = bounds[1]; } else if (0 < sample.GetRecastResults().Count) { foreach (RecastBuilderResult result in sample.GetRecastResults()) { if (result.GetSolidHeightfield() != null) { if (bminN == null) { bminN = Vector3f.Of(float.PositiveInfinity, float.PositiveInfinity, float.PositiveInfinity); bmaxN = Vector3f.Of(float.NegativeInfinity, float.NegativeInfinity, float.NegativeInfinity); } bminN = Vector3f.Of( Math.Min(bminN.Value.x, result.GetSolidHeightfield().bmin.x), Math.Min(bminN.Value.y, result.GetSolidHeightfield().bmin.y), Math.Min(bminN.Value.z, result.GetSolidHeightfield().bmin.z) ); bmaxN = Vector3f.Of( Math.Max(bmaxN.Value.x, result.GetSolidHeightfield().bmax.x), Math.Max(bmaxN.Value.y, result.GetSolidHeightfield().bmax.y), Math.Max(bmaxN.Value.z, result.GetSolidHeightfield().bmax.z) ); } } } if (bminN != null && bmaxN != null) { Vector3f bmin = bminN.Value; Vector3f bmax = bmaxN.Value; camr = (float)(Math.Sqrt( Sqr(bmax.x - bmin.x) + Sqr(bmax.y - bmin.y) + Sqr(bmax.z - bmin.z)) / 2); cameraPos.x = (bmax.x + bmin.x) / 2 + camr; cameraPos.y = (bmax.y + bmin.y) / 2 + camr; cameraPos.z = (bmax.z + bmin.z) / 2 + camr; camr *= 3; cameraEulers[0] = 45; cameraEulers[1] = -45; } sample.SetChanged(false); toolsUI.SetSample(sample); } var io = ImGui.GetIO(); io.DisplaySize = new Vector2(width, height); io.DisplayFramebufferScale = Vector2.One; io.DeltaTime = (float)dt; _canvas.Update(dt); _imgui.Update((float)dt); } private void OnWindowRender(double dt) { // Clear the screen dd.Clear(); projectionMatrix = dd.ProjectionMatrix(50f, (float)width / (float)height, 1.0f, camr); modelviewMatrix = dd.ViewMatrix(cameraPos, cameraEulers); dd.Fog(camr * 0.1f, camr * 1.25f); renderer.Render(sample); Tool tool = toolsUI.GetTool(); if (tool != null) { tool.HandleRender(renderer); } dd.Fog(false); _canvas.Draw(dt); _mouseOverMenu = _canvas.IsMouseOverUI(); _imgui.Render(); window.SwapBuffers(); } }