-
Notifications
You must be signed in to change notification settings - Fork 69
/
Copy pathMRMeshComponents.cs
156 lines (133 loc) · 6.42 KB
/
MRMeshComponents.cs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
using System;
using System.Collections.Generic;
using System.Runtime.InteropServices;
namespace MR
{
public partial class DotNet
{
public class MeshComponents
{
public struct MeshRegions
{
public FaceBitSet faces;
public int numRegions = 0;
public MeshRegions(FaceBitSet faces, int numRegions)
{
this.faces = faces;
this.numRegions = numRegions;
}
};
public enum FaceIncidence
{
PerEdge, /// face can have neighbor only via edge
PerVertex /// face can have neighbor via vertex
};
public class MeshComponentsMap : List<RegionId>, IDisposable
{
internal MRMeshComponentsMap mrMap_;
private bool disposed_ = false;
public int NumComponents = 0;
[DllImport("MRMeshC", CharSet = CharSet.Auto)]
unsafe private static extern void mrMeshComponentsAllComponentsMapFree(MRMeshComponentsMap* map);
unsafe internal MeshComponentsMap(MRMeshComponentsMap mrMap)
: base()
{
mrMap_ = mrMap;
NumComponents = mrMap.numComponents;
for (int i = 0; i < (int)mrMap.faceMap->size; i++)
{
Add(new RegionId(Marshal.ReadInt32(IntPtr.Add(mrMap.faceMap->data, i * sizeof(int)))));
}
}
public void Dispose()
{
Dispose(true);
GC.SuppressFinalize(this);
}
unsafe protected virtual void Dispose(bool disposing)
{
if (!disposed_)
{
if (mrMap_.faceMap->data != IntPtr.Zero)
fixed (MRMeshComponentsMap* p = &mrMap_) mrMeshComponentsAllComponentsMapFree(p);
disposed_ = true;
}
}
~MeshComponentsMap()
{
Dispose(false);
}
}
[StructLayout(LayoutKind.Sequential)]
internal struct MRFace2RegionMap
{
public IntPtr data = IntPtr.Zero;
public ulong size = 0;
public IntPtr reserved = IntPtr.Zero;
public MRFace2RegionMap() { }
}
[StructLayout(LayoutKind.Sequential)]
unsafe internal struct MRMeshComponentsMap
{
public MRFace2RegionMap* faceMap = null;
public int numComponents = 0;
public MRMeshComponentsMap() { }
};
[StructLayout(LayoutKind.Sequential)]
internal struct MRMeshRegions
{
public IntPtr faces = IntPtr.Zero;
public int numRegions = 0;
public MRMeshRegions() { }
};
[DllImport("MRMeshC", CharSet = CharSet.Auto)]
private static extern IntPtr mrMeshComponentsGetComponent(ref MRMeshPart mp, FaceId id, FaceIncidence incidence, IntPtr cb);
[DllImport("MRMeshC", CharSet = CharSet.Auto)]
unsafe private static extern IntPtr mrMeshComponentsGetLargestComponent(ref MRMeshPart mp, FaceIncidence incidence, IntPtr cb, float minArea, int* numSmallerComponents);
[DllImport("MRMeshC", CharSet = CharSet.Auto)]
private static extern IntPtr mrMeshComponentsGetLargeByAreaComponents(ref MRMeshPart mp, float minArea, IntPtr cb);
[DllImport("MRMeshC", CharSet = CharSet.Auto)]
private static extern MRMeshComponentsMap mrMeshComponentsGetAllComponentsMap(ref MRMeshPart mp, FaceIncidence incidence);
[DllImport("MRMeshC", CharSet = CharSet.Auto)]
unsafe private static extern MRMeshRegions mrMeshComponentsGetLargeByAreaRegions(ref MRMeshPart mp, MRFace2RegionMap* face2RegionMap, int numRegions, float minArea);
/// gets all connected components of mesh part as
/// 1. the mapping: FaceId -> Component ID in [0, 1, 2, ...)
/// 2. the total number of components
unsafe static public MeshComponentsMap GetAllComponentsMap(MeshPart mp, FaceIncidence incidence)
{
var mrMap = mrMeshComponentsGetAllComponentsMap(ref mp.mrMeshPart, incidence);
var res = new MeshComponentsMap(mrMap);
return res;
}
/// returns
/// 1. the union of all regions with area >= minArea
/// 2. the number of such regions
unsafe static public MeshRegions GetLargeByAreaRegions(MeshPart mp, MeshComponentsMap map, int numRegions, float minArea)
{
var mrRegions = mrMeshComponentsGetLargeByAreaRegions(ref mp.mrMeshPart, map.mrMap_.faceMap, numRegions, minArea);
return new MeshRegions
{
faces = new FaceBitSet(mrRegions.faces),
numRegions = mrRegions.numRegions
};
}
/// returns the union of connected components, each having at least given area
static public FaceBitSet GetLargeByAreaComponents(MeshPart mp, float minArea)
{
var components = mrMeshComponentsGetLargeByAreaComponents(ref mp.mrMeshPart, minArea, IntPtr.Zero);
return new FaceBitSet(components);
}
/// returns the largest by surface area component or empty set if its area is smaller than \param minArea
unsafe static public FaceBitSet GetLargestComponent(MeshPart mp, FaceIncidence incidence, float minArea, out int numSmallerComponents)
{
fixed (int* p = &numSmallerComponents)
return new FaceBitSet(mrMeshComponentsGetLargestComponent(ref mp.mrMeshPart, incidence, IntPtr.Zero, minArea, p));
}
/// not effective to call more than once, if several components are needed use GetAllComponentsMap
static public FaceBitSet GetComponent(MeshPart mp, FaceId id, FaceIncidence incidence)
{
return new FaceBitSet(mrMeshComponentsGetComponent(ref mp.mrMeshPart, id, incidence, IntPtr.Zero));
}
}
}
}