19#include <lagrange/common.h>
20#include <lagrange/legacy/inline.h>
21#include <lagrange/utils/safe_cast.h>
42template <
typename VertexArray,
typename Index>
44 const VertexArray& vertices,
45 const std::vector<Index>& chain,
50 const Index chain_size = safe_cast<Index>(chain.size());
51 auto next = [&chain_size](Index i) {
return (i + 1) % chain_size; };
52 auto prev = [&chain_size](Index i) {
return (i + chain_size - 1) % chain_size; };
53 auto sq_length = [&vertices, &chain](Index vi, Index vj) {
54 return (vertices.row(chain[vi]) - vertices.row(chain[vj])).squaredNorm();
56 auto is_corner = [v0, v1, v2](Index v) {
return v == v0 || v == v1 || v == v2; };
57 std::vector<Eigen::Matrix<Index, 3, 1>> facets;
59 const double NOT_USED = std::numeric_limits<double>::max();
60 Eigen::Matrix<Index, 3, 6, Eigen::RowMajor> candidates;
61 candidates << v0, next(v0), prev(v0), 0, 0, 0, v1, next(v1), prev(v1), 0, 0, 0, v2, next(v2),
64 Eigen::Matrix<double, 3, 2, Eigen::RowMajor> candidate_lengths;
65 candidate_lengths << sq_length(candidates(0, 1), candidates(0, 2)), NOT_USED,
66 sq_length(candidates(1, 1), candidates(1, 2)), NOT_USED,
67 sq_length(candidates(2, 1), candidates(2, 2)), NOT_USED;
69 auto index_comp = [&candidate_lengths](Index i, Index j) {
71 return candidate_lengths.row(i).minCoeff() > candidate_lengths.row(j).minCoeff();
73 std::priority_queue<Index, std::vector<Index>,
decltype(index_comp)> Q(index_comp);
78 Eigen::Matrix<Index, Eigen::Dynamic, 3, Eigen::RowMajor> visited(chain_size, 3);
88 if (candidate_lengths(idx, 0) != NOT_USED &&
89 candidate_lengths(idx, 0) <= candidate_lengths(idx, 1)) {
92 candidate_lengths(idx, 1) != NOT_USED &&
93 candidate_lengths(idx, 1) <= candidate_lengths(idx, 0)) {
99 la_runtime_assert(candidate_lengths.row(idx).minCoeff() <= candidate_lengths.minCoeff());
101 Index base_v = candidates(idx, selection * 3);
102 Index right_v = candidates(idx, selection * 3 + 1);
103 Index left_v = candidates(idx, selection * 3 + 2);
110 if (is_corner(base_v) && is_corner(right_v) && is_corner(left_v)) {
111 if (chain_size != 3) {
112 candidate_lengths(idx, selection) = NOT_USED;
118 if (visited.row(base_v).sum() > 1 || visited(right_v, idx) > 1 ||
119 visited(left_v, idx) > 1) {
121 candidate_lengths(idx, selection) = NOT_USED;
126 visited(base_v, idx) = 2;
127 visited(right_v, idx) = 1;
128 visited(left_v, idx) = 1;
129 facets.push_back({chain[base_v], chain[right_v], chain[left_v]});
132 if (visited.row(right_v).sum() == 1) {
133 Index right_to_right = next(right_v);
134 candidate_lengths(idx, 0) = sq_length(left_v, right_to_right);
135 candidates(idx, 0) = right_v;
136 candidates(idx, 1) = right_to_right;
137 candidates(idx, 2) = left_v;
139 candidate_lengths(idx, 0) = NOT_USED;
142 if (visited.row(left_v).sum() == 1) {
143 Index left_to_left = prev(left_v);
144 candidate_lengths(idx, 1) = sq_length(right_v, left_to_left);
145 candidates(idx, 3) = left_v;
146 candidates(idx, 4) = right_v;
147 candidates(idx, 5) = left_to_left;
149 candidate_lengths(idx, 1) = NOT_USED;
154 Eigen::Matrix<Eigen::Index, Eigen::Dynamic, 1> visited_sum =
155 (visited.array() > 0).rowwise().count();
156 if ((visited_sum.array() > 1).count() == 3) {
159 for (Index i = 0; i < chain_size; i++) {
160 if (visited_sum[i] > 1) {
167 facets.push_back({f[0], f[1], f[2]});
#define la_runtime_assert(...)
Runtime assertion check.
Definition: assert.h:169
Main namespace for Lagrange.
Definition: AABBIGL.h:30
void split_triangle(size_t num_points, span< const Scalar > points, span< const Index > chain, span< Index > visited_buffer, std::vector< Index > &queue_buffer, const Index v0, const Index v1, const Index v2, span< Index > triangulation)
Split a triangle into smaller triangles based on the chain of splitting points on the edges.
Definition: split_triangle.cpp:27