-
Notifications
You must be signed in to change notification settings - Fork 18
/
Copy pathfindLsTriplets.m
122 lines (101 loc) · 3.77 KB
/
findLsTriplets.m
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
function [ls_groups, mesh] = findLsTriplets(ls_dirs, OMIT_LARGE_TRI, aperture_lim)
%FINDLSTRIPLETS Returns valid loudspeaker triplets for 3D layouts
%
% INPUTS:
%
% ls_dirs: (Nspeakers x 2) matrix of loudspeaker directions in degrees,
% in azimuth elevation convention
% OMIT_LARGE_TRI: {0,1} False to return all triplets, true to keep only
% ones that their larger side has a span less than a limit set in
% aperture_lim. If not set, all valid triplets are returned.
% aperture_lim: span limit in degrees above which respective triplets are
% omitted
%
% OUTPUTS:
%
% ls_groups: (Ntriplets x 3) matrix. Each row has the speaker indices
% corresponding to a valid triplet.
%
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%
% Archontis Politis, 1/11/2015
% archontis.politis@aalto.fi
%
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
if nargin<2 || isempty(OMIT_LARGE_TRI)
OMIT_LARGE_TRI = 0;
end
ls_dirs_rad = ls_dirs*pi/180;
% triangulate with delaunay triangulation on the unit sphere (convex hull)
[mesh.vert(:,1), mesh.vert(:,2), mesh.vert(:,3)] = sph2cart(ls_dirs_rad(:,1), ls_dirs_rad(:,2), 1);
mesh.faces = sphDelaunayTriangulation(ls_dirs_rad);
% discard invalid faces, and omit large triangles if asked
mesh = keepValidTriangles(mesh);
% omit large triangles, if asked
if OMIT_LARGE_TRI
mesh = omitLargeTriangles(mesh, aperture_lim);
end
ls_groups = mesh.faces;
end
function mesh = keepValidTriangles(mesh)
%KEEPVALIDFACES Omit triplets that their normals and the centroid to the
% triplets have angle larger than pi/2
valid_faces = [];
for nf=1:size(mesh.faces,1)
temp = mesh.faces(nf,:);
vec = mesh.vert(temp,:);
cvec = cross(vec(2,:)-vec(1,:), vec(3,:)-vec(2,:));
centroid = mean(vec);
if acos(dot(centroid, cvec))<pi/2, valid_faces = [valid_faces, nf]; end
end
mesh.faces = mesh.faces(valid_faces,:);
end
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
function mesh = omitLargeTriangles(mesh, aperture_lim)
%OMITLARGETRIANGLES Omit triplets that any of their sides have span larger
% than aperture_lim
valid_faces = [];
for nf=1:size(mesh.faces,1)
temp = mesh.faces(nf,:);
vec = mesh.vert(temp,:);
a = acos(dot(vec(1,:), vec(2,:)));
b = acos(dot(vec(2,:), vec(3,:)));
c = acos(dot(vec(3,:), vec(1,:)));
abc = [a b c];
if all(abc<=aperture_lim*pi/180), valid_faces = [valid_faces; nf]; end
end
mesh.faces = mesh.faces(valid_faces,:);
end
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
function faces = sphDelaunayTriangulation(dirs_rad)
%DELAUNAYTRIANGULATION Computes the Delaunay triangulation on the unit sphere
% Convert to cartesian
N_vert = size(dirs_rad, 1);
[tempx, tempy, tempz] = sph2cart(dirs_rad(:,1), dirs_rad(:,2), ones(N_vert,1));
U_vert = [tempx, tempy, tempz];
% Find the convex hull of the points on the sphere - in this special case
% the result equals the Delaunay triangulation of the points
faces = convhulln(U_vert);
% Invert the triangles
faces = faces(:, 3:-1:1);
% Shift the results to begin each triangle from the smallest entry
for n = 1:size(faces,1)
tempface = faces(n,:);
[~, minIdx] = min(tempface);
faces(n, :) = circshift(tempface, [0 1-minIdx]);
end
% Sort through triangles with smaller entries first
faces = sortrows(faces, 1); % sort through first entry
maxentry = max(faces(:,1)); % sort through second entry
n = 1;
while n <= maxentry
startIdx = find(faces(:,1) == n, 1, 'first');
if ~isempty(startIdx)
endIdx = find(faces(:,1) == n, 1, 'last');
faces(startIdx:endIdx, :) = sortrows(faces(startIdx:endIdx, :), 2);
n = n + 1;
else
n = n + 1;
end
end
end