404 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
			
		
		
	
	
			404 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
| /************************************************************************************
 | |
| 
 | |
| Depthkit Unity SDK License v1
 | |
| Copyright 2016-2024 Simile Inc dba Scatter. All Rights reserved.  
 | |
| 
 | |
| Licensed under the the Simile Inc dba Scatter ("Scatter")
 | |
| Software Development Kit License Agreement (the "License"); 
 | |
| you may not use this SDK except in compliance with the License, 
 | |
| which is provided at the time of installation or download, 
 | |
| or which otherwise accompanies this software in either electronic or hard copy form.  
 | |
| 
 | |
| You may obtain a copy of the License at http://www.depthkit.tv/license-agreement-v1
 | |
| 
 | |
| Unless required by applicable law or agreed to in writing, 
 | |
| the SDK distributed under the License is distributed on an "AS IS" BASIS, 
 | |
| WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 
 | |
| See the License for the specific language governing permissions and limitations under the License. 
 | |
| 
 | |
| ************************************************************************************/
 | |
| 
 | |
| using UnityEngine;
 | |
| using UnityEditor;
 | |
| using System.Collections;
 | |
| using System.Collections.Generic;
 | |
| using System.IO;
 | |
| using System;
 | |
| using System.Runtime.InteropServices;
 | |
| 
 | |
| namespace Depthkit
 | |
