closest_facet.cpp 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324
  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. igl::cgal::order_facets_around_edge(V, F, s, d,
  141. intersected_face_signed_indices,
  142. pivot, order);
  143. orientation = intersected_face_signed_indices[order[0]] < 0;
  144. return intersected_face_indices[order[0]];
  145. };
  146. auto process_face_case = [&](
  147. const size_t query_idx, const size_t fid, bool& orientation) {
  148. const auto& f = F.row(I(fid, 0));
  149. return process_edge_case(query_idx, f[0], f[1], orientation);
  150. };
  151. auto process_vertex_case = [&](const size_t query_idx, size_t s,
  152. bool& orientation) {
  153. Point_3 closest_point(V(s, 0), V(s, 1), V(s, 2));
  154. Point_3 query_point(P(query_idx, 0), P(query_idx, 1), P(query_idx, 2));
  155. std::vector<Tree::Primitive_id> intersected_faces;
  156. tree.all_intersected_primitives(Segment_3(closest_point, query_point),
  157. std::back_inserter(intersected_faces));
  158. const size_t num_intersected_faces = intersected_faces.size();
  159. std::vector<size_t> intersected_face_indices(num_intersected_faces);
  160. std::transform(intersected_faces.begin(),
  161. intersected_faces.end(),
  162. intersected_face_indices.begin(),
  163. [&](const Tree::Primitive_id& itr) -> size_t
  164. { return I(itr-triangles.begin(), 0); });
  165. std::set<size_t> adj_vertices_set;
  166. for (auto fid : intersected_face_indices) {
  167. const auto& f = F.row(fid);
  168. if (f[0] != s) adj_vertices_set.insert(f[0]);
  169. if (f[1] != s) adj_vertices_set.insert(f[1]);
  170. if (f[2] != s) adj_vertices_set.insert(f[2]);
  171. }
  172. const size_t num_adj_vertices = adj_vertices_set.size();
  173. std::vector<size_t> adj_vertices(num_adj_vertices);
  174. std::copy(adj_vertices_set.begin(), adj_vertices_set.end(),
  175. adj_vertices.begin());
  176. std::vector<Point_3> adj_points;
  177. for (size_t vid : adj_vertices) {
  178. adj_points.emplace_back(V(vid,0), V(vid,1), V(vid,2));
  179. }
  180. // A plane is on the exterior if all adj_points lies on or to
  181. // one side of the plane.
  182. auto is_on_exterior = [&](const Plane_3& separator) {
  183. size_t positive=0;
  184. size_t negative=0;
  185. size_t coplanar=0;
  186. for (const auto& point : adj_points) {
  187. switch(separator.oriented_side(point)) {
  188. case CGAL::ON_POSITIVE_SIDE:
  189. positive++;
  190. break;
  191. case CGAL::ON_NEGATIVE_SIDE:
  192. negative++;
  193. break;
  194. case CGAL::ON_ORIENTED_BOUNDARY:
  195. coplanar++;
  196. break;
  197. default:
  198. throw "Unknown plane-point orientation";
  199. }
  200. }
  201. auto query_orientation = separator.oriented_side(query_point);
  202. bool r = (positive == 0 && query_orientation == CGAL::POSITIVE)
  203. || (negative == 0 && query_orientation == CGAL::NEGATIVE);
  204. return r;
  205. };
  206. size_t d = std::numeric_limits<size_t>::max();
  207. for (size_t i=0; i<num_adj_vertices; i++) {
  208. const size_t vi = adj_vertices[i];
  209. for (size_t j=i+1; j<num_adj_vertices; j++) {
  210. Plane_3 separator(closest_point, adj_points[i], adj_points[j]);
  211. if (separator.is_degenerate()) {
  212. throw std::runtime_error(
  213. "Input mesh contains degenerated faces");
  214. }
  215. if (is_on_exterior(separator)) {
  216. d = vi;
  217. assert(!CGAL::collinear(
  218. query_point, adj_points[i], closest_point));
  219. break;
  220. }
  221. }
  222. }
  223. assert(d != std::numeric_limits<size_t>::max());
  224. return process_edge_case(query_idx, s, d, orientation);
  225. };
  226. const size_t num_queries = P.rows();
  227. R.resize(num_queries, 1);
  228. S.resize(num_queries, 1);
  229. for (size_t i=0; i<num_queries; i++) {
  230. const Point_3 query(P(i,0), P(i,1), P(i,2));
  231. auto projection = tree.closest_point_and_primitive(query);
  232. const Point_3 closest_point = projection.first;
  233. size_t fid = projection.second - triangles.begin();
  234. bool fid_ori = false;
  235. // Gether all facets sharing the closest point.
  236. std::vector<Tree::Primitive_id> intersected_faces;
  237. tree.all_intersected_primitives(Segment_3(closest_point, query),
  238. std::back_inserter(intersected_faces));
  239. const size_t num_intersected_faces = intersected_faces.size();
  240. std::vector<size_t> intersected_face_indices(num_intersected_faces);
  241. std::transform(intersected_faces.begin(),
  242. intersected_faces.end(),
  243. intersected_face_indices.begin(),
  244. [&](const Tree::Primitive_id& itr) -> size_t
  245. { return I(itr-triangles.begin(), 0); });
  246. size_t element_index;
  247. auto element_type = determine_element_type(closest_point, fid,
  248. element_index);
  249. switch(element_type) {
  250. case VERTEX:
  251. {
  252. const auto& f = F.row(I(fid, 0));
  253. const size_t s = f[element_index];
  254. fid = process_vertex_case(i, s, fid_ori);
  255. }
  256. break;
  257. case EDGE:
  258. {
  259. const auto& f = F.row(I(fid, 0));
  260. const size_t s = f[(element_index+1)%3];
  261. const size_t d = f[(element_index+2)%3];
  262. fid = process_edge_case(i, s, d, fid_ori);
  263. }
  264. break;
  265. case FACE:
  266. {
  267. fid = process_face_case(i, fid, fid_ori);
  268. }
  269. break;
  270. default:
  271. throw std::runtime_error("Unknown element type.");
  272. }
  273. R(i,0) = fid;
  274. S(i,0) = fid_ori;
  275. }
  276. }
  277. template<
  278. typename DerivedV,
  279. typename DerivedF,
  280. typename DerivedP,
  281. typename DerivedR,
  282. typename DerivedS >
  283. IGL_INLINE void igl::cgal::closest_facet(
  284. const Eigen::PlainObjectBase<DerivedV>& V,
  285. const Eigen::PlainObjectBase<DerivedF>& F,
  286. const Eigen::PlainObjectBase<DerivedP>& P,
  287. Eigen::PlainObjectBase<DerivedR>& R,
  288. Eigen::PlainObjectBase<DerivedS>& S) {
  289. const size_t num_faces = F.rows();
  290. Eigen::VectorXi I(num_faces);
  291. I.setLinSpaced(num_faces, 0, num_faces-1);
  292. igl::cgal::closest_facet(V, F, I, P, R, S);
  293. }