|
using UnityEngine; |
|
using UnityEditor.AssetImporters; |
|
using System; |
|
using System.IO; |
|
using System.Text.RegularExpressions; |
|
using System.Collections.Generic; |
|
|
|
[ScriptedImporter(1, "fim")] |
|
public class FIMImporter : ScriptedImporter |
|
{ |
|
public float m_Scale = 1; |
|
|
|
public override void OnImportAsset(AssetImportContext ctx) |
|
{ |
|
var tokens = Regex.Split(File.ReadAllText(ctx.assetPath), @"\s+"); |
|
|
|
var i = 0; |
|
var depth = 0; |
|
var num_meshes = 0; |
|
for (; i < tokens.Length; i++) { |
|
var t = tokens[i]; |
|
switch (t) { |
|
case "": |
|
// Edge case where the end of the file has a newline/space/etc. |
|
break; |
|
case "MESH": |
|
++i; |
|
++depth; |
|
|
|
var mesh_id = num_meshes++; |
|
var mesh = new Mesh(); |
|
|
|
// Meta information (collected first, then applied after) |
|
var up_axis = Vector3.up; |
|
|
|
for (; i < tokens.Length; i++) { |
|
var t2 = tokens[i]; |
|
switch (t2) { |
|
case "HSEM": |
|
--depth; |
|
goto commit_mesh; |
|
case "META": |
|
++i; |
|
++depth; |
|
|
|
for (; i < tokens.Length; i++) { |
|
var t3 = tokens[i]; |
|
if (t3 == "ATEM") { |
|
--depth; |
|
break; |
|
} |
|
|
|
if (++i >= tokens.Length) { |
|
// Meh, we could probably handle this better but this is simpler. |
|
throw new Exception(string.Format("unexpected EOF")); |
|
} |
|
|
|
var k = t3; |
|
var v = tokens[i]; |
|
|
|
switch (k) { |
|
case "upaxis": |
|
switch (v) { |
|
case "+Z": |
|
up_axis = new Vector3(0, 0, 1); |
|
break; |
|
case "-Z": |
|
up_axis = new Vector3(0, 0, -1); |
|
break; |
|
case "+Y": |
|
up_axis = new Vector3(0, 1, 0); |
|
break; |
|
case "-Y": |
|
up_axis = new Vector3(0, -1, 0); |
|
break; |
|
case "+X": |
|
up_axis = new Vector3(1, 0, 0); |
|
break; |
|
case "-X": |
|
up_axis = new Vector3(-1, 0, 0); |
|
break; |
|
default: |
|
Debug.LogWarningFormat("FIM meta upaxis has unknown axis: {0} (mesh offset {1}; skipping upaxis)", v, mesh_id); |
|
break; |
|
} |
|
break; |
|
case "forwardaxis": |
|
// Unused property |
|
break; |
|
default: |
|
++i; |
|
Debug.LogWarningFormat("FIM meta unknown property name: {0} (mesh offset {1}; skipping meta)", k, mesh_id); |
|
break; |
|
} |
|
} |
|
|
|
break; |
|
case "VERT": |
|
++i; |
|
++depth; |
|
|
|
var vertices = new List<Vector3>(); |
|
for (; i < tokens.Length; i++) { |
|
var t3 = tokens[i]; |
|
if (t3 == "TREV") { |
|
--depth; |
|
break; |
|
} |
|
|
|
string[] vals = t3.Split(','); |
|
if (vals.Length != 3) { |
|
Debug.LogWarningFormat("FIM mesh vertex is malformed: {0} (in mesh offset {1}; skipping vertex)", t3, mesh_id); |
|
continue; |
|
} |
|
|
|
var vert = new Vector3(float.Parse(vals[0]), float.Parse(vals[1]), float.Parse(vals[2])); |
|
vertices.Add(vert); |
|
} |
|
|
|
mesh.vertices = vertices.ToArray(); |
|
break; |
|
case "NORM": |
|
++i; |
|
++depth; |
|
|
|
var normals = new List<Vector3>(); |
|
for (; i < tokens.Length; i++) { |
|
var t3 = tokens[i]; |
|
if (t3 == "MRON") { |
|
--depth; |
|
break; |
|
} |
|
|
|
string[] vals = t3.Split(','); |
|
if (vals.Length != 3) { |
|
Debug.LogWarningFormat("FIM mesh normal is malformed: {0} (in mesh offset {1}; skipping normal)", t3, mesh_id); |
|
continue; |
|
} |
|
|
|
var norm = new Vector3(float.Parse(vals[0]), float.Parse(vals[1]), float.Parse(vals[2])); |
|
normals.Add(norm); |
|
} |
|
|
|
mesh.normals = normals.ToArray(); |
|
break; |
|
case "FACE": |
|
++i; |
|
++depth; |
|
|
|
var tri_list = new List<int>(); |
|
for (; i < tokens.Length; i++) { |
|
var t3 = tokens[i]; |
|
if (t3 == "ECAF") { |
|
--depth; |
|
break; |
|
} |
|
|
|
tri_list.Add(int.Parse(t3)); |
|
} |
|
|
|
mesh.triangles = tri_list.ToArray(); |
|
break; |
|
case "COLR": |
|
++i; |
|
++depth; |
|
|
|
var colors = new List<Color>(); |
|
for (; i < tokens.Length; i++) { |
|
var t3 = tokens[i]; |
|
if (t3 == "RLOC") { |
|
--depth; |
|
break; |
|
} |
|
|
|
string[] vals = t3.Split(','); |
|
if (vals.Length < 3 || vals.Length > 4) { |
|
Debug.LogWarningFormat("FIM mesh color is malformed: {0} (in mesh offset {1}; skipping color)", t3, mesh_id); |
|
continue; |
|
} |
|
|
|
var color = new Color(float.Parse(vals[0]), float.Parse(vals[1]), float.Parse(vals[2]), vals.Length == 4 ? float.Parse(vals[3]) : 1.0f); |
|
colors.Add(color); |
|
} |
|
|
|
mesh.colors = colors.ToArray(); |
|
break; |
|
default: |
|
Debug.LogWarningFormat("FIM unknown MESH block token: {0} (in mesh offset {1}; skipping block)", t2, mesh_id); |
|
{ |
|
var end = ReverseString(t2); |
|
++depth; |
|
for (; i < tokens.Length; i++) { |
|
if (tokens[i] == end) { |
|
--depth; |
|
break; |
|
} |
|
} |
|
} |
|
break; |
|
} |
|
} |
|
|
|
break; |
|
commit_mesh: |
|
ReorientMesh(mesh, up_axis); |
|
ctx.AddObjectToAsset(mesh_id.ToString(), mesh); |
|
break; |
|
default: |
|
Debug.LogWarningFormat("FIM unknown top-level token: {0} (skipping block)", t); |
|
{ |
|
var end = ReverseString(t); |
|
++depth; |
|
for (; i < tokens.Length; i++) { |
|
if (tokens[i] == end) { |
|
--depth; |
|
break; |
|
} |
|
} |
|
} |
|
break; |
|
} |
|
} |
|
|
|
if (depth > 0) { |
|
throw new System.Exception("FIM parser hit EOF unexpectedly"); |
|
} |
|
|
|
|
|
} |
|
|
|
static string ReverseString(string input) |
|
{ |
|
char[] charArray = input.ToCharArray(); |
|
Array.Reverse(charArray); |
|
return new string(charArray); |
|
} |
|
|
|
static void ReorientMesh(Mesh mesh, Vector3 up_axis) { |
|
if (up_axis == Vector3.up) return; |
|
|
|
// Calculate the rotation quaternion to align the mesh with the custom axes |
|
Quaternion rotation = Quaternion.FromToRotation(up_axis, Vector3.up); |
|
|
|
// Get the vertices and normals from the mesh |
|
Vector3[] vertices = mesh.vertices; |
|
Vector3[] normals = mesh.normals; |
|
|
|
// Apply the rotation to the vertices and normals |
|
for (int i = 0; i < vertices.Length; i++) |
|
{ |
|
vertices[i] = rotation * vertices[i]; |
|
normals[i] = rotation * normals[i]; |
|
} |
|
|
|
// Update the modified vertices and normals back to the mesh |
|
mesh.vertices = vertices; |
|
mesh.normals = normals; |
|
|
|
// Recalculate the bounds and tangents of the mesh |
|
mesh.RecalculateBounds(); |
|
mesh.RecalculateTangents(); |
|
} |
|
} |