| {
 | |
|     public class TriangleMesh
 | |
|     {
 | |
|         [HideInInspector]
 | |
|         public Depthkit.MeshSource source;
 | |
| 
 | |
|         [System.NonSerialized]
 | |
|         private Mesh m_mesh;
 | |
| 
 | |
|         [HideInInspector, SerializeField]
 | |
|         private int m_triangleCount = 0;
 | |
| 
 | |
|         public int TriangleCount
 | |
|         {
 | |
|             get { return m_triangleCount; }
 | |
|             set
 | |
|             {
 | |
|                 EnsureTriangleMesh(value);
 | |
|             }
 | |
|         }
 | |
|         public Mesh mesh
 | |
|         {
 | |
|             get
 | |
|             {
 | |
|                 if (m_mesh == null)
 | |
|                 {
 | |
|                     m_mesh = CreateMesh();
 | |
|                 }
 | |
|                 return m_mesh;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         public void EnsureTriangleMesh()
 | |
|         {
 | |
|             if (source != null && source.clip != null && source.clip.metadata != null && m_triangleCount > 0)
 | |
|             {
 | |
|                 if (m_mesh == null)
 | |
|                 {
 | |
|                     m_mesh = CreateMesh();
 | |
|                     Bounds bounds = source.GetLocalBounds();
 | |
|                     m_mesh.bounds = bounds;
 | |
|                     CreateCubeMesh(m_mesh, bounds.center, bounds.size, m_triangleCount);
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         public void EnsureTriangleMesh(Vector2 size)
 | |
|         {
 | |
|             if (source != null && source.clip != null && source.clip.metadata != null && size.x > 0 && size.y > 0)
 | |
|             {
 | |
|                 if (m_mesh == null)
 | |
|                 {
 | |
|                     m_mesh = CreateMesh();
 | |
|                 }
 | |
|                 int count = ((int)size.x - 1) * ((int)size.y - 1) * 2;
 | |
|                 Bounds bounds = source.GetLocalBounds();
 | |
|                 m_mesh.bounds = bounds;
 | |
|                 if (m_mesh.vertexCount / 3 == count && m_mesh.name == "Depthkit Mesh") return; //no need to update
 | |
|                 m_triangleCount = count;
 | |
|                 CreateCubeMesh(m_mesh, bounds.center, bounds.size, m_triangleCount);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         public void EnsureTriangleMesh(int triangles)
 | |
|         {
 | |
|             if (source != null && source.clip != null && source.clip.metadata != null && triangles > 0)
 | |
|             {
 | |
|                 if (m_mesh == null)
 | |
|                 {
 | |
|                     m_mesh = CreateMesh();
 | |
|                 }
 | |
|                 Bounds bounds = source.GetLocalBounds();
 | |
|                 m_mesh.bounds = bounds;
 | |
|                 if (m_mesh.vertexCount / 3 == triangles && m_mesh.name == "Depthkit Mesh") return; //no need to update
 | |
|                 m_triangleCount = triangles;
 | |
|                 CreateCubeMesh(m_mesh, bounds.center, bounds.size, m_triangleCount);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         static private Mesh CreateMesh()
 | |
|         {
 | |
|             Mesh mesh = new Mesh();
 | |
|             mesh.name = "Depthkit Mesh";
 | |
|             mesh.hideFlags = HideFlags.DontSave;
 | |
|             return mesh;
 | |
|         }
 | |
| 
 | |
|         public void ReleaseMesh()
 | |
|         {
 | |
|             if (m_mesh != null)
 | |
|             {
 | |
|                 if (Application.isEditor && !Application.isPlaying)
 | |
|                 {
 | |
|                     UnityEngine.Object.DestroyImmediate(m_mesh);
 | |
|                 }
 | |
|                 else
 | |
|                 {
 | |
|                     UnityEngine.Object.Destroy(m_mesh);
 | |
|                 }
 | |
|                 m_mesh = null;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         static Vector3[] s_cubeVerts = null;
 | |
|         static int[] s_cubeTriangles = null;
 | |
| 
 | |
|         static Vector3[] GetCubeVerts()
 | |
|         {
 | |
|             if (s_cubeVerts == null)
 | |
|             {
 | |
|                 s_cubeVerts = new Vector3[] {
 | |
|                     new Vector3 (-0.5f, -0.5f, -0.5f),
 | |
|                     new Vector3 (0.5f, -0.5f, -0.5f),
 | |
|                     new Vector3 (0.5f, 0.5f, -0.5f),
 | |
|                     new Vector3 (-0.5f, 0.5f, -0.5f),
 | |
|                     new Vector3 (-0.5f, 0.5f, 0.5f),
 | |
|                     new Vector3 (0.5f, 0.5f, 0.5f),
 | |
|                     new Vector3 (0.5f, -0.5f, 0.5f),
 | |
|                     new Vector3 (-0.5f, -0.5f, 0.5f),
 | |
|                 };
 | |
|             }
 | |
|             return s_cubeVerts;
 | |
|         }
 | |
| 
 | |
|         static int[] GetCubeTriangles()
 | |
|         {
 | |
|             if (s_cubeTriangles == null)
 | |
|             {
 | |
|                 s_cubeTriangles = new int[] {
 | |
|                     0, 2, 1, //face front
 | |
|                     0, 3, 2,
 | |
|                     2, 3, 4, //face top
 | |
|                     2, 4, 5,
 | |
|                     1, 2, 5, //face right
 | |
|                     1, 5, 6,
 | |
|                     0, 7, 4, //face left
 | |
|                     0, 4, 3,
 | |
|                     5, 4, 7, //face back
 | |
|                     5, 7, 6,
 | |
|                     0, 6, 7, //face bottom
 | |
|                     0, 1, 6
 | |
|                 };
 | |
|             }
 | |
|             return s_cubeTriangles;
 | |
|         }
 | |
| 
 | |
|         public void ResetMeshCube(Vector3 center, Vector3 size)
 | |
|         {
 | |
|             if (size == Vector3.zero) return;
 | |
| 
 | |
|             Vector3[] positions = mesh.vertices;
 | |
| 
 | |
|             Vector3[] cubeVerts = GetCubeVerts();
 | |
|             int[] cubeTriangles = GetCubeTriangles();
 | |
| 
 | |
|             for (int i = 0; i < 12; ++i)
 | |
|             {
 | |
|                 int index = i * 3;
 | |
|                 Vector3 v1 = cubeVerts[cubeTriangles[index]];
 | |
|                 positions[index] = new Vector3(v1.x * size.x + center.x, v1.y * size.y + center.y, v1.z * size.z + center.z);
 | |
| 
 | |
|                 ++index;
 | |
|                 Vector3 v2 = cubeVerts[cubeTriangles[index]];
 | |
|                 positions[index] = new Vector3(v2.x * size.x + center.x, v2.y * size.y + center.y, v2.z * size.z + center.z);
 | |
| 
 | |
|                 ++index;
 | |
|                 Vector3 v3 = cubeVerts[cubeTriangles[index]];
 | |
|                 positions[index] = new Vector3(v3.x * size.x + center.x, v3.y * size.y + center.y, v3.z * size.z + center.z);
 | |
|             }
 | |
| 
 | |
|             mesh.vertices = positions;
 | |
|         }
 | |
| 
 | |
|         //static factory functions
 | |
|         public static void CreateLattice(Mesh mesh, Vector2 dims, CoordinateRangeType rangeType)
 | |
|         {
 | |
| 
 | |
|             int w = (int)dims.x;
 | |
|             int h = (int)dims.y;
 | |
|             float fw = dims.x;
 | |
|             float fh = dims.y;
 | |
|             int sw = w - 1;
 | |
|             int sh = h - 1;
 | |
| 
 | |
|             Vector3[] positions = new Vector3[w * h];
 | |
|             Vector2[] texCoords = new Vector2[w * h];
 | |
|             int indexCount = sw * sh * 6;
 | |
|             int[] indices = new int[indexCount];
 | |
| 
 | |
|             mesh.Clear();
 | |
|             mesh.indexFormat = UnityEngine.Rendering.IndexFormat.UInt32;
 | |
| 
 | |
|             int a = 0;
 | |
|             int b = 0;
 | |
|             int c = 0;
 | |
|             int tri = 0;
 | |
| 
 | |
|             for (int y = 0; y < sh; y++)
 | |
|             {
 | |
|                 for (int x = 0; x < sw; x++)
 | |
|                 {
 | |
|                     a = x + y * w;
 | |
|                     b = (x + 1) + y * w;
 | |
|                     c = (x + 1) + (y + 1) * w;
 | |
| 
 | |
|                     indices[tri++] = c;
 | |
|                     indices[tri++] = b;
 | |
|                     indices[tri++] = a;
 | |
| 
 | |
|                     a = x + y * w;
 | |
|                     b = (x + 1) + (y + 1) * w;
 | |
|                     c = x + (y + 1) * w;
 | |
| 
 | |
|                     indices[tri++] = c;
 | |
|                     indices[tri++] = b;
 | |
|                     indices[tri++] = a;
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             for (int y = 0; y < h; y++)
 | |
|             {
 | |
|                 for (int x = 0; x < w; x++)
 | |
|                 {
 | |
| 
 | |
|                     int ind = x + y * w;
 | |
|                     switch (rangeType)
 | |
|                     {
 | |
|                         case CoordinateRangeType.NDC:
 | |
|                             positions[ind] = new Vector3(((float)x / fw) * 2.0f - 1.0f, ((float)y / fh) * 2.0f - 1.0f, 0); // NDC space units (-1 .. 1)
 | |
|                             break;
 | |
|                         case CoordinateRangeType.Normalized:
 | |
|                             positions[ind] = new Vector3(((float)x / fw), ((float)y / fh), 0); // normalized space units (0 .. 1)
 | |
|                             break;
 | |
|                         case CoordinateRangeType.Pixels:
 | |
|                             positions[ind] = new Vector3(x, y, 0); // world space units (1 unit / meter per pixel)
 | |
|                             break;
 | |
|                     }
 | |
|                     texCoords[ind] = new Vector2((float)x / fw, (float)y / fh);
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             mesh.vertices = positions;
 | |
|             mesh.uv = texCoords;
 | |
|             mesh.triangles = indices;
 | |
|             mesh.RecalculateNormals();
 | |
|         }
 | |
| 
 | |
|         internal static void addVertex(int triangle, int tri_pos, int vert, int x, int y, float fw, float fh, Vector3[] positions, Vector2[] uv, int[] indices, CoordinateRangeType rangeType)
 | |
|         {
 | |
|             indices[vert] = vert;
 | |
|             switch (rangeType)
 | |
|             {
 | |
|                 case CoordinateRangeType.NDC:
 | |
|                     positions[vert] = new Vector3(((float)x / fw) * 2.0f - 1.0f, ((float)y / fh) * 2.0f - 1.0f, 0); // NDC space units (-1 .. 1)
 | |
|                     break;
 | |
|                 case CoordinateRangeType.Normalized:
 | |
|                     positions[vert] = new Vector3(((float)x / fw), ((float)y / fh), 0); // normalized space units (0 .. 1)
 | |
|                     break;
 | |
|                 case CoordinateRangeType.Pixels:
 | |
|                     positions[vert] = new Vector3(x, y, 0); // world space units (1 unit / meter per pixel)
 | |
|                     break;
 | |
|             }
 | |
|             uv[vert] = new Vector2((float)triangle, (float)tri_pos);
 | |
|         }
 | |
| 
 | |
|         public static void CreateTriangleLattice(Mesh mesh, Vector2 dims, CoordinateRangeType rangeType)
 | |
|         {
 | |
|             int w = (int)dims.x;
 | |
|             int h = (int)dims.y;
 | |
| 
 | |
|             int vertexCount = (w - 1) * (h - 1) * 6;
 | |
|             Vector3[] positions = new Vector3[vertexCount];
 | |
|             Vector2[] uv = new Vector2[vertexCount];
 | |
|             int[] indices = new int[vertexCount];
 | |
| 
 | |
|             mesh.Clear();
 | |
|             mesh.indexFormat = UnityEngine.Rendering.IndexFormat.UInt32;
 | |
| 
 | |
|             int vert = 0;
 | |
|             int tri = 0;
 | |
| 
 | |
|             for (int y = 0; y < h - 1; y++)
 | |
|             {
 | |
|                 for (int x = 0; x < w - 1; x++)
 | |
|                 {
 | |
|                     addVertex(tri, 0, vert, x, y, dims.x, dims.y, positions, uv, indices, rangeType);
 | |
|                     vert++;
 | |
| 
 | |
|                     addVertex(tri, 1, vert, x, y + 1, dims.x, dims.y, positions, uv, indices, rangeType);
 | |
|                     vert++;
 | |
| 
 | |
|                     addVertex(tri, 2, vert, x + 1, y, dims.x, dims.y, positions, uv, indices, rangeType);
 | |
|                     vert++;
 | |
|                     tri++;
 | |
| 
 | |
|                     addVertex(tri, 0, vert, x + 1, y + 1, dims.x, dims.y, positions, uv, indices, rangeType);
 | |
|                     vert++;
 | |
| 
 | |
|                     addVertex(tri, 1, vert, x + 1, y, dims.x, dims.y, positions, uv, indices, rangeType);
 | |
|                     vert++;
 | |
| 
 | |
|                     addVertex(tri, 2, vert, x, y + 1, dims.x, dims.y, positions, uv, indices, rangeType);
 | |
|                     vert++;
 | |
|                     tri++;
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             mesh.vertices = positions;
 | |
|             mesh.uv = uv;
 | |
|             mesh.triangles = indices;
 | |
|         }
 | |
| 
 | |
|         public static void CreateCubeMesh(Mesh mesh, Vector3 center, Vector3 size, int totalTriangles)
 | |
|         {
 | |
|             Vector3[] vertices = new Vector3[totalTriangles * 3];
 | |
|             Vector2[] uvs = new Vector2[totalTriangles * 3];
 | |
|             int[] triangles = new int[totalTriangles * 3];
 | |
| 
 | |
|             Vector3[] cubeVerts = GetCubeVerts();
 | |
|             int[] cubeTriangles = GetCubeTriangles();
 | |
| 
 | |
|             int tri = 0;
 | |
|             int i = 0;
 | |
|             for (; i < 12; ++i)
 | |
|             {
 | |
|                 int index = i * 3;
 | |
|                 Vector3 v1 = cubeVerts[cubeTriangles[index]];
 | |
|                 vertices[index] = new Vector3(v1.x * size.x + center.x, v1.y * size.y + center.y, v1.z * size.z + center.z);
 | |
|                 uvs[index] = new Vector2(tri, 0);
 | |
|                 triangles[index] = index;
 | |
| 
 | |
|                 ++index;
 | |
|                 Vector3 v2 = cubeVerts[cubeTriangles[index]];
 | |
|                 vertices[index] = new Vector3(v2.x * size.x + center.x, v2.y * size.y + center.y, v2.z * size.z + center.z);
 | |
|                 uvs[index] = new Vector2(tri, 1);
 | |
|                 triangles[index] = index;
 | |
| 
 | |
|                 ++index;
 | |
|                 Vector3 v3 = cubeVerts[cubeTriangles[index]];
 | |
|                 vertices[index] = new Vector3(v3.x * size.x + center.x, v3.y * size.y + center.y, v3.z * size.z + center.z);
 | |
|                 uvs[index] = new Vector2(tri, 2);
 | |
|                 triangles[index] = index;
 | |
| 
 | |
|                 ++tri;
 | |
|             }
 | |
| 
 | |
|             for (; i < totalTriangles; ++i)
 | |
|             {
 | |
|                 int index = i * 3;
 | |
|                 vertices[index] = new Vector3(0, 0, 0);
 | |
|                 uvs[index] = new Vector2(tri, 0);
 | |
|                 triangles[index] = index;
 | |
| 
 | |
|                 index++;
 | |
|                 vertices[index] = new Vector3(0, 0, 0);
 | |
|                 uvs[index] = new Vector2(tri, 1);
 | |
|                 triangles[index] = index;
 | |
| 
 | |
|                 index++;
 | |
|                 vertices[index] = new Vector3(0, 0, 0);
 | |
|                 uvs[index] = new Vector2(tri, 2);
 | |
|                 triangles[index] = index;
 | |
| 
 | |
|                 ++tri;
 | |
|             }
 | |
| 
 | |
|             mesh.Clear();
 | |
|             mesh.indexFormat = UnityEngine.Rendering.IndexFormat.UInt32;
 | |
|             mesh.vertices = vertices;
 | |
|             mesh.uv = uvs;
 | |
|             mesh.triangles = triangles;
 | |
|         }
 | |
| 
 | |
|     }
 | |
| } |