行进立方体未创建预期的网格
Marching cubes not creating expected meshes
我一直在尝试使用 c# 实现统一的行进立方体,但是正在创建的网格都旋转了 90 度。
在此示例中,我尝试创建一个多维数据集。
这是我目前的代码,如果有人能找出问题所在,那将非常感谢。
using System.Collections;
using System.Collections.Generic;
using UnityEngine;
[RequireComponent(typeof(MeshFilter))]
public class train : MonoBehaviour
{
//setup e.g varables and lists
//----------------------------------------------------------------------------------------------------
//size of the chunk
public int size = 10;
//binary stuff
public List<int> binary = new List<int>();
int bitewisenum = 0;
//posithions
public static Vector3 xyz = new Vector3();
public Vector3 list_edges = new Vector3();
//how much air the cut off point bettween air and solid 0.45 - 0.55 works best
public float dencity = 0.4f;
//mesh
public Vector3[] vertices;
public List<Vector3> vertices_list = new List<Vector3>();
public List<Vector3> vertices_list_store = new List<Vector3>();
public List<Vector3> inverlist = new List<Vector3>();
//we can convert it back to an array later
public List<int> tri = new List<int>();
Mesh mesh;
private int[] num;
private float airsolid_test;
public float cutoff; // 11 works best
//perlin noise
//---------------------------------------------------------------------------------------------------
public static float Perlin3D(float x, float y, float z, float cut)
{
//float AB = Mathf.PerlinNoise(x, y);
//float BC = Mathf.PerlinNoise(y, z);
//float AC = Mathf.PerlinNoise(x, z);
//
//
//float BA = Mathf.PerlinNoise(y, x);
//float CB = Mathf.PerlinNoise(z, y);
//float CA = Mathf.PerlinNoise(z, x);
//
////get the avarge
//float ABC = AB + BC + AC + BA + CB + CA;
//float per = ABC / 6f;
//if (y > cut)
//{
// per = Mathf.Pow(per * (y / 8.5f), 2.718f) + 0.3f;
//}
//testing
float per;
if (x >= cut)
{
per = 0f;
} else if (y >= cut)
{
per = 0f;
}
else if (z <= 0)
{
per = 0f;
}
else if (x <= 0)
{
per = 0f;
}
else if (z >= cut)
{
per = 0f;
}
else if (y <= 0)
{
per = 0f;
}else
{
per = 10f;
}
//testing ends
return per;
}
//----------------------------------------------------------------------------------------
private void Start()
{
//set up binary
binary.Add(1);
binary.Add(2);
binary.Add(4);
binary.Add(8);
binary.Add(16);
binary.Add(32);
binary.Add(64);
binary.Add(128);
//setup mesh
mesh = new Mesh();
GetComponent<MeshFilter>().mesh = mesh;
tri.Clear();
//testing --------------------------------------
airsolid_test = dencity;
//0.1 is because of the perlin noise
for (float x = 0.1f; x < 16; x++)
{
for (int y = 0; y < 15; y++)
{
for (int z = 0; z < 15; z++)
{
//airsolid_test = dencity;
//if (y >= cutoff)
//{//mountans
// airsolid_test += 0.0f;//0.2f
//}
//else
//{//caves
// airsolid_test -= 0.005f;//0.005
//}
xyz.x = x;
xyz.y = y;
xyz.z = z;
createshape(x, y, z);
upmesh();
}
}
}
}
//xyz.x = x;
// xyz.y = y;
// xyz.z = z;
// createshape(x, y, z);
//upmesh();
//upmesh stands for update mesh
void upmesh()
{
mesh.Clear();
mesh.vertices = vertices;
mesh.triangles = tri.ToArray();
mesh.RecalculateNormals();
}
//-------------------------------------------------------------------------------------
// this is the main function and you give it an xyz cordanit and it sets the mesh's triangles and veticeys
void createshape(float x, float y, float z)
{
//list for the binary
List<int> bite = new List<int>();
bite.Clear();
bitewisenum = 0;
//this is getting the binay representation for each point on the cube if the point is > the density value store it as a 1 else as a 0
// so we end up with a binary number like 10011101
//position 7
if (Perlin3D(x + 1, y, z + 1, cutoff) > airsolid_test)
{
bite.Add(1);
//these store the value for intertopilation later but im not using it yet
float Pose_7 = Perlin3D(x + 1, y, z + 1,cutoff);
}
else
{
bite.Add(0);
}
//position 6
if (Perlin3D(x + 1, y + 1, z + 1, cutoff) > airsolid_test)
{
bite.Add(1);
float Pose_6 = Perlin3D(x + 1, y + 1, z + 1, cutoff);
}
else
{
bite.Add(0);
}
//position 5
if (Perlin3D(x, y + 1, z + 1, cutoff) > airsolid_test)
{
bite.Add(1);
float Pose_5 = Perlin3D(x, y + 1, z + 1, cutoff);
}
else
{
bite.Add(0);
}
//position 4
if (Perlin3D(x, y, z + 1, cutoff) > airsolid_test)
{
bite.Add(1);
float Pose_4 = Perlin3D(x, y, z + 1, cutoff);
}
else
{
bite.Add(0);
}
//position 3
if (Perlin3D(x + 1, y, z, cutoff) > airsolid_test)
{
bite.Add(1);
float Pose_3 = Perlin3D(x + 1, y, z, cutoff);
}
else
{
bite.Add(0);
}
//position 2
if (Perlin3D(x + 1, y + 1, z, cutoff) > airsolid_test)
{
bite.Add(1);
float Pose_2 = Perlin3D(x + 1, y + 1, z, cutoff);
}
else
{
bite.Add(0);
}
//position 1
if (Perlin3D(x, y + 1, z, cutoff) > airsolid_test)
{
bite.Add(1);
float Pose_1 = Perlin3D(x, y + 1, z, cutoff);
}
else
{
bite.Add(0);
}
//position 0
if (Perlin3D(x, y, z, cutoff) > airsolid_test)
{
bite.Add(1);
float Pose_0 = Perlin3D(x, y, z, cutoff);
}
else
{
bite.Add(0);
}
//convert binary in to decimal because i dont know an easy way to do this but
//loop throught all items in the binary list and change them
for (int i = 0; i <= 7; i++)
{
if (bite[i] == 1)
{
bitewisenum += binary[i];
}
}
//print(bitewisenum);
//check if its not all air or inside an object
//this means that we check if its 0 or 255
//now we need to make the mesh and show it
if (bitewisenum != 0 && bitewisenum != 255)
{
//Debug.Log("made a mesh");
//store the int array for easyer use
int[] trian = table.TriangleTable[bitewisenum];
/*
* at this point the int array describes edges not postions
* and we need to change that so what we do is loop through ever item in the list and
* parce it through a function(table.transform_to_postion) this returns a Vector 3
* which we then add to a list
*/
vertices_list.Clear();
for (int num = 0; num < trian.Length; num++)
{
//get the actule postions
//we give the function a edge index as an int which represents a edge index and it gives you the actule postion back as a vector 3
list_edges = transform_to_postion(trian[num]);
//add them to the list
vertices_list.Add(list_edges);
vertices_list_store.Add(list_edges);
//end of the for loop
}
// set vetaceys
vertices = vertices_list_store.ToArray();
//
//setting the triangles based off how many items there are on the verticeys list
if (vertices_list.Count / 3 == 1)
{
for (int i = 1; i < 4; i++)
{
tri.Add(tri.Count);
}
}
else if (vertices_list.Count / 3 == 2)
{
for (int i = 1; i < 7; i++)
{
tri.Add(tri.Count);
}
}
else if (vertices_list.Count / 3 == 3)
{
for (int i = 1; i < 10; i++)
{
tri.Add(tri.Count);
}
}
else if (vertices_list.Count / 3 == 4)
{
for (int i = 1; i < 13; i++)
{
tri.Add(tri.Count);
}
}
else if (vertices_list.Count / 3 == 5)
{
for (int i = 1; i < 16; i++)
{
tri.Add(tri.Count);
}
}
/*
Debug.Log("the legnth of verticys: "+vertices.Length);
Debug.Log("the legnth of tri: "+tri.Length);
Debug.Log("tri legnth " + tri.Length);
end of the if loop
*/
}
//end of the function
}
public static Vector3 transform_to_postion(int number_edge)
{
Vector3 xyz_of_edge = new Vector3();
//check which one it is and then set the xyz
if (number_edge == 0)
{
xyz_of_edge.x = xyz.x + 0.5f;
xyz_of_edge.y = xyz.y;
xyz_of_edge.z = xyz.z + 1f;
}
else if (number_edge == 1)
{
xyz_of_edge.x = xyz.x + 1f;
xyz_of_edge.y = xyz.y;
xyz_of_edge.z = xyz.z + 0.5f;
}
else if (number_edge == 2)
{
xyz_of_edge.x = xyz.x + 0.5f;
xyz_of_edge.y = xyz.y;
xyz_of_edge.z = xyz.z;
}
else if (number_edge == 3)
{
xyz_of_edge.x = xyz.x;
xyz_of_edge.y = xyz.y;
xyz_of_edge.z = xyz.z + 0.5f;
}
else if (number_edge == 4)
{
xyz_of_edge.x = xyz.x + 0.5f;
xyz_of_edge.y = xyz.y + 1f;
xyz_of_edge.z = xyz.z + 1f;
}
else if (number_edge == 5)
{
xyz_of_edge.x = xyz.x + 1f;
xyz_of_edge.y = xyz.y + 1f;
xyz_of_edge.z = xyz.z + 0.5f;
}
else if (number_edge == 6)
{
xyz_of_edge.x = xyz.x + 0.5f;
xyz_of_edge.y = xyz.y + 1f;
xyz_of_edge.z = xyz.z;
}
else if (number_edge == 7)
{
xyz_of_edge.x = xyz.x;
xyz_of_edge.y = xyz.y + 1;
xyz_of_edge.z = xyz.z + 0.5f;
}
else if (number_edge == 8)
{
xyz_of_edge.x = xyz.x;
xyz_of_edge.y = xyz.y + 0.5f;
xyz_of_edge.z = xyz.z + 1f;
}
else if (number_edge == 9)
{
xyz_of_edge.x = xyz.x + 1f;
xyz_of_edge.y = xyz.y + 0.5f;
xyz_of_edge.z = xyz.z + 1f;
}
else if (number_edge == 10)
{
xyz_of_edge.x = xyz.x + 1f;
xyz_of_edge.y = xyz.y + 0.5f;
xyz_of_edge.z = xyz.z;
}
else if (number_edge == 11)
{
xyz_of_edge.x = xyz.x;
xyz_of_edge.y = xyz.y + 0.5f;
xyz_of_edge.z = xyz.z;
}
//Debug.Log("edge index " + number_edge);
//Debug.Log("edges "+xyz_of_edge);
//Debug.Log(xyz_of_edge);
return xyz_of_edge;
}
}
我发现问题出在我标记立方体点的方式上。我错了三角测量 table 这导致了方向错误。
为了解决这个问题,我所做的就是在我的代码中重新定义了实际值的点。
我一直在尝试使用 c# 实现统一的行进立方体,但是正在创建的网格都旋转了 90 度。
在此示例中,我尝试创建一个多维数据集。
这是我目前的代码,如果有人能找出问题所在,那将非常感谢。
using System.Collections;
using System.Collections.Generic;
using UnityEngine;
[RequireComponent(typeof(MeshFilter))]
public class train : MonoBehaviour
{
//setup e.g varables and lists
//----------------------------------------------------------------------------------------------------
//size of the chunk
public int size = 10;
//binary stuff
public List<int> binary = new List<int>();
int bitewisenum = 0;
//posithions
public static Vector3 xyz = new Vector3();
public Vector3 list_edges = new Vector3();
//how much air the cut off point bettween air and solid 0.45 - 0.55 works best
public float dencity = 0.4f;
//mesh
public Vector3[] vertices;
public List<Vector3> vertices_list = new List<Vector3>();
public List<Vector3> vertices_list_store = new List<Vector3>();
public List<Vector3> inverlist = new List<Vector3>();
//we can convert it back to an array later
public List<int> tri = new List<int>();
Mesh mesh;
private int[] num;
private float airsolid_test;
public float cutoff; // 11 works best
//perlin noise
//---------------------------------------------------------------------------------------------------
public static float Perlin3D(float x, float y, float z, float cut)
{
//float AB = Mathf.PerlinNoise(x, y);
//float BC = Mathf.PerlinNoise(y, z);
//float AC = Mathf.PerlinNoise(x, z);
//
//
//float BA = Mathf.PerlinNoise(y, x);
//float CB = Mathf.PerlinNoise(z, y);
//float CA = Mathf.PerlinNoise(z, x);
//
////get the avarge
//float ABC = AB + BC + AC + BA + CB + CA;
//float per = ABC / 6f;
//if (y > cut)
//{
// per = Mathf.Pow(per * (y / 8.5f), 2.718f) + 0.3f;
//}
//testing
float per;
if (x >= cut)
{
per = 0f;
} else if (y >= cut)
{
per = 0f;
}
else if (z <= 0)
{
per = 0f;
}
else if (x <= 0)
{
per = 0f;
}
else if (z >= cut)
{
per = 0f;
}
else if (y <= 0)
{
per = 0f;
}else
{
per = 10f;
}
//testing ends
return per;
}
//----------------------------------------------------------------------------------------
private void Start()
{
//set up binary
binary.Add(1);
binary.Add(2);
binary.Add(4);
binary.Add(8);
binary.Add(16);
binary.Add(32);
binary.Add(64);
binary.Add(128);
//setup mesh
mesh = new Mesh();
GetComponent<MeshFilter>().mesh = mesh;
tri.Clear();
//testing --------------------------------------
airsolid_test = dencity;
//0.1 is because of the perlin noise
for (float x = 0.1f; x < 16; x++)
{
for (int y = 0; y < 15; y++)
{
for (int z = 0; z < 15; z++)
{
//airsolid_test = dencity;
//if (y >= cutoff)
//{//mountans
// airsolid_test += 0.0f;//0.2f
//}
//else
//{//caves
// airsolid_test -= 0.005f;//0.005
//}
xyz.x = x;
xyz.y = y;
xyz.z = z;
createshape(x, y, z);
upmesh();
}
}
}
}
//xyz.x = x;
// xyz.y = y;
// xyz.z = z;
// createshape(x, y, z);
//upmesh();
//upmesh stands for update mesh
void upmesh()
{
mesh.Clear();
mesh.vertices = vertices;
mesh.triangles = tri.ToArray();
mesh.RecalculateNormals();
}
//-------------------------------------------------------------------------------------
// this is the main function and you give it an xyz cordanit and it sets the mesh's triangles and veticeys
void createshape(float x, float y, float z)
{
//list for the binary
List<int> bite = new List<int>();
bite.Clear();
bitewisenum = 0;
//this is getting the binay representation for each point on the cube if the point is > the density value store it as a 1 else as a 0
// so we end up with a binary number like 10011101
//position 7
if (Perlin3D(x + 1, y, z + 1, cutoff) > airsolid_test)
{
bite.Add(1);
//these store the value for intertopilation later but im not using it yet
float Pose_7 = Perlin3D(x + 1, y, z + 1,cutoff);
}
else
{
bite.Add(0);
}
//position 6
if (Perlin3D(x + 1, y + 1, z + 1, cutoff) > airsolid_test)
{
bite.Add(1);
float Pose_6 = Perlin3D(x + 1, y + 1, z + 1, cutoff);
}
else
{
bite.Add(0);
}
//position 5
if (Perlin3D(x, y + 1, z + 1, cutoff) > airsolid_test)
{
bite.Add(1);
float Pose_5 = Perlin3D(x, y + 1, z + 1, cutoff);
}
else
{
bite.Add(0);
}
//position 4
if (Perlin3D(x, y, z + 1, cutoff) > airsolid_test)
{
bite.Add(1);
float Pose_4 = Perlin3D(x, y, z + 1, cutoff);
}
else
{
bite.Add(0);
}
//position 3
if (Perlin3D(x + 1, y, z, cutoff) > airsolid_test)
{
bite.Add(1);
float Pose_3 = Perlin3D(x + 1, y, z, cutoff);
}
else
{
bite.Add(0);
}
//position 2
if (Perlin3D(x + 1, y + 1, z, cutoff) > airsolid_test)
{
bite.Add(1);
float Pose_2 = Perlin3D(x + 1, y + 1, z, cutoff);
}
else
{
bite.Add(0);
}
//position 1
if (Perlin3D(x, y + 1, z, cutoff) > airsolid_test)
{
bite.Add(1);
float Pose_1 = Perlin3D(x, y + 1, z, cutoff);
}
else
{
bite.Add(0);
}
//position 0
if (Perlin3D(x, y, z, cutoff) > airsolid_test)
{
bite.Add(1);
float Pose_0 = Perlin3D(x, y, z, cutoff);
}
else
{
bite.Add(0);
}
//convert binary in to decimal because i dont know an easy way to do this but
//loop throught all items in the binary list and change them
for (int i = 0; i <= 7; i++)
{
if (bite[i] == 1)
{
bitewisenum += binary[i];
}
}
//print(bitewisenum);
//check if its not all air or inside an object
//this means that we check if its 0 or 255
//now we need to make the mesh and show it
if (bitewisenum != 0 && bitewisenum != 255)
{
//Debug.Log("made a mesh");
//store the int array for easyer use
int[] trian = table.TriangleTable[bitewisenum];
/*
* at this point the int array describes edges not postions
* and we need to change that so what we do is loop through ever item in the list and
* parce it through a function(table.transform_to_postion) this returns a Vector 3
* which we then add to a list
*/
vertices_list.Clear();
for (int num = 0; num < trian.Length; num++)
{
//get the actule postions
//we give the function a edge index as an int which represents a edge index and it gives you the actule postion back as a vector 3
list_edges = transform_to_postion(trian[num]);
//add them to the list
vertices_list.Add(list_edges);
vertices_list_store.Add(list_edges);
//end of the for loop
}
// set vetaceys
vertices = vertices_list_store.ToArray();
//
//setting the triangles based off how many items there are on the verticeys list
if (vertices_list.Count / 3 == 1)
{
for (int i = 1; i < 4; i++)
{
tri.Add(tri.Count);
}
}
else if (vertices_list.Count / 3 == 2)
{
for (int i = 1; i < 7; i++)
{
tri.Add(tri.Count);
}
}
else if (vertices_list.Count / 3 == 3)
{
for (int i = 1; i < 10; i++)
{
tri.Add(tri.Count);
}
}
else if (vertices_list.Count / 3 == 4)
{
for (int i = 1; i < 13; i++)
{
tri.Add(tri.Count);
}
}
else if (vertices_list.Count / 3 == 5)
{
for (int i = 1; i < 16; i++)
{
tri.Add(tri.Count);
}
}
/*
Debug.Log("the legnth of verticys: "+vertices.Length);
Debug.Log("the legnth of tri: "+tri.Length);
Debug.Log("tri legnth " + tri.Length);
end of the if loop
*/
}
//end of the function
}
public static Vector3 transform_to_postion(int number_edge)
{
Vector3 xyz_of_edge = new Vector3();
//check which one it is and then set the xyz
if (number_edge == 0)
{
xyz_of_edge.x = xyz.x + 0.5f;
xyz_of_edge.y = xyz.y;
xyz_of_edge.z = xyz.z + 1f;
}
else if (number_edge == 1)
{
xyz_of_edge.x = xyz.x + 1f;
xyz_of_edge.y = xyz.y;
xyz_of_edge.z = xyz.z + 0.5f;
}
else if (number_edge == 2)
{
xyz_of_edge.x = xyz.x + 0.5f;
xyz_of_edge.y = xyz.y;
xyz_of_edge.z = xyz.z;
}
else if (number_edge == 3)
{
xyz_of_edge.x = xyz.x;
xyz_of_edge.y = xyz.y;
xyz_of_edge.z = xyz.z + 0.5f;
}
else if (number_edge == 4)
{
xyz_of_edge.x = xyz.x + 0.5f;
xyz_of_edge.y = xyz.y + 1f;
xyz_of_edge.z = xyz.z + 1f;
}
else if (number_edge == 5)
{
xyz_of_edge.x = xyz.x + 1f;
xyz_of_edge.y = xyz.y + 1f;
xyz_of_edge.z = xyz.z + 0.5f;
}
else if (number_edge == 6)
{
xyz_of_edge.x = xyz.x + 0.5f;
xyz_of_edge.y = xyz.y + 1f;
xyz_of_edge.z = xyz.z;
}
else if (number_edge == 7)
{
xyz_of_edge.x = xyz.x;
xyz_of_edge.y = xyz.y + 1;
xyz_of_edge.z = xyz.z + 0.5f;
}
else if (number_edge == 8)
{
xyz_of_edge.x = xyz.x;
xyz_of_edge.y = xyz.y + 0.5f;
xyz_of_edge.z = xyz.z + 1f;
}
else if (number_edge == 9)
{
xyz_of_edge.x = xyz.x + 1f;
xyz_of_edge.y = xyz.y + 0.5f;
xyz_of_edge.z = xyz.z + 1f;
}
else if (number_edge == 10)
{
xyz_of_edge.x = xyz.x + 1f;
xyz_of_edge.y = xyz.y + 0.5f;
xyz_of_edge.z = xyz.z;
}
else if (number_edge == 11)
{
xyz_of_edge.x = xyz.x;
xyz_of_edge.y = xyz.y + 0.5f;
xyz_of_edge.z = xyz.z;
}
//Debug.Log("edge index " + number_edge);
//Debug.Log("edges "+xyz_of_edge);
//Debug.Log(xyz_of_edge);
return xyz_of_edge;
}
}
我发现问题出在我标记立方体点的方式上。我错了三角测量 table 这导致了方向错误。 为了解决这个问题,我所做的就是在我的代码中重新定义了实际值的点。