Files
libremetaverse/PrimMesher/ObjMesh.cs
2025-05-28 19:34:27 -05:00

213 lines
7.4 KiB
C#

/*
* Copyright (c) Contributors
* See CONTRIBUTORS.TXT for a full list of copyright holders.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are met:
* * Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* * Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
* * Neither the name of the OpenSimulator Project nor the
* names of its contributors may be used to endorse or promote products
* derived from this software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE DEVELOPERS ``AS IS'' AND ANY
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
* DISCLAIMED. IN NO EVENT SHALL THE CONTRIBUTORS BE LIABLE FOR ANY
* DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
using System.Collections.Generic;
using System.Globalization;
using System.IO;
using System.Text.RegularExpressions;
namespace LibreMetaverse.PrimMesher
{
public class ObjMesh
{
private readonly List<Coord> coords = new List<Coord>();
private List<ViewerPolygon> facePolygons = new List<ViewerPolygon>();
private List<ViewerVertex> faceVertices = new List<ViewerVertex>();
public string meshName = string.Empty;
private readonly List<Coord> normals = new List<Coord>();
public int numPrimFaces;
private readonly List<UVCoord> uvs = new List<UVCoord>();
public List<List<ViewerPolygon>> viewerPolygons = new List<List<ViewerPolygon>>();
private Dictionary<int, int> viewerVertexLookup = new Dictionary<int, int>();
public List<List<ViewerVertex>> viewerVertices = new List<List<ViewerVertex>>();
public ObjMesh(string path)
{
ProcessStream(new StreamReader(path));
}
public ObjMesh(StreamReader sr)
{
ProcessStream(sr);
}
private void ProcessStream(StreamReader s)
{
numPrimFaces = 0;
while (!s.EndOfStream)
{
var line = s.ReadLine()?.Trim();
var tokens = Regex.Split(line, @"\s+");
// Skip blank lines and comments
if (tokens.Length > 0 && tokens[0] != string.Empty && !tokens[0].StartsWith("#"))
ProcessTokens(tokens);
}
MakePrimFace();
}
public VertexIndexer GetVertexIndexer()
{
var vi = new VertexIndexer();
vi.numPrimFaces = numPrimFaces;
vi.viewerPolygons = viewerPolygons;
vi.viewerVertices = viewerVertices;
return vi;
}
private void ProcessTokens(string[] tokens)
{
switch (tokens[0].ToLower())
{
case "o":
meshName = tokens[1];
break;
case "v":
coords.Add(ParseCoord(tokens));
break;
case "vt":
{
uvs.Add(ParseUVCoord(tokens));
break;
}
case "vn":
normals.Add(ParseCoord(tokens));
break;
case "g":
MakePrimFace();
break;
case "s":
break;
case "f":
var vertIndices = new int[3];
for (var vertexIndex = 1; vertexIndex <= 3; vertexIndex++)
{
var indices = tokens[vertexIndex].Split('/');
var positionIndex = int.Parse(indices[0],
CultureInfo.InvariantCulture) - 1;
var texCoordIndex = -1;
var normalIndex = -1;
if (indices.Length > 1)
if (int.TryParse(indices[1], NumberStyles.Integer, CultureInfo.InvariantCulture,
out texCoordIndex))
texCoordIndex--;
else texCoordIndex = -1;
if (indices.Length > 2)
if (int.TryParse(indices[1], NumberStyles.Integer, CultureInfo.InvariantCulture,
out normalIndex))
normalIndex--;
else normalIndex = -1;
var hash = hashInts(positionIndex, texCoordIndex, normalIndex);
if (viewerVertexLookup.TryGetValue(hash, out var value))
{
vertIndices[vertexIndex - 1] = viewerVertexLookup[hash];
}
else
{
var vv = new ViewerVertex {v = coords[positionIndex]};
if (normalIndex > -1)
vv.n = normals[normalIndex];
if (texCoordIndex > -1)
vv.uv = uvs[texCoordIndex];
faceVertices.Add(vv);
vertIndices[vertexIndex - 1] = viewerVertexLookup[hash] = faceVertices.Count - 1;
}
}
facePolygons.Add(new ViewerPolygon(vertIndices[0], vertIndices[1], vertIndices[2]));
break;
case "mtllib":
break;
case "usemtl":
break;
}
}
private void MakePrimFace()
{
if (faceVertices.Count > 0 && facePolygons.Count > 0)
{
viewerVertices.Add(faceVertices);
faceVertices = new List<ViewerVertex>();
viewerPolygons.Add(facePolygons);
facePolygons = new List<ViewerPolygon>();
viewerVertexLookup = new Dictionary<int, int>();
numPrimFaces++;
}
}
private UVCoord ParseUVCoord(string[] tokens)
{
return new UVCoord(
float.Parse(tokens[1], CultureInfo.InvariantCulture),
float.Parse(tokens[1], CultureInfo.InvariantCulture));
}
private Coord ParseCoord(string[] tokens)
{
return new Coord(
float.Parse(tokens[1], CultureInfo.InvariantCulture),
float.Parse(tokens[2], CultureInfo.InvariantCulture),
float.Parse(tokens[3], CultureInfo.InvariantCulture));
}
private int hashInts(int i1, int i2, int i3)
{
return (i1 + " " + i2 + " " + i3).GetHashCode();
}
}
}