2019-04-30 23:07:25 +02:00
|
|
|
|
using System;
|
|
|
|
|
using System.Collections.Generic;
|
|
|
|
|
using System.Linq;
|
|
|
|
|
using System.Text;
|
|
|
|
|
using System.Threading.Tasks;
|
|
|
|
|
using Collada141;
|
2019-05-01 21:38:14 +02:00
|
|
|
|
using System.Runtime.InteropServices;
|
|
|
|
|
using System.Diagnostics;
|
|
|
|
|
using System.Globalization;
|
|
|
|
|
using System.Xml;
|
2019-06-13 23:31:35 +02:00
|
|
|
|
using OpenTK;
|
2019-07-16 23:35:21 +02:00
|
|
|
|
using Toolbox.Library.Rendering;
|
2019-11-12 23:11:35 +01:00
|
|
|
|
using Toolbox.Library.Collada;
|
2019-11-16 19:48:06 +01:00
|
|
|
|
using Toolbox.Library.IO;
|
2019-04-30 23:07:25 +02:00
|
|
|
|
|
2019-07-16 23:35:21 +02:00
|
|
|
|
namespace Toolbox.Library
|
2019-04-30 23:07:25 +02:00
|
|
|
|
{
|
2019-11-16 01:27:03 +01:00
|
|
|
|
public class DAE
|
2019-04-30 23:07:25 +02:00
|
|
|
|
{
|
2019-11-09 23:36:52 +01:00
|
|
|
|
public class ExportSettings
|
|
|
|
|
{
|
2019-11-16 22:51:20 +01:00
|
|
|
|
public bool SuppressConfirmDialog = false;
|
2019-11-16 19:03:24 +01:00
|
|
|
|
public bool OptmizeZeroWeights = true;
|
2019-11-16 14:58:10 +01:00
|
|
|
|
public bool UseOldExporter = false;
|
2019-11-23 23:04:58 +01:00
|
|
|
|
public bool UseVertexColors = true;
|
2019-11-16 01:27:03 +01:00
|
|
|
|
public bool FlipTexCoordsVertical = true;
|
2019-11-27 01:54:59 +01:00
|
|
|
|
public bool OnlyExportRiggedBones = false;
|
2020-05-05 23:23:11 +02:00
|
|
|
|
public bool UseTextureChannelComponents = true;
|
2019-11-16 01:27:03 +01:00
|
|
|
|
|
2019-12-12 02:37:21 +01:00
|
|
|
|
public bool TransformColorUVs = false;
|
|
|
|
|
|
2019-12-08 02:16:13 +01:00
|
|
|
|
public bool AddLeafBones = false;
|
|
|
|
|
|
2019-11-09 23:36:52 +01:00
|
|
|
|
public Version FileVersion = new Version();
|
2019-11-10 14:41:17 +01:00
|
|
|
|
|
2019-11-16 01:27:03 +01:00
|
|
|
|
public ProgramPreset Preset = ProgramPreset.NONE;
|
|
|
|
|
|
|
|
|
|
public bool ExportTextures = true;
|
|
|
|
|
|
2019-11-10 14:41:17 +01:00
|
|
|
|
public string ImageExtension = ".png";
|
|
|
|
|
public string ImageFolder = "";
|
2019-11-09 23:36:52 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public class Version
|
|
|
|
|
{
|
|
|
|
|
public int Major = 1;
|
|
|
|
|
public int Minor = 4;
|
|
|
|
|
public int Micro = 1;
|
|
|
|
|
}
|
|
|
|
|
|
2019-11-16 01:27:03 +01:00
|
|
|
|
|
|
|
|
|
public static void Export(string FileName, ExportSettings settings, STGenericObject mesh)
|
|
|
|
|
{
|
|
|
|
|
Export(FileName, settings, new List<STGenericObject>() { mesh },
|
|
|
|
|
new List<STGenericMaterial>(), new List<STGenericTexture>());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public static void Export(string FileName, ExportSettings settings, STGenericModel model, List<STGenericTexture> Textures, STSkeleton skeleton = null, List<int> NodeArray = null) {
|
|
|
|
|
Export(FileName, settings, model.Objects.ToList(), model.Materials.ToList(), Textures, skeleton, NodeArray);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public static void Export(string FileName, ExportSettings settings,
|
|
|
|
|
List<STGenericObject> Meshes, List<STGenericMaterial> Materials,
|
|
|
|
|
List<STGenericTexture> Textures, STSkeleton skeleton = null, List<int> NodeArray = null)
|
2019-11-09 23:36:52 +01:00
|
|
|
|
{
|
2019-11-16 01:27:03 +01:00
|
|
|
|
if (Materials == null)
|
|
|
|
|
Materials = new List<STGenericMaterial>();
|
2020-04-18 22:07:06 +02:00
|
|
|
|
if (skeleton != null && skeleton.BoneIndices != null)
|
|
|
|
|
NodeArray = skeleton.BoneIndices.ToList();
|
2019-11-16 01:27:03 +01:00
|
|
|
|
|
2019-11-17 14:59:11 +01:00
|
|
|
|
List<string> failedTextureExport = new List<string>();
|
|
|
|
|
|
2019-11-16 22:27:57 +01:00
|
|
|
|
STProgressBar progressBar = new STProgressBar();
|
|
|
|
|
progressBar.Task = "Exporting Model...";
|
|
|
|
|
progressBar.Value = 0;
|
|
|
|
|
progressBar.StartPosition = System.Windows.Forms.FormStartPosition.CenterScreen;
|
|
|
|
|
progressBar.Show();
|
|
|
|
|
progressBar.Refresh();
|
|
|
|
|
|
2019-11-16 14:58:10 +01:00
|
|
|
|
if (settings.UseOldExporter)
|
|
|
|
|
{
|
|
|
|
|
AssimpSaver saver = new AssimpSaver();
|
|
|
|
|
STGenericModel model = new STGenericModel();
|
|
|
|
|
model.Objects = Meshes;
|
|
|
|
|
model.Materials = Materials;
|
|
|
|
|
saver.SaveFromModel(model, FileName, Textures, skeleton, NodeArray);
|
2019-11-16 14:59:28 +01:00
|
|
|
|
return;
|
2019-11-16 14:58:10 +01:00
|
|
|
|
}
|
|
|
|
|
|
2019-11-16 01:27:03 +01:00
|
|
|
|
string TexturePath = System.IO.Path.GetDirectoryName(FileName);
|
2020-05-30 02:23:48 +02:00
|
|
|
|
Dictionary<string, STGenericMaterial> MaterialRemapper = new Dictionary<string, STGenericMaterial>();
|
2019-11-16 01:27:03 +01:00
|
|
|
|
|
|
|
|
|
using (ColladaWriter writer = new ColladaWriter(FileName, settings))
|
2019-11-09 23:36:52 +01:00
|
|
|
|
{
|
2019-11-16 01:27:03 +01:00
|
|
|
|
writer.WriteAsset();
|
|
|
|
|
|
|
|
|
|
if (Materials.Count > 0)
|
|
|
|
|
{
|
|
|
|
|
List<string> textureNames = new List<string>();
|
|
|
|
|
for (int i = 0; i < Textures?.Count; i++)
|
|
|
|
|
{
|
2020-01-15 02:43:36 +01:00
|
|
|
|
if (!textureNames.Contains(Textures[i].Text))
|
|
|
|
|
textureNames.Add(Textures[i].Text);
|
2019-11-16 01:27:03 +01:00
|
|
|
|
|
|
|
|
|
if (settings.ExportTextures) {
|
2019-11-16 22:27:57 +01:00
|
|
|
|
|
|
|
|
|
progressBar.Task = $"Exporting Texture {Textures[i].Text}";
|
|
|
|
|
progressBar.Value = ((i * 100) / Textures.Count);
|
|
|
|
|
progressBar.Refresh();
|
|
|
|
|
|
2019-11-17 14:59:11 +01:00
|
|
|
|
try
|
2019-11-16 19:48:06 +01:00
|
|
|
|
{
|
2019-11-17 14:59:11 +01:00
|
|
|
|
var bitmap = Textures[i].GetBitmap();
|
|
|
|
|
if (bitmap != null)
|
|
|
|
|
{
|
2020-05-05 23:23:11 +02:00
|
|
|
|
if (settings.UseTextureChannelComponents)
|
|
|
|
|
bitmap = Textures[i].GetComponentBitmap(bitmap);
|
2019-11-17 14:59:11 +01:00
|
|
|
|
string textureName = Textures[i].Text;
|
|
|
|
|
if (textureName.RemoveIllegaleFileNameCharacters() != textureName)
|
|
|
|
|
{
|
|
|
|
|
string properName = textureName.RemoveIllegaleFileNameCharacters();
|
|
|
|
|
for (int m = 0; m < Materials?.Count; m++)
|
|
|
|
|
{
|
|
|
|
|
foreach (var tex in Materials[m].TextureMaps)
|
|
|
|
|
{
|
|
|
|
|
if (tex.Name == textureName)
|
|
|
|
|
tex.Name = properName;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
textureName = properName;
|
2019-11-16 19:48:06 +01:00
|
|
|
|
}
|
|
|
|
|
|
2019-11-17 14:59:11 +01:00
|
|
|
|
bitmap.Save($"{TexturePath}/{textureName}.png");
|
|
|
|
|
bitmap.Dispose();
|
2019-11-16 01:27:03 +01:00
|
|
|
|
|
2019-11-17 14:59:11 +01:00
|
|
|
|
GC.Collect();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
catch (Exception ex) {
|
|
|
|
|
failedTextureExport.Add(Textures[i].Text);
|
|
|
|
|
}
|
2019-11-16 01:27:03 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
List<Material> materials = new List<Material>();
|
|
|
|
|
foreach (var mat in Materials)
|
|
|
|
|
{
|
|
|
|
|
Material material = new Material();
|
|
|
|
|
material.Name = mat.Text;
|
2020-05-30 02:23:48 +02:00
|
|
|
|
|
|
|
|
|
if (!MaterialRemapper.ContainsKey(mat.Text))
|
|
|
|
|
{
|
|
|
|
|
MaterialRemapper.Add(mat.Text, mat);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
string name = Utils.RenameDuplicateString(MaterialRemapper.Keys.ToList(), mat.Text);
|
|
|
|
|
MaterialRemapper.Add(name, mat);
|
|
|
|
|
material.Name = name;
|
|
|
|
|
}
|
|
|
|
|
|
2019-11-16 01:27:03 +01:00
|
|
|
|
materials.Add(material);
|
|
|
|
|
|
|
|
|
|
foreach (var tex in mat.TextureMaps)
|
|
|
|
|
{
|
|
|
|
|
TextureMap texMap = new TextureMap();
|
|
|
|
|
texMap.Name = tex.Name;
|
|
|
|
|
if (tex.Type == STGenericMatTexture.TextureType.Diffuse)
|
|
|
|
|
texMap.Type = PhongTextureType.diffuse;
|
|
|
|
|
else if (tex.Type == STGenericMatTexture.TextureType.Normal)
|
|
|
|
|
texMap.Type = PhongTextureType.bump;
|
|
|
|
|
else if (tex.Type == STGenericMatTexture.TextureType.Specular)
|
|
|
|
|
texMap.Type = PhongTextureType.specular;
|
|
|
|
|
else if (tex.Type == STGenericMatTexture.TextureType.Emission)
|
|
|
|
|
texMap.Type = PhongTextureType.emission;
|
|
|
|
|
else
|
|
|
|
|
continue; //Skip adding unknown types
|
|
|
|
|
|
|
|
|
|
if (tex.WrapModeS == STTextureWrapMode.Repeat)
|
|
|
|
|
texMap.WrapModeS = SamplerWrapMode.WRAP;
|
|
|
|
|
else if (tex.WrapModeS == STTextureWrapMode.Mirror)
|
|
|
|
|
texMap.WrapModeS = SamplerWrapMode.MIRROR;
|
|
|
|
|
else if (tex.WrapModeS == STTextureWrapMode.Clamp)
|
|
|
|
|
texMap.WrapModeS = SamplerWrapMode.CLAMP;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (tex.WrapModeT == STTextureWrapMode.Repeat)
|
|
|
|
|
texMap.WrapModeT = SamplerWrapMode.WRAP;
|
|
|
|
|
else if (tex.WrapModeT == STTextureWrapMode.Mirror)
|
|
|
|
|
texMap.WrapModeT = SamplerWrapMode.MIRROR;
|
|
|
|
|
else if(tex.WrapModeT == STTextureWrapMode.Clamp)
|
|
|
|
|
texMap.WrapModeT = SamplerWrapMode.CLAMP;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
//If no textures are saved, still keep images references
|
|
|
|
|
//So the user can still dump textures after
|
2020-01-15 02:43:36 +01:00
|
|
|
|
if (Textures?.Count == 0 && !textureNames.Contains(texMap.Name))
|
2019-11-16 01:27:03 +01:00
|
|
|
|
textureNames.Add($"{texMap.Name}");
|
|
|
|
|
|
|
|
|
|
material.Textures.Add(texMap);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
writer.WriteLibraryImages(textureNames.ToArray());
|
|
|
|
|
|
|
|
|
|
writer.WriteLibraryMaterials(materials);
|
|
|
|
|
writer.WriteLibraryEffects(materials);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
writer.WriteLibraryImages();
|
|
|
|
|
|
|
|
|
|
if (skeleton != null) {
|
2019-11-27 01:54:59 +01:00
|
|
|
|
//Search for bones with rigging first
|
|
|
|
|
List<string> riggedBones = new List<string>();
|
|
|
|
|
if (settings.OnlyExportRiggedBones)
|
|
|
|
|
{
|
|
|
|
|
for (int i = 0; i < Meshes.Count; i++)
|
|
|
|
|
{
|
|
|
|
|
for (int v = 0; v < Meshes[i].vertices.Count; v++)
|
|
|
|
|
{
|
|
|
|
|
var vertex = Meshes[i].vertices[v];
|
|
|
|
|
for (int j = 0; j < vertex.boneIds.Count; j++)
|
|
|
|
|
{
|
|
|
|
|
int id = -1;
|
|
|
|
|
if (NodeArray != null && NodeArray.Count > vertex.boneIds[j]) {
|
|
|
|
|
id = NodeArray[vertex.boneIds[j]];
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
id = vertex.boneIds[j];
|
|
|
|
|
|
|
|
|
|
if (id < skeleton.bones.Count && id != -1)
|
|
|
|
|
riggedBones.Add(skeleton.bones[id].Text);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2019-11-16 01:27:03 +01:00
|
|
|
|
foreach (var bone in skeleton.bones)
|
|
|
|
|
{
|
2019-11-27 01:54:59 +01:00
|
|
|
|
if (settings.OnlyExportRiggedBones && !riggedBones.Contains(bone.Text))
|
|
|
|
|
{
|
|
|
|
|
Console.WriteLine("Skipping " + bone.Text);
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
|
2019-11-16 01:27:03 +01:00
|
|
|
|
//Set the inverse matrix
|
2019-11-16 15:56:47 +01:00
|
|
|
|
var inverse = skeleton.GetBoneTransform(bone).Inverted();
|
2019-11-16 01:27:03 +01:00
|
|
|
|
var transform = bone.GetTransform();
|
|
|
|
|
|
|
|
|
|
float[] Transform = new float[] {
|
|
|
|
|
transform.M11, transform.M21, transform.M31, transform.M41,
|
|
|
|
|
transform.M12, transform.M22, transform.M32, transform.M42,
|
|
|
|
|
transform.M13, transform.M23, transform.M33, transform.M43,
|
|
|
|
|
transform.M14, transform.M24, transform.M34, transform.M44
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
float[] InvTransform = new float[] {
|
|
|
|
|
inverse.M11, inverse.M21, inverse.M31, inverse.M41,
|
|
|
|
|
inverse.M12, inverse.M22, inverse.M32, inverse.M42,
|
|
|
|
|
inverse.M13, inverse.M23, inverse.M33, inverse.M43,
|
|
|
|
|
inverse.M14, inverse.M24, inverse.M34, inverse.M44
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
writer.AddJoint(bone.Text, bone.parentIndex == -1 ? "" :
|
2020-02-08 18:03:37 +01:00
|
|
|
|
skeleton.bones[bone.parentIndex].Text, Transform, InvTransform,
|
|
|
|
|
new float[3] { bone.Position.X, bone.Position.Y, bone.Position.Z },
|
|
|
|
|
new float[3] { bone.EulerRotation.X, bone.EulerRotation.Y, bone.EulerRotation.Z },
|
|
|
|
|
new float[3] { bone.Scale.X, bone.Scale.Y, bone.Scale.Z });
|
2019-11-16 01:27:03 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2019-11-16 22:27:57 +01:00
|
|
|
|
int meshIndex = 0;
|
|
|
|
|
|
2019-11-16 01:27:03 +01:00
|
|
|
|
writer.StartLibraryGeometries();
|
|
|
|
|
foreach (var mesh in Meshes)
|
|
|
|
|
{
|
2019-11-16 22:27:57 +01:00
|
|
|
|
progressBar.Task = $"Exporting Mesh {mesh.Text}";
|
|
|
|
|
progressBar.Value = ((meshIndex++ * 100) / Meshes.Count);
|
|
|
|
|
progressBar.Refresh();
|
|
|
|
|
|
2019-11-16 01:27:03 +01:00
|
|
|
|
int[] IndexTable = null;
|
|
|
|
|
if (NodeArray != null)
|
|
|
|
|
IndexTable = NodeArray.ToArray();
|
|
|
|
|
|
|
|
|
|
writer.StartGeometry(mesh.Text);
|
|
|
|
|
|
2019-11-16 22:54:15 +01:00
|
|
|
|
if (mesh.MaterialIndex != -1 && Materials.Count > mesh.MaterialIndex)
|
2019-11-16 01:27:03 +01:00
|
|
|
|
{
|
|
|
|
|
writer.CurrentMaterial = Materials[mesh.MaterialIndex].Text;
|
2020-04-01 17:45:08 +02:00
|
|
|
|
Console.WriteLine($"MaterialIndex {mesh.MaterialIndex } {Materials[mesh.MaterialIndex].Text}");
|
2019-11-16 01:27:03 +01:00
|
|
|
|
}
|
|
|
|
|
|
2020-01-18 23:30:19 +01:00
|
|
|
|
|
|
|
|
|
if (settings.TransformColorUVs)
|
|
|
|
|
{
|
2020-01-18 23:41:36 +01:00
|
|
|
|
List<Vertex> transformedVertices = new List<Vertex>();
|
2020-01-18 23:30:19 +01:00
|
|
|
|
foreach (var poly in mesh.PolygonGroups)
|
|
|
|
|
{
|
|
|
|
|
var mat = poly.Material;
|
|
|
|
|
if (mat == null) continue;
|
|
|
|
|
|
|
|
|
|
var faces = poly.GetDisplayFace();
|
|
|
|
|
for (int v = 0; v < poly.displayFaceSize; v += 3)
|
|
|
|
|
{
|
|
|
|
|
if (faces.Count < v + 2)
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
var diffuse = mat.TextureMaps.FirstOrDefault(x => x.Type == STGenericMatTexture.TextureType.Diffuse);
|
|
|
|
|
STTextureTransform transform = new STTextureTransform();
|
|
|
|
|
if (diffuse != null)
|
|
|
|
|
transform = diffuse.Transform;
|
|
|
|
|
|
2020-01-18 23:41:36 +01:00
|
|
|
|
var vertexA = mesh.vertices[faces[v]];
|
|
|
|
|
var vertexB = mesh.vertices[faces[v+1]];
|
|
|
|
|
var vertexC = mesh.vertices[faces[v+2]];
|
2020-01-18 23:30:19 +01:00
|
|
|
|
|
2020-01-18 23:41:36 +01:00
|
|
|
|
if (!transformedVertices.Contains(vertexA)) {
|
|
|
|
|
vertexA.uv0 = (vertexA.uv0 * transform.Scale) + transform.Translate;
|
|
|
|
|
transformedVertices.Add(vertexA);
|
|
|
|
|
}
|
|
|
|
|
if (!transformedVertices.Contains(vertexB)) {
|
|
|
|
|
vertexB.uv0 = (vertexB.uv0 * transform.Scale) + transform.Translate;
|
|
|
|
|
transformedVertices.Add(vertexB);
|
|
|
|
|
}
|
|
|
|
|
if (!transformedVertices.Contains(vertexC)) {
|
|
|
|
|
vertexC.uv0 = (vertexC.uv0 * transform.Scale) + transform.Translate;
|
|
|
|
|
transformedVertices.Add(vertexC);
|
|
|
|
|
}
|
2020-01-18 23:30:19 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2019-11-16 01:27:03 +01:00
|
|
|
|
// collect sources
|
|
|
|
|
List<float> Position = new List<float>();
|
|
|
|
|
List<float> Normal = new List<float>();
|
|
|
|
|
List<float> UV0 = new List<float>();
|
|
|
|
|
List<float> UV1 = new List<float>();
|
|
|
|
|
List<float> UV2 = new List<float>();
|
|
|
|
|
List<float> UV3 = new List<float>();
|
|
|
|
|
List<float> Color = new List<float>();
|
2019-12-14 19:28:10 +01:00
|
|
|
|
List<float> Color2 = new List<float>();
|
2019-11-16 01:27:03 +01:00
|
|
|
|
List<int[]> BoneIndices = new List<int[]>();
|
|
|
|
|
List<float[]> BoneWeights = new List<float[]>();
|
|
|
|
|
|
|
|
|
|
bool HasNormals = false;
|
|
|
|
|
bool HasColors = false;
|
2019-12-14 19:28:10 +01:00
|
|
|
|
bool HasColors2 = false;
|
2019-11-16 01:27:03 +01:00
|
|
|
|
bool HasUV0 = false;
|
|
|
|
|
bool HasUV1 = false;
|
|
|
|
|
bool HasUV2 = false;
|
|
|
|
|
bool HasBoneIds = false;
|
|
|
|
|
|
|
|
|
|
foreach (var vertex in mesh.vertices)
|
|
|
|
|
{
|
|
|
|
|
if (vertex.nrm != Vector3.Zero) HasNormals = true;
|
2019-11-23 23:04:58 +01:00
|
|
|
|
if (vertex.col != Vector4.One && settings.UseVertexColors) HasColors = true;
|
2019-12-14 19:28:10 +01:00
|
|
|
|
if (vertex.col2 != Vector4.One && settings.UseVertexColors) HasColors2 = true;
|
2019-11-16 01:27:03 +01:00
|
|
|
|
if (vertex.uv0 != Vector2.Zero) HasUV0 = true;
|
|
|
|
|
if (vertex.uv1 != Vector2.Zero) HasUV1 = true;
|
|
|
|
|
if (vertex.uv2 != Vector2.Zero) HasUV2 = true;
|
|
|
|
|
if (vertex.boneIds.Count > 0) HasBoneIds = true;
|
|
|
|
|
|
|
|
|
|
Position.Add(vertex.pos.X); Position.Add(vertex.pos.Y); Position.Add(vertex.pos.Z);
|
|
|
|
|
Normal.Add(vertex.nrm.X); Normal.Add(vertex.nrm.Y); Normal.Add(vertex.nrm.Z);
|
|
|
|
|
|
|
|
|
|
if (settings.FlipTexCoordsVertical)
|
|
|
|
|
{
|
|
|
|
|
UV0.Add(vertex.uv0.X); UV0.Add(1 - vertex.uv0.Y);
|
|
|
|
|
UV1.Add(vertex.uv1.X); UV1.Add(1 - vertex.uv1.Y);
|
|
|
|
|
UV2.Add(vertex.uv2.X); UV2.Add(1 - vertex.uv2.Y);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
UV0.Add(vertex.uv0.X); UV0.Add(vertex.uv0.Y);
|
|
|
|
|
UV1.Add(vertex.uv1.X); UV1.Add(vertex.uv1.Y);
|
|
|
|
|
UV2.Add(vertex.uv2.X); UV2.Add(vertex.uv2.Y);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Color.AddRange(new float[] { vertex.col.X, vertex.col.Y, vertex.col.Z, vertex.col.W });
|
2019-12-14 19:28:10 +01:00
|
|
|
|
Color2.AddRange(new float[] { vertex.col2.X, vertex.col2.Y, vertex.col2.Z, vertex.col2.W });
|
2019-11-16 01:27:03 +01:00
|
|
|
|
|
|
|
|
|
List<int> bIndices = new List<int>();
|
|
|
|
|
List<float> bWeights = new List<float>();
|
|
|
|
|
for (int b = 0; b < vertex.boneIds.Count; b++)
|
|
|
|
|
{
|
2019-11-16 19:03:24 +01:00
|
|
|
|
if (b > mesh.VertexSkinCount - 1)
|
|
|
|
|
continue;
|
|
|
|
|
|
|
|
|
|
if (vertex.boneWeights.Count > b)
|
|
|
|
|
{
|
2019-11-16 01:27:03 +01:00
|
|
|
|
if (vertex.boneWeights[b] == 0)
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
|
2019-11-16 22:27:57 +01:00
|
|
|
|
int index = -1;
|
2019-11-16 01:27:03 +01:00
|
|
|
|
if (IndexTable != null)
|
2019-11-16 22:27:57 +01:00
|
|
|
|
index = (int)IndexTable[vertex.boneIds[b]];
|
2019-11-16 01:27:03 +01:00
|
|
|
|
else
|
2019-11-16 22:27:57 +01:00
|
|
|
|
index = (int)vertex.boneIds[b];
|
|
|
|
|
|
|
|
|
|
if (index != -1 && index < skeleton?.bones.Count)
|
|
|
|
|
bIndices.Add(index);
|
2019-11-16 01:27:03 +01:00
|
|
|
|
|
|
|
|
|
if (vertex.boneWeights.Count > b)
|
|
|
|
|
bWeights.Add(vertex.boneWeights[b]);
|
|
|
|
|
else
|
2021-11-19 22:47:41 +01:00
|
|
|
|
bWeights.Add(0);
|
2019-11-16 01:27:03 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (bIndices.Count == 0 && mesh.BoneIndex != -1) {
|
|
|
|
|
HasBoneIds = true;
|
|
|
|
|
bIndices.Add(mesh.BoneIndex);
|
|
|
|
|
bWeights.Add(1);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
BoneIndices.Add(bIndices.ToArray());
|
|
|
|
|
BoneWeights.Add(bWeights.ToArray());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
List<TriangleList> triangleLists = new List<TriangleList>();
|
|
|
|
|
if (mesh.lodMeshes.Count > 0)
|
|
|
|
|
{
|
|
|
|
|
TriangleList triangleList = new TriangleList();
|
|
|
|
|
triangleLists.Add(triangleList);
|
|
|
|
|
|
|
|
|
|
var lodMesh = mesh.lodMeshes[mesh.DisplayLODIndex];
|
2019-11-16 20:08:27 +01:00
|
|
|
|
|
|
|
|
|
List<int> faces = new List<int>();
|
2019-11-16 22:41:16 +01:00
|
|
|
|
if (lodMesh.PrimativeType == STPrimitiveType.TrangleStrips)
|
2019-11-16 20:08:27 +01:00
|
|
|
|
faces = STGenericObject.ConvertTriangleStripsToTriangles(lodMesh.faces);
|
|
|
|
|
else
|
|
|
|
|
faces = lodMesh.faces;
|
|
|
|
|
|
|
|
|
|
for (int i = 0; i < faces.Count; i++)
|
|
|
|
|
triangleList.Indices.Add((uint)faces[i]);
|
2019-11-16 01:27:03 +01:00
|
|
|
|
}
|
|
|
|
|
if (mesh.PolygonGroups.Count > 0)
|
|
|
|
|
{
|
|
|
|
|
foreach (var group in mesh.PolygonGroups)
|
|
|
|
|
{
|
|
|
|
|
TriangleList triangleList = new TriangleList();
|
|
|
|
|
|
|
|
|
|
triangleLists.Add(triangleList);
|
|
|
|
|
|
2020-05-30 02:23:48 +02:00
|
|
|
|
STGenericMaterial material = new STGenericMaterial();
|
|
|
|
|
|
2019-11-16 22:54:15 +01:00
|
|
|
|
if (group.MaterialIndex != -1 && Materials.Count > group.MaterialIndex)
|
2020-05-30 02:23:48 +02:00
|
|
|
|
material = Materials[group.MaterialIndex];
|
2019-11-16 01:27:03 +01:00
|
|
|
|
|
2020-02-08 16:52:07 +01:00
|
|
|
|
if (group.Material != null)
|
2020-05-30 02:23:48 +02:00
|
|
|
|
material = group.Material;
|
|
|
|
|
|
|
|
|
|
if (MaterialRemapper.Values.Any(x => x == material))
|
|
|
|
|
{
|
|
|
|
|
var key = MaterialRemapper.FirstOrDefault(x => x.Value == material).Key;
|
|
|
|
|
triangleList.Material = key;
|
|
|
|
|
}
|
|
|
|
|
else if (material.Text != string.Empty)
|
|
|
|
|
triangleList.Material = material.Text;
|
2020-02-08 16:52:07 +01:00
|
|
|
|
|
2019-11-16 20:08:27 +01:00
|
|
|
|
List<int> faces = new List<int>();
|
2019-11-16 22:41:16 +01:00
|
|
|
|
if (group.PrimativeType == STPrimitiveType.TrangleStrips)
|
2019-11-16 20:08:27 +01:00
|
|
|
|
faces = STGenericObject.ConvertTriangleStripsToTriangles(group.faces);
|
|
|
|
|
else
|
|
|
|
|
faces = group.faces;
|
|
|
|
|
|
|
|
|
|
for (int i = 0; i < faces.Count; i++)
|
|
|
|
|
triangleList.Indices.Add((uint)faces[i]);
|
2019-11-16 01:27:03 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// write sources
|
|
|
|
|
writer.WriteGeometrySource(mesh.Text, SemanticType.POSITION, Position.ToArray(), triangleLists.ToArray());
|
|
|
|
|
|
|
|
|
|
if (HasNormals)
|
|
|
|
|
writer.WriteGeometrySource(mesh.Text, SemanticType.NORMAL, Normal.ToArray(), triangleLists.ToArray());
|
|
|
|
|
|
|
|
|
|
if (HasColors)
|
2019-12-14 19:28:10 +01:00
|
|
|
|
writer.WriteGeometrySource(mesh.Text, SemanticType.COLOR, Color.ToArray(), triangleLists.ToArray(), 0);
|
|
|
|
|
|
|
|
|
|
if (HasColors2)
|
|
|
|
|
writer.WriteGeometrySource(mesh.Text, SemanticType.COLOR, Color2.ToArray(), triangleLists.ToArray(), 1);
|
2019-11-16 01:27:03 +01:00
|
|
|
|
|
|
|
|
|
if (HasUV0)
|
|
|
|
|
writer.WriteGeometrySource(mesh.Text, SemanticType.TEXCOORD, UV0.ToArray(), triangleLists.ToArray(), 0);
|
|
|
|
|
|
|
|
|
|
if (HasUV1)
|
|
|
|
|
writer.WriteGeometrySource(mesh.Text, SemanticType.TEXCOORD, UV1.ToArray(), triangleLists.ToArray(), 1);
|
|
|
|
|
|
|
|
|
|
if (HasUV2)
|
|
|
|
|
writer.WriteGeometrySource(mesh.Text, SemanticType.TEXCOORD, UV2.ToArray(), triangleLists.ToArray(), 2);
|
|
|
|
|
|
|
|
|
|
if (HasBoneIds)
|
|
|
|
|
writer.AttachGeometryController(BoneIndices, BoneWeights);
|
2019-11-09 23:36:52 +01:00
|
|
|
|
|
2019-11-16 01:27:03 +01:00
|
|
|
|
writer.EndGeometryMesh();
|
|
|
|
|
}
|
|
|
|
|
writer.EndGeometrySection();
|
2019-11-09 23:36:52 +01:00
|
|
|
|
}
|
2019-11-16 22:27:57 +01:00
|
|
|
|
|
|
|
|
|
progressBar?.Close();
|
|
|
|
|
|
2019-11-16 22:51:20 +01:00
|
|
|
|
if (!settings.SuppressConfirmDialog)
|
|
|
|
|
System.Windows.Forms.MessageBox.Show($"Exported {FileName} Successfuly!");
|
2019-11-09 23:36:52 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2019-11-12 23:11:35 +01:00
|
|
|
|
|
2019-05-01 21:38:14 +02:00
|
|
|
|
public List<STGenericObject> objects = new List<STGenericObject>();
|
|
|
|
|
public List<STGenericMaterial> materials = new List<STGenericMaterial>();
|
|
|
|
|
public STSkeleton skeleton;
|
|
|
|
|
public List<string> BoneNames = new List<string>();
|
|
|
|
|
|
2019-04-30 23:07:25 +02:00
|
|
|
|
public bool UseTransformMatrix = true;
|
|
|
|
|
|
2019-06-13 23:31:35 +02:00
|
|
|
|
Dictionary<string, Vertex> VertexSkinSources = new Dictionary<string, Vertex>();
|
|
|
|
|
Dictionary<string, Matrix4> MatrixSkinSources = new Dictionary<string, Matrix4>();
|
|
|
|
|
|
2019-09-16 01:13:01 +02:00
|
|
|
|
private Matrix4 GlobalTransform = Matrix4.Identity;
|
2019-06-13 23:31:35 +02:00
|
|
|
|
public bool LoadFile(string FileName)
|
2019-04-30 23:07:25 +02:00
|
|
|
|
{
|
2019-09-16 01:13:01 +02:00
|
|
|
|
GlobalTransform = Matrix4.Identity;
|
|
|
|
|
|
2019-04-30 23:07:25 +02:00
|
|
|
|
COLLADA collada = COLLADA.Load(FileName);
|
2019-10-27 02:28:56 +02:00
|
|
|
|
|
|
|
|
|
|
2019-09-16 01:13:01 +02:00
|
|
|
|
//Check axis up
|
|
|
|
|
if (collada.asset != null)
|
|
|
|
|
{
|
|
|
|
|
switch (collada.asset.up_axis)
|
|
|
|
|
{
|
|
|
|
|
case UpAxisType.X_UP:
|
|
|
|
|
GlobalTransform = Matrix4.CreateRotationX(90);
|
|
|
|
|
break;
|
|
|
|
|
case UpAxisType.Y_UP:
|
|
|
|
|
GlobalTransform = Matrix4.CreateRotationY(90);
|
|
|
|
|
break;
|
|
|
|
|
case UpAxisType.Z_UP:
|
|
|
|
|
GlobalTransform = Matrix4.CreateRotationZ(90);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (collada.asset.unit != null)
|
|
|
|
|
{
|
|
|
|
|
var amount = collada.asset.unit.meter;
|
|
|
|
|
var type = collada.asset.unit.name;
|
|
|
|
|
if (type == "meter")
|
|
|
|
|
{
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
else if (type == "centimeter")
|
|
|
|
|
{
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2019-04-30 23:07:25 +02:00
|
|
|
|
|
|
|
|
|
foreach (var item in collada.Items)
|
|
|
|
|
{
|
2019-12-14 19:28:10 +01:00
|
|
|
|
if (item is library_controllers)
|
|
|
|
|
LoadControllers((library_controllers)item);
|
2019-04-30 23:07:25 +02:00
|
|
|
|
if (item is library_geometries)
|
2019-05-01 21:38:14 +02:00
|
|
|
|
LoadGeometry((library_geometries)item);
|
2019-09-16 01:13:01 +02:00
|
|
|
|
if (item is library_images)
|
|
|
|
|
LoadImages((library_images)item);
|
|
|
|
|
if (item is library_controllers)
|
|
|
|
|
LoadControllers((library_controllers)item);
|
|
|
|
|
if (item is library_nodes)
|
|
|
|
|
LoadNodes((library_nodes)item);
|
|
|
|
|
if (item is library_visual_scenes)
|
|
|
|
|
LoadVisualScenes((library_visual_scenes)item);
|
2019-04-30 23:07:25 +02:00
|
|
|
|
}
|
2019-06-13 23:31:35 +02:00
|
|
|
|
|
|
|
|
|
return true;
|
2019-04-30 23:07:25 +02:00
|
|
|
|
}
|
|
|
|
|
|
2019-12-14 19:28:10 +01:00
|
|
|
|
private void LoadControllers(library_controllers controllers)
|
2019-09-16 01:13:01 +02:00
|
|
|
|
{
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
2019-12-14 19:28:10 +01:00
|
|
|
|
private void LoadVisualScenes(library_visual_scenes nodes)
|
2019-09-16 01:13:01 +02:00
|
|
|
|
{
|
2019-12-14 19:28:10 +01:00
|
|
|
|
|
2019-09-16 01:13:01 +02:00
|
|
|
|
}
|
|
|
|
|
|
2019-12-14 19:28:10 +01:00
|
|
|
|
private void LoadNodes(library_nodes nodes)
|
2019-05-01 21:38:14 +02:00
|
|
|
|
{
|
2019-12-14 19:28:10 +01:00
|
|
|
|
|
2019-05-01 21:38:14 +02:00
|
|
|
|
}
|
|
|
|
|
|
2019-09-16 01:13:01 +02:00
|
|
|
|
private void LoadMaterials(library_materials materials)
|
|
|
|
|
{
|
|
|
|
|
|
|
|
|
|
}
|
2019-06-13 23:31:35 +02:00
|
|
|
|
|
2019-09-16 01:13:01 +02:00
|
|
|
|
private void LoadImages(library_images images)
|
2019-04-30 23:07:25 +02:00
|
|
|
|
{
|
2019-09-16 01:13:01 +02:00
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void LoadGeometry(library_geometries geometries)
|
|
|
|
|
{
|
|
|
|
|
foreach (var geom in geometries.geometry)
|
2019-05-01 21:38:14 +02:00
|
|
|
|
{
|
|
|
|
|
var mesh = geom.Item as mesh;
|
|
|
|
|
if (mesh == null)
|
|
|
|
|
continue;
|
|
|
|
|
|
|
|
|
|
foreach (var source in mesh.source)
|
|
|
|
|
{
|
|
|
|
|
var float_array = source.Item as float_array;
|
|
|
|
|
if (float_array == null)
|
|
|
|
|
continue;
|
2019-04-30 23:07:25 +02:00
|
|
|
|
|
2019-05-01 21:38:14 +02:00
|
|
|
|
Console.Write("Geometry {0} source {1} : ", geom.id, source.id);
|
|
|
|
|
foreach (var mesh_source_value in float_array.Values)
|
|
|
|
|
Console.Write("{0} ", mesh_source_value);
|
|
|
|
|
Console.WriteLine();
|
|
|
|
|
}
|
|
|
|
|
}
|
2019-04-30 23:07:25 +02:00
|
|
|
|
}
|
2019-06-13 23:31:35 +02:00
|
|
|
|
|
2019-09-16 01:13:01 +02:00
|
|
|
|
public bool ExportFile(string FileName, List<STGenericObject> meshes, STSkeleton skeleton = null)
|
|
|
|
|
{
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
2019-06-13 23:31:35 +02:00
|
|
|
|
private List<STGenericObject> CreateGenericObjects(string Name, library_geometries Geometries)
|
|
|
|
|
{
|
|
|
|
|
List<STGenericObject> objects = new List<STGenericObject>();
|
|
|
|
|
foreach (var geom in Geometries.geometry)
|
|
|
|
|
{
|
|
|
|
|
var daeMesh = geom.Item as mesh;
|
|
|
|
|
if (daeMesh == null)
|
|
|
|
|
continue;
|
|
|
|
|
|
|
|
|
|
STGenericObject mesh = new STGenericObject();
|
|
|
|
|
mesh.ObjectName = Name;
|
|
|
|
|
|
|
|
|
|
foreach (var source in daeMesh.source)
|
|
|
|
|
{
|
|
|
|
|
var float_array = source.Item as float_array;
|
|
|
|
|
if (float_array == null)
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
objects.Add(mesh);
|
|
|
|
|
}
|
|
|
|
|
return objects;
|
|
|
|
|
}
|
2019-04-30 23:07:25 +02:00
|
|
|
|
}
|
|
|
|
|
}
|