2024-08-25 15:47:05 +00:00
|
|
|
using System.Collections.Generic;
|
2024-09-05 17:49:55 +00:00
|
|
|
using System.IO;
|
2024-08-25 15:47:05 +00:00
|
|
|
using Godot;
|
|
|
|
using KeepersCompound.LGS;
|
|
|
|
|
|
|
|
namespace KeepersCompound.TMV;
|
|
|
|
|
2024-08-26 11:38:13 +00:00
|
|
|
public class ModelLoader
|
2024-08-25 15:47:05 +00:00
|
|
|
{
|
2024-09-05 17:52:44 +00:00
|
|
|
private readonly Dictionary<(string, string), MeshInstance3D> _cache = new();
|
2024-08-26 11:38:13 +00:00
|
|
|
|
|
|
|
public MeshInstance3D Load(string campaignName, string modelName, bool forceLoad = false)
|
|
|
|
{
|
|
|
|
campaignName ??= "";
|
|
|
|
|
|
|
|
if (!forceLoad)
|
|
|
|
{
|
|
|
|
if (_cache.TryGetValue((campaignName, modelName), out var fmModel))
|
|
|
|
{
|
2024-08-30 16:56:28 +00:00
|
|
|
return fmModel?.Duplicate() as MeshInstance3D;
|
2024-08-26 11:38:13 +00:00
|
|
|
}
|
|
|
|
else if (_cache.TryGetValue(("", modelName), out var omModel))
|
|
|
|
{
|
2024-08-30 16:56:28 +00:00
|
|
|
return omModel?.Duplicate() as MeshInstance3D;
|
2024-08-26 11:38:13 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// We don't care if this is null actually, we'll still cache that it's null lol
|
2024-09-05 17:52:44 +00:00
|
|
|
var model = Timing.TimeStage("Load Models", () => { return LoadModel(ref campaignName, modelName); });
|
2024-08-29 17:09:10 +00:00
|
|
|
_cache[(campaignName, modelName)] = model;
|
2024-08-31 11:03:29 +00:00
|
|
|
return model?.Duplicate() as MeshInstance3D;
|
2024-08-26 11:38:13 +00:00
|
|
|
}
|
|
|
|
|
2024-09-05 17:52:44 +00:00
|
|
|
public static MeshInstance3D LoadModel(ref string campaignName, string modelName)
|
2024-08-25 15:47:05 +00:00
|
|
|
{
|
2024-09-05 17:52:44 +00:00
|
|
|
var pathManager = Context.Instance.PathManager;
|
2024-09-05 17:49:55 +00:00
|
|
|
var (newCampaignName, modelPath) = pathManager.GetResourcePath(ResourceType.Object, campaignName, modelName);
|
|
|
|
campaignName = newCampaignName;
|
2024-08-25 15:47:05 +00:00
|
|
|
if (modelPath == null)
|
|
|
|
{
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
var modelFile = new ModelFile(modelPath);
|
|
|
|
if (modelFile == null)
|
|
|
|
{
|
|
|
|
GD.Print($"Failed to load model file: {modelPath}");
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
var materials = new List<StandardMaterial3D>();
|
|
|
|
foreach (var material in modelFile.Materials)
|
|
|
|
{
|
|
|
|
if (material.Type == 0)
|
|
|
|
{
|
2024-09-07 07:01:15 +00:00
|
|
|
var convertedName = ResourcePathManager.ConvertSeparator(material.Name);
|
|
|
|
var resName = Path.GetFileNameWithoutExtension(convertedName);
|
2024-09-05 17:49:55 +00:00
|
|
|
var (_, path) = pathManager.GetResourcePath(ResourceType.ObjectTexture, campaignName, resName);
|
2024-08-25 15:49:43 +00:00
|
|
|
if (path == null)
|
|
|
|
{
|
2024-09-07 12:43:11 +00:00
|
|
|
// Might fail in exported projects
|
|
|
|
path = "res://project/jorge.png";
|
2024-08-25 15:49:43 +00:00
|
|
|
GD.Print($"Failed to load model texture: {material.Name}");
|
|
|
|
}
|
2024-08-25 15:47:05 +00:00
|
|
|
|
2024-08-31 10:59:54 +00:00
|
|
|
var mat = new StandardMaterial3D
|
2024-08-25 15:47:05 +00:00
|
|
|
{
|
2024-09-01 12:51:05 +00:00
|
|
|
AlbedoTexture = TextureLoader.LoadTexture(path),
|
|
|
|
Transparency = BaseMaterial3D.TransparencyEnum.AlphaDepthPrePass,
|
2024-08-31 10:59:54 +00:00
|
|
|
};
|
|
|
|
var name = material.Name.ToLower();
|
|
|
|
for (var i = 0; i < 4; i++)
|
|
|
|
{
|
|
|
|
if (name.Contains($"replace{i}"))
|
|
|
|
{
|
|
|
|
mat.SetMeta($"TxtRepl{i}", true);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
materials.Add(mat);
|
2024-08-25 15:47:05 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
var b = (material.Handle) & 0xff;
|
|
|
|
var g = (material.Handle >> 8) & 0xff;
|
|
|
|
var r = (material.Handle >> 16) & 0xff;
|
|
|
|
var colour = new Color(r / 255.0f, g / 255.0f, b / 255.0f, 1.0f);
|
|
|
|
materials.Add(new StandardMaterial3D
|
|
|
|
{
|
|
|
|
AlbedoColor = colour
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
var surfaceDataMap = new Dictionary<int, MeshSurfaceData>();
|
|
|
|
foreach (var poly in modelFile.Polygons)
|
|
|
|
{
|
|
|
|
var vertices = new List<Vector3>();
|
|
|
|
var normal = modelFile.Normals[poly.Normal].ToGodotVec3();
|
|
|
|
var uvs = new List<Vector2>();
|
|
|
|
for (var i = 0; i < poly.VertexCount; i++)
|
|
|
|
{
|
|
|
|
var vertex = modelFile.Vertices[poly.VertexIndices[i]];
|
|
|
|
vertices.Add(vertex.ToGodotVec3());
|
|
|
|
if (i < poly.UvIndices.Length)
|
|
|
|
{
|
|
|
|
var uv = modelFile.Uvs[poly.UvIndices[i]];
|
|
|
|
uvs.Add(new Vector2(uv.X, uv.Y));
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
uvs.Add(Vector2.Zero);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!surfaceDataMap.ContainsKey(poly.Data))
|
|
|
|
{
|
|
|
|
surfaceDataMap.Add(poly.Data, new MeshSurfaceData());
|
|
|
|
}
|
|
|
|
|
|
|
|
surfaceDataMap[poly.Data].AddPolygon(vertices, normal, uvs, uvs);
|
|
|
|
}
|
|
|
|
|
|
|
|
var mesh = new ArrayMesh();
|
|
|
|
foreach (var (materialId, surfaceData) in surfaceDataMap)
|
|
|
|
{
|
|
|
|
var array = surfaceData.BuildSurfaceArray();
|
|
|
|
var surfaceIdx = mesh.GetSurfaceCount();
|
|
|
|
mesh.AddSurfaceFromArrays(Mesh.PrimitiveType.Triangles, array);
|
|
|
|
for (var i = 0; i < materials.Count; i++)
|
|
|
|
{
|
|
|
|
var m = modelFile.Materials[i];
|
|
|
|
if (m.Slot == materialId)
|
|
|
|
{
|
|
|
|
mesh.SurfaceSetMaterial(surfaceIdx, materials[i]);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
var pos = -modelFile.Header.Center.ToGodotVec3();
|
|
|
|
var meshInstance = new MeshInstance3D { Mesh = mesh, Position = pos };
|
|
|
|
return meshInstance;
|
|
|
|
}
|
|
|
|
}
|