closest_facet.cpp 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330
  1. #include "closest_facet.h"
  2. #include <vector>
  3. #include <stdexcept>
  4. #include <CGAL/AABB_tree.h>
  5. #include <CGAL/AABB_traits.h>
  6. #include <CGAL/AABB_triangle_primitive.h>
  7. #include <CGAL/intersections.h>
  8. #include <CGAL/Exact_predicates_exact_constructions_kernel.h>
  9. #include "order_facets_around_edge.h"
  10. template<
  11. typename DerivedV,
  12. typename DerivedF,
  13. typename DerivedI,
  14. typename DerivedP,
  15. typename DerivedR,
  16. typename DerivedS >
  17. IGL_INLINE void igl::cgal::closest_facet(
  18. const Eigen::PlainObjectBase<DerivedV>& V,
  19. const Eigen::PlainObjectBase<DerivedF>& F,
  20. const Eigen::PlainObjectBase<DerivedI>& I,
  21. const Eigen::PlainObjectBase<DerivedP>& P,
  22. Eigen::PlainObjectBase<DerivedR>& R,
  23. Eigen::PlainObjectBase<DerivedS>& S) {
  24. typedef CGAL::Exact_predicates_exact_constructions_kernel Kernel;
  25. typedef Kernel::Point_3 Point_3;
  26. typedef Kernel::Plane_3 Plane_3;
  27. typedef Kernel::Segment_3 Segment_3;
  28. typedef Kernel::Triangle_3 Triangle;
  29. typedef std::vector<Triangle>::iterator Iterator;
  30. typedef CGAL::AABB_triangle_primitive<Kernel, Iterator> Primitive;
  31. typedef CGAL::AABB_traits<Kernel, Primitive> AABB_triangle_traits;
  32. typedef CGAL::AABB_tree<AABB_triangle_traits> Tree;
  33. if (F.rows() <= 0 || I.rows() <= 0) {
  34. throw std::runtime_error(
  35. "Closest facet cannot be computed on empty mesh.");
  36. }
  37. const size_t num_faces = I.rows();
  38. std::vector<Triangle> triangles;
  39. for (size_t i=0; i<num_faces; i++) {
  40. const Eigen::Vector3i f = F.row(I(i, 0));
  41. triangles.emplace_back(
  42. Point_3(V(f[0], 0), V(f[0], 1), V(f[0], 2)),
  43. Point_3(V(f[1], 0), V(f[1], 1), V(f[1], 2)),
  44. Point_3(V(f[2], 0), V(f[2], 1), V(f[2], 2)));
  45. if (triangles.back().is_degenerate()) {
  46. throw std::runtime_error(
  47. "Input facet components contains degenerated triangles");
  48. }
  49. }
  50. Tree tree(triangles.begin(), triangles.end());
  51. tree.accelerate_distance_queries();
  52. auto on_the_positive_side = [&](size_t fid, const Point_3& p) {
  53. const auto& f = F.row(fid).eval();
  54. Point_3 v0(V(f[0], 0), V(f[0], 1), V(f[0], 2));
  55. Point_3 v1(V(f[1], 0), V(f[1], 1), V(f[1], 2));
  56. Point_3 v2(V(f[2], 0), V(f[2], 1), V(f[2], 2));
  57. auto ori = CGAL::orientation(v0, v1, v2, p);
  58. switch (ori) {
  59. case CGAL::POSITIVE:
  60. return true;
  61. case CGAL::NEGATIVE:
  62. return false;
  63. case CGAL::COPLANAR:
  64. throw std::runtime_error(
  65. "It seems input mesh contains self intersection");
  66. default:
  67. throw std::runtime_error("Unknown CGAL state.");
  68. }
  69. return false;
  70. };
  71. auto get_orientation = [&](size_t fid, size_t s, size_t d) -> bool {
  72. const auto& f = F.row(fid);
  73. if (f[0] == s && f[1] == d) return false;
  74. else if (f[1] == s && f[2] == d) return false;
  75. else if (f[2] == s && f[0] == d) return false;
  76. else if (f[0] == d && f[1] == s) return true;
  77. else if (f[1] == d && f[2] == s) return true;
  78. else if (f[2] == d && f[0] == s) return true;
  79. else {
  80. throw std::runtime_error(
  81. "Cannot compute orientation due to incorrect connectivity");
  82. return false;
  83. }
  84. };
  85. auto index_to_signed_index = [&](size_t index, bool ori) -> int{
  86. return (index+1) * (ori? 1:-1);
  87. };
  88. auto signed_index_to_index = [&](int signed_index) -> size_t {
  89. return abs(signed_index) - 1;
  90. };
  91. enum ElementType { VERTEX, EDGE, FACE };
  92. auto determine_element_type = [&](const Point_3& p, const size_t fid,
  93. size_t& element_index) {
  94. const auto& tri = triangles[fid];
  95. const Point_3 p0 = tri[0];
  96. const Point_3 p1 = tri[1];
  97. const Point_3 p2 = tri[2];
  98. if (p == p0) { element_index = 0; return VERTEX; }
  99. if (p == p1) { element_index = 1; return VERTEX; }
  100. if (p == p2) { element_index = 2; return VERTEX; }
  101. if (CGAL::collinear(p0, p1, p)) { element_index = 2; return EDGE; }
  102. if (CGAL::collinear(p1, p2, p)) { element_index = 0; return EDGE; }
  103. if (CGAL::collinear(p2, p0, p)) { element_index = 1; return EDGE; }
  104. element_index = 0;
  105. return FACE;
  106. };
  107. auto process_edge_case = [&](
  108. size_t query_idx,
  109. const size_t s, const size_t d,
  110. bool& orientation) {
  111. Point_3 mid_edge_point(
  112. (V(s,0) + V(d,0)) * 0.5,
  113. (V(s,1) + V(d,1)) * 0.5,
  114. (V(s,2) + V(d,2)) * 0.5);
  115. Point_3 query_point(
  116. P(query_idx, 0),
  117. P(query_idx, 1),
  118. P(query_idx, 2));
  119. std::vector<Tree::Primitive_id> intersected_faces;
  120. tree.all_intersected_primitives(Segment_3(mid_edge_point, query_point),
  121. std::back_inserter(intersected_faces));
  122. const size_t num_intersected_faces = intersected_faces.size();
  123. std::vector<size_t> intersected_face_indices(num_intersected_faces);
  124. std::vector<int> intersected_face_signed_indices(num_intersected_faces);
  125. std::transform(intersected_faces.begin(),
  126. intersected_faces.end(),
  127. intersected_face_indices.begin(),
  128. [&](const Tree::Primitive_id& itr) -> size_t
  129. { return I(itr-triangles.begin(), 0); });
  130. std::transform(
  131. intersected_face_indices.begin(),
  132. intersected_face_indices.end(),
  133. intersected_face_signed_indices.begin(),
  134. [&](size_t index) {
  135. return index_to_signed_index(
  136. index, get_orientation(index, s,d));
  137. });
  138. Eigen::VectorXi order;
  139. DerivedP pivot = P.row(query_idx).eval();
  140. auto get_opposite_vertex = [&](size_t fid) {
  141. const auto& f = F.row(fid);
  142. if (f[0] != s && f[0] != d) return V.row(f[0]).eval();
  143. if (f[1] != s && f[1] != d) return V.row(f[1]).eval();
  144. if (f[2] != s && f[2] != d) return V.row(f[2]).eval();
  145. };
  146. igl::cgal::order_facets_around_edge(V, F, s, d,
  147. intersected_face_signed_indices,
  148. pivot, order);
  149. orientation = intersected_face_signed_indices[order[0]] < 0;
  150. return intersected_face_indices[order[0]];
  151. };
  152. auto process_face_case = [&](
  153. const size_t query_idx, const size_t fid, bool& orientation) {
  154. const auto& f = F.row(I(fid, 0));
  155. return process_edge_case(query_idx, f[0], f[1], orientation);
  156. };
  157. auto process_vertex_case = [&](const size_t query_idx, size_t s,
  158. bool& orientation) {
  159. Point_3 closest_point(V(s, 0), V(s, 1), V(s, 2));
  160. Point_3 query_point(P(query_idx, 0), P(query_idx, 1), P(query_idx, 2));
  161. std::vector<Tree::Primitive_id> intersected_faces;
  162. tree.all_intersected_primitives(Segment_3(closest_point, query_point),
  163. std::back_inserter(intersected_faces));
  164. const size_t num_intersected_faces = intersected_faces.size();
  165. std::vector<size_t> intersected_face_indices(num_intersected_faces);
  166. std::transform(intersected_faces.begin(),
  167. intersected_faces.end(),
  168. intersected_face_indices.begin(),
  169. [&](const Tree::Primitive_id& itr) -> size_t
  170. { return I(itr-triangles.begin(), 0); });
  171. std::set<size_t> adj_vertices_set;
  172. for (auto fid : intersected_face_indices) {
  173. const auto& f = F.row(fid);
  174. if (f[0] != s) adj_vertices_set.insert(f[0]);
  175. if (f[1] != s) adj_vertices_set.insert(f[1]);
  176. if (f[2] != s) adj_vertices_set.insert(f[2]);
  177. }
  178. const size_t num_adj_vertices = adj_vertices_set.size();
  179. std::vector<size_t> adj_vertices(num_adj_vertices);
  180. std::copy(adj_vertices_set.begin(), adj_vertices_set.end(),
  181. adj_vertices.begin());
  182. std::vector<Point_3> adj_points;
  183. for (size_t vid : adj_vertices) {
  184. adj_points.emplace_back(V(vid,0), V(vid,1), V(vid,2));
  185. }
  186. // A plane is on the exterior if all adj_points lies on or to
  187. // one side of the plane.
  188. auto is_on_exterior = [&](const Plane_3& separator) {
  189. size_t positive=0;
  190. size_t negative=0;
  191. size_t coplanar=0;
  192. for (const auto& point : adj_points) {
  193. switch(separator.oriented_side(point)) {
  194. case CGAL::ON_POSITIVE_SIDE:
  195. positive++;
  196. break;
  197. case CGAL::ON_NEGATIVE_SIDE:
  198. negative++;
  199. break;
  200. case CGAL::ON_ORIENTED_BOUNDARY:
  201. coplanar++;
  202. break;
  203. default:
  204. throw "Unknown plane-point orientation";
  205. }
  206. }
  207. auto query_orientation = separator.oriented_side(query_point);
  208. bool r = (positive == 0 && query_orientation == CGAL::POSITIVE)
  209. || (negative == 0 && query_orientation == CGAL::NEGATIVE);
  210. return r;
  211. };
  212. size_t d = std::numeric_limits<size_t>::max();
  213. for (size_t i=0; i<num_adj_vertices; i++) {
  214. const size_t vi = adj_vertices[i];
  215. for (size_t j=i+1; j<num_adj_vertices; j++) {
  216. Plane_3 separator(closest_point, adj_points[i], adj_points[j]);
  217. if (separator.is_degenerate()) {
  218. throw std::runtime_error(
  219. "Input mesh contains degenerated faces");
  220. }
  221. if (is_on_exterior(separator)) {
  222. d = vi;
  223. assert(!CGAL::collinear(
  224. query_point, adj_points[i], closest_point));
  225. break;
  226. }
  227. }
  228. }
  229. assert(d != std::numeric_limits<size_t>::max());
  230. return process_edge_case(query_idx, s, d, orientation);
  231. };
  232. const size_t num_queries = P.rows();
  233. R.resize(num_queries, 1);
  234. S.resize(num_queries, 1);
  235. for (size_t i=0; i<num_queries; i++) {
  236. const Point_3 query(P(i,0), P(i,1), P(i,2));
  237. auto projection = tree.closest_point_and_primitive(query);
  238. const Point_3 closest_point = projection.first;
  239. size_t fid = projection.second - triangles.begin();
  240. bool fid_ori = false;
  241. // Gether all facets sharing the closest point.
  242. std::vector<Tree::Primitive_id> intersected_faces;
  243. tree.all_intersected_primitives(Segment_3(closest_point, query),
  244. std::back_inserter(intersected_faces));
  245. const size_t num_intersected_faces = intersected_faces.size();
  246. std::vector<size_t> intersected_face_indices(num_intersected_faces);
  247. std::transform(intersected_faces.begin(),
  248. intersected_faces.end(),
  249. intersected_face_indices.begin(),
  250. [&](const Tree::Primitive_id& itr) -> size_t
  251. { return I(itr-triangles.begin(), 0); });
  252. size_t element_index;
  253. auto element_type = determine_element_type(closest_point, fid,
  254. element_index);
  255. switch(element_type) {
  256. case VERTEX:
  257. {
  258. const auto& f = F.row(I(fid, 0));
  259. const size_t s = f[element_index];
  260. fid = process_vertex_case(i, s, fid_ori);
  261. }
  262. break;
  263. case EDGE:
  264. {
  265. const auto& f = F.row(I(fid, 0));
  266. const size_t s = f[(element_index+1)%3];
  267. const size_t d = f[(element_index+2)%3];
  268. fid = process_edge_case(i, s, d, fid_ori);
  269. }
  270. break;
  271. case FACE:
  272. {
  273. fid = process_face_case(i, fid, fid_ori);
  274. }
  275. break;
  276. default:
  277. throw std::runtime_error("Unknown element type.");
  278. }
  279. R(i,0) = fid;
  280. S(i,0) = fid_ori;
  281. }
  282. }
  283. template<
  284. typename DerivedV,
  285. typename DerivedF,
  286. typename DerivedP,
  287. typename DerivedR,
  288. typename DerivedS >
  289. IGL_INLINE void igl::cgal::closest_facet(
  290. const Eigen::PlainObjectBase<DerivedV>& V,
  291. const Eigen::PlainObjectBase<DerivedF>& F,
  292. const Eigen::PlainObjectBase<DerivedP>& P,
  293. Eigen::PlainObjectBase<DerivedR>& R,
  294. Eigen::PlainObjectBase<DerivedS>& S) {
  295. const size_t num_faces = F.rows();
  296. Eigen::VectorXi I(num_faces);
  297. I.setLinSpaced(num_faces, 0, num_faces-1);
  298. igl::cgal::closest_facet(V, F, I, P, R, S);
  299. }