2024-07-21 21:13:31 +00:00
|
|
|
using Godot;
|
|
|
|
using KeepersCompound.LGS;
|
|
|
|
using KeepersCompound.LGS.Database;
|
|
|
|
using KeepersCompound.LGS.Database.Chunks;
|
|
|
|
using Microsoft.VisualBasic;
|
|
|
|
using System;
|
|
|
|
using System.Collections.Generic;
|
|
|
|
|
|
|
|
namespace KeepersCompound;
|
|
|
|
|
|
|
|
[Tool]
|
|
|
|
public partial class Mission : Node3D
|
|
|
|
{
|
|
|
|
[Export(PropertyHint.GlobalFile, "*.mis")]
|
|
|
|
public string FileName { get; set; }
|
|
|
|
[Export]
|
|
|
|
public bool Build = false;
|
2024-07-22 18:37:27 +00:00
|
|
|
[Export]
|
|
|
|
public bool Clear = false;
|
2024-07-21 21:13:31 +00:00
|
|
|
|
|
|
|
|
|
|
|
DbFile _file;
|
|
|
|
|
|
|
|
public override void _Ready()
|
|
|
|
{
|
|
|
|
RebuildMap();
|
|
|
|
}
|
|
|
|
|
|
|
|
public override void _Process(double delta)
|
|
|
|
{
|
|
|
|
if (Build)
|
|
|
|
{
|
|
|
|
RebuildMap();
|
|
|
|
Build = false;
|
|
|
|
}
|
2024-07-22 18:37:27 +00:00
|
|
|
if (Clear)
|
|
|
|
{
|
|
|
|
ClearMap();
|
|
|
|
Clear = false;
|
|
|
|
}
|
2024-07-21 21:13:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public override void _Input(InputEvent @event)
|
|
|
|
{
|
|
|
|
if (@event is InputEventKey keyEvent && keyEvent.Pressed)
|
|
|
|
{
|
|
|
|
if (keyEvent.Keycode == Key.R)
|
|
|
|
{
|
|
|
|
Build = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-07-22 18:37:27 +00:00
|
|
|
public void ClearMap()
|
2024-07-21 21:13:31 +00:00
|
|
|
{
|
|
|
|
foreach (var node in GetChildren())
|
|
|
|
{
|
|
|
|
node.QueueFree();
|
|
|
|
}
|
2024-07-22 18:37:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public void RebuildMap()
|
|
|
|
{
|
|
|
|
ClearMap();
|
2024-07-21 21:13:31 +00:00
|
|
|
|
|
|
|
_file = new(FileName);
|
|
|
|
var wr = (WorldRep)_file.Chunks["WREXT"];
|
|
|
|
|
|
|
|
foreach (var cell in wr.Cells)
|
|
|
|
{
|
|
|
|
BuildCellMesh(cell);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private void BuildCellMesh(WorldRep.Cell cell)
|
|
|
|
{
|
|
|
|
var numPolys = cell.PolyCount;
|
|
|
|
var numRenderPolys = cell.RenderPolyCount;
|
|
|
|
var numPortalPolys = cell.PortalPolyCount;
|
|
|
|
|
2024-07-22 18:37:27 +00:00
|
|
|
if (numRenderPolys == 0 || numPortalPolys >= numPolys)
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2024-07-21 21:13:31 +00:00
|
|
|
var vertices = new List<Vector3>();
|
|
|
|
var normals = new List<Vector3>();
|
2024-07-22 18:37:56 +00:00
|
|
|
var indices = new List<int>();
|
2024-07-21 21:13:31 +00:00
|
|
|
var idxOffset = 0;
|
|
|
|
for (int i = 0; i < numPolys; i++)
|
|
|
|
{
|
|
|
|
var poly = cell.Polys[i];
|
|
|
|
// TODO: Make this break lol
|
|
|
|
if (i >= numRenderPolys || i >= numPolys - numPortalPolys)
|
|
|
|
{
|
|
|
|
idxOffset += poly.VertexCount;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
var plane = cell.Planes[poly.PlaneId];
|
|
|
|
var normal = plane.Normal;
|
|
|
|
|
|
|
|
var numPolyVertices = poly.VertexCount;
|
2024-07-22 18:37:56 +00:00
|
|
|
var baseIdx = vertices.Count;
|
|
|
|
for (var j = 0; j < numPolyVertices; j++)
|
|
|
|
{
|
|
|
|
var vertex = cell.Vertices[cell.Indices[idxOffset + j]];
|
|
|
|
vertices.Add(vertex.ToGodotVec3(4.0f));
|
|
|
|
normals.Add(normal.ToGodotVec3(4.0f));
|
|
|
|
}
|
|
|
|
|
2024-07-21 21:13:31 +00:00
|
|
|
for (int j = 1; j < numPolyVertices - 1; j++)
|
|
|
|
{
|
|
|
|
foreach (var offset in new int[] { 0, j, j + 1 })
|
|
|
|
{
|
2024-07-22 18:37:56 +00:00
|
|
|
indices.Add(baseIdx + offset);
|
2024-07-21 21:13:31 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
idxOffset += poly.VertexCount;
|
|
|
|
}
|
|
|
|
|
2024-07-22 18:52:40 +00:00
|
|
|
var cellNode = new Node3D();
|
|
|
|
|
2024-07-21 21:13:31 +00:00
|
|
|
var arrMesh = new ArrayMesh();
|
|
|
|
var arrays = new Godot.Collections.Array();
|
|
|
|
arrays.Resize((int)Mesh.ArrayType.Max);
|
|
|
|
arrays[(int)Mesh.ArrayType.Vertex] = vertices.ToArray();
|
|
|
|
arrays[(int)Mesh.ArrayType.Normal] = normals.ToArray();
|
2024-07-22 18:37:56 +00:00
|
|
|
arrays[(int)Mesh.ArrayType.Index] = indices.ToArray();
|
2024-07-21 21:13:31 +00:00
|
|
|
|
|
|
|
arrMesh.AddSurfaceFromArrays(Mesh.PrimitiveType.Triangles, arrays);
|
|
|
|
var meshInstance = new MeshInstance3D
|
|
|
|
{
|
|
|
|
Mesh = arrMesh,
|
|
|
|
CastShadow = GeometryInstance3D.ShadowCastingSetting.On
|
|
|
|
};
|
2024-07-22 18:52:40 +00:00
|
|
|
cellNode.AddChild(meshInstance);
|
|
|
|
|
|
|
|
var occluderInstance = new OccluderInstance3D();
|
|
|
|
var occluder = new ArrayOccluder3D();
|
|
|
|
occluder.SetArrays(vertices.ToArray(), indices.ToArray());
|
|
|
|
occluderInstance.Occluder = occluder;
|
2024-07-22 19:17:12 +00:00
|
|
|
occluderInstance.BakeSimplificationDistance = 0.0f;
|
2024-07-22 18:52:40 +00:00
|
|
|
cellNode.AddChild(occluderInstance);
|
|
|
|
|
2024-07-22 18:37:27 +00:00
|
|
|
|
|
|
|
var r = new Random();
|
|
|
|
if (r.NextSingle() > 0.9 && cell.SphereRadius > 5.0)
|
|
|
|
{
|
|
|
|
var light = new OmniLight3D
|
|
|
|
{
|
|
|
|
Position = cell.SphereCenter.ToGodotVec3(4.0f),
|
|
|
|
OmniRange = cell.SphereRadius * (r.NextSingle() + 1.0f) * 0.5f,
|
|
|
|
};
|
2024-07-22 18:52:40 +00:00
|
|
|
cellNode.AddChild(light);
|
2024-07-22 18:37:27 +00:00
|
|
|
}
|
2024-07-22 18:52:40 +00:00
|
|
|
|
|
|
|
AddChild(cellNode);
|
2024-07-21 21:13:31 +00:00
|
|
|
}
|
|
|
|
}
|