Skip to content

Commit 01ee09c

Browse files
+++
1 parent aa526cb commit 01ee09c

File tree

6 files changed

+199
-2819
lines changed

6 files changed

+199
-2819
lines changed
13.3 KB
Binary file not shown.

Ext/findMeshHoles/demo.m

+46
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,46 @@
1+
%% DEMO
2+
function demo()
3+
% load sample mesh - created from circleMesh.m
4+
load circleMeshSample.mat
5+
6+
xy = p';
7+
faces = t(1:3,:)';
8+
9+
% add 3rd dimension (z)
10+
z = ones(size(xy,1),1);
11+
vertices = [xy,z];
12+
13+
figure; trimesh(faces,vertices(:,1),vertices(:,2),vertices(:,3)); axis equal;
14+
title('Original'), xlabel('x'), ylabel('y'), zlabel('z')
15+
16+
% create hole
17+
holeSize = (max(p(1,:))-min(p(2,1)))/4;
18+
[vertices,faces] = createHole(vertices,faces,holeSize);
19+
figure; trimesh(faces,vertices(:,1),vertices(:,2),vertices(:,3)); axis equal;
20+
title('Add Hole'), xlabel('x'), ylabel('y'), zlabel('z')
21+
22+
% find hole
23+
holeCellArray = findTriMeshHoles(faces,vertices);
24+
25+
% view holes
26+
figure; trimesh(faces,vertices(:,1),vertices(:,2),vertices(:,3));
27+
title('Identify Holes'); xlabel('x');ylabel('y'); hold on; axis equal;
28+
for i = 1:length(holeCellArray)
29+
hole = holeCellArray{i};
30+
line(vertices(hole,1),vertices(hole,2),vertices(hole,3),'Color','r')
31+
end
32+
end
33+
34+
function [vertices,faces] = createHole(vertices,faces,holeSize)
35+
% create hole in circle mesh
36+
37+
dimens = size(vertices,2);
38+
39+
vertices_norm = vertices - repmat(mean(vertices),size(vertices,1),1);
40+
temp3 = vertices_norm(faces,:);
41+
matrix3 = reshape(temp3,size(faces,1),dimens,3);
42+
faceCenters = sum(matrix3,2)./3;
43+
dist2 = sqrt(sum(faceCenters.^2,3));
44+
faces = faces(dist2 >= holeSize,:);
45+
46+
end

Ext/findMeshHoles/findTriMeshHoles.m

+128
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,128 @@
1+
function [holeCellArray,bounding_triangles,holeLengths] = findTriMeshHoles(faces,vertices)
2+
% Finds holes in a triangular mesh
3+
% Note: Does not work if a hole shares more than one vertex with other holes
4+
% Input:
5+
% faces = M x 3
6+
% vertices = N x 3 (optional if you want the hole lengths)
7+
8+
% Output:
9+
% holeCellArray = P x 1 cell array containing a list of holes, which are
10+
% traced in consecutive order (list of scalar indices)
11+
% bounding_triangles = Q x 3 list of faces that contain a bounding edge (does
12+
% not contain triangles that only has a single bounding vertex)
13+
% holeLengths = P x 1 vector containing the perimeter of each hole
14+
15+
if nargout > 2 && nargin < 2
16+
error('Please input vertices if you want the hole lengths')
17+
end
18+
19+
% sort each row
20+
faces = sort(faces,2);
21+
22+
% face error checking (ensure no faces contain an index more than once)
23+
sel = faces(:,1) == faces(:,2) | faces(:,2) == faces(:,3) | faces(:,1) == faces(:,3);
24+
if sum(sel) > 0
25+
warning('Triangle refers to a vertex more than once... Consider fixing your data...')
26+
faces = faces(~sel,:);
27+
end
28+
29+
% get edges from faces
30+
edge1 = faces(:,1:2);
31+
edge2 = faces(:,[1,3]);
32+
edge3 = faces(:,2:3);
33+
edges = vertcat(edge1,edge2,edge3);
34+
35+
% find edges that have no replicates (find boundary edges)
36+
edges = sortrows(edges);
37+
vec = diff(edges); % diff btw rows
38+
vec = any(vec,2); % unique rows
39+
vec1 = [1;vec]; % doesn't match with previous row
40+
vec2 = [vec;1]; % doesn't match with next row
41+
vec = vec1&vec2; % doesn't match previous and next row
42+
bounding_edges = edges(vec,:); % unique edges
43+
44+
% bounding triangles - triangles that contain a bounding edge
45+
% Note: you can modify to get triangles that contain a bounding vertex
46+
match1 = ismember(edge1,bounding_edges,'rows');
47+
match2 = ismember(edge2,bounding_edges,'rows');
48+
match3 = ismember(edge3,bounding_edges,'rows');
49+
bounding_triangles = faces(match1|match2|match3,:);
50+
51+
if nargin > 1 && nargout > 1
52+
% length of each edge
53+
edgeVertex1 = vertices(bounding_edges(:,1),:);
54+
edgeVertex2 = vertices(bounding_edges(:,2),:);
55+
edge_lengths = sqrt(sum((edgeVertex1-edgeVertex2).^2,2));
56+
[holeCellArray,holeLengths]=getTraces(bounding_edges,edge_lengths);
57+
else
58+
holeCellArray=getTraces(bounding_edges);
59+
end
60+
61+
62+
end
63+
64+
function [tracesDB,lengthDB,completeDB]=getTraces(bounding_edges,edge_lengths)
65+
% Connects the unordered edges to form holes that list the edges in
66+
% consecutive order.
67+
% The number of formed holes from the edges determines the size of tracesDB
68+
69+
% INPUT
70+
% bounding_edges = N x 2 list of connected border vertices
71+
% edge_lengths = N x 1 the corresponding length of the connected vertices
72+
73+
% OUTPUT
74+
% tracesDB = cell array containing a list of holes, which are traced in
75+
% consecutive order
76+
% lengthDB = 1-D vector containing the perimeter of each hole that
77+
% corresponds to tracesDB
78+
% completeDB = 1-D vector that indicates if corresponding hole in tracesDB is completely traced
79+
% (technically should be all 1's) - for testing purposes
80+
81+
if nargin < 2
82+
edge_lengths = zeros(size(bounding_edges,1),1);
83+
end
84+
85+
tracesDB = {};
86+
lengthDB = [];
87+
completeDB = [];
88+
trace12 = [];
89+
idx = 1;
90+
while size(bounding_edges,1) > 0
91+
trace = bounding_edges(1,:)'; % add first trace
92+
bounding_edges = bounding_edges(2:end,:);
93+
total_length = edge_lengths(1); % add first trace length
94+
edge_lengths = edge_lengths(2:end);
95+
flip_flag = 0;
96+
complete_flag = 1;
97+
while trace(1) ~= trace(end)
98+
[row_idx,col_idx] = find(bounding_edges == trace(end));
99+
col_idx = -col_idx+3;
100+
% if more than one bounding edge connects to the end of the
101+
% trace then trace in the other direction (occurs when only one
102+
% triangle separates two holes so both holes share a single vertex)
103+
if length(row_idx) > 1 || length(row_idx) < 1
104+
if flip_flag == 0
105+
trace = flip(trace);
106+
flip_flag = 1;
107+
continue;
108+
else
109+
complete_flag = 0;
110+
% disp('Incomplete hole tracing... May need to make more conditions')
111+
break;
112+
end
113+
end
114+
trace = [trace;bounding_edges(row_idx,col_idx)];
115+
total_length = total_length+edge_lengths(row_idx);
116+
bounding_edges(row_idx,:) = [];
117+
edge_lengths(row_idx) = [];
118+
end
119+
tracesDB = [tracesDB;{trace}];
120+
lengthDB = [lengthDB; total_length];
121+
completeDB = [completeDB;complete_flag];
122+
trace12 = [trace12;trace(1),trace(end)];
123+
idx = idx+1;
124+
end
125+
if sum(~completeDB) > 0
126+
warning('Not all holes are completely traced due to holes sharing more than one vertex with other holes')
127+
end
128+
end

Ext/findMeshHoles/license.txt

+24
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,24 @@
1+
Copyright (c) 2017, Audrey Cheong
2+
All rights reserved.
3+
4+
Redistribution and use in source and binary forms, with or without
5+
modification, are permitted provided that the following conditions are
6+
met:
7+
8+
* Redistributions of source code must retain the above copyright
9+
notice, this list of conditions and the following disclaimer.
10+
* Redistributions in binary form must reproduce the above copyright
11+
notice, this list of conditions and the following disclaimer in
12+
the documentation and/or other materials provided with the distribution
13+
14+
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
15+
AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
16+
IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
17+
ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
18+
LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
19+
CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
20+
SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
21+
INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
22+
CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
23+
ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
24+
POSSIBILITY OF SUCH DAMAGE.

atemplate.m

+1-1
Original file line numberDiff line numberDiff line change
@@ -2352,7 +2352,7 @@
23522352

23532353

23542354
switch hemisphere
2355-
case {'left','L','l'}
2355+
case{'left','L','l'}
23562356
pg.vertices = g.vleft;
23572357
pg.faces = g.fleft;
23582358

0 commit comments

Comments
 (0)