1 // ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
3 // + This file is part of enGrid. +
5 // + Copyright 2008-2014 enGits GmbH +
7 // + enGrid is free software: you can redistribute it and/or modify +
8 // + it under the terms of the GNU General Public License as published by +
9 // + the Free Software Foundation, either version 3 of the License, or +
10 // + (at your option) any later version. +
12 // + enGrid is distributed in the hope that it will be useful, +
13 // + but WITHOUT ANY WARRANTY; without even the implied warranty of +
14 // + MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +
15 // + GNU General Public License for more details. +
17 // + You should have received a copy of the GNU General Public License +
18 // + along with enGrid. If not, see <http://www.gnu.org/licenses/>. +
20 // ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
22 #include "cgaltricadinterface.h"
23 #include "meshpartition.h"
25 #include <CGAL/exceptions.h>
27 CgalTriCadInterface::CgalTriCadInterface(vtkUnstructuredGrid
*grid
)
29 m_BGrid
= vtkUnstructuredGrid::New();
30 makeCopy(grid
, m_BGrid
);
31 m_BPart
.setGrid(m_BGrid
, true);
35 //EG_STOPDATE("2015-06-01");
36 //getXmlSetting("engrid/surface/settings", "feature_angle", feature_angle);
37 getSet("surface meshing", "feature angle", 20, feature_angle
);
38 feature_angle
= GeometryTools::deg2rad(feature_angle
);
40 // build triangle tree
43 QVector
<vtkIdType
> tris
;
44 getAllCellsOfType(VTK_TRIANGLE
, tris
, m_BGrid
);
45 m_Triangles
.fill(Triangle(), tris
.size());
46 m_Tri2Grid
.fill(-1, tris
.size());
48 foreach (vtkIdType id_cell
, tris
) {
49 EG_GET_CELL(id_cell
, m_BGrid
);
51 m_BGrid
->GetPoint(pts
[0], a
.data());
52 m_BGrid
->GetPoint(pts
[1], b
.data());
53 m_BGrid
->GetPoint(pts
[2], c
.data());
54 m_Triangles
[i
] = Triangle(Point(a
[0], a
[1], a
[2]), Point(b
[0], b
[1], b
[2]), Point(c
[0], c
[1], c
[2]));
55 m_Tri2Grid
[i
] = id_cell
;
58 m_TriangleTree
.rebuild(m_Triangles
.begin(), m_Triangles
.end());
59 m_TriangleTree
.accelerate_distance_queries();
66 EG_VTKDCC(vtkIntArray
, cell_code
, m_BGrid
, "cell_code");
67 EG_FORALL_NODES(id_node1
, m_BGrid
) {
68 for (int i
= 0; i
< m_BPart
.n2nGSize(id_node1
); ++i
) {
69 vtkIdType id_node2
= m_BPart
.n2nGG(id_node1
, i
);
70 if (id_node1
< id_node2
) {
71 QVector
<vtkIdType
> id_face
;
72 m_BPart
.getEdgeFaces(id_node1
, id_node2
, id_face
);
73 if (id_face
.size() == 2) {
74 bool append_edge
= false;
75 if (cell_code
->GetValue(id_face
[0]) != cell_code
->GetValue(id_face
[1])) {
78 vec3_t n1
= GeometryTools::cellNormal(m_BGrid
, id_face
[0]);
79 vec3_t n2
= GeometryTools::cellNormal(m_BGrid
, id_face
[1]);
80 double alpha
= GeometryTools::angle(n1
, n2
);
81 if (alpha
>= feature_angle
) {
87 m_BGrid
->GetPoint(id_node1
, x1
.data());
88 m_BGrid
->GetPoint(id_node2
, x2
.data());
89 segs
.append(Segment(Point(x1
[0], x1
[1], x1
[2]), Point(x2
[0], x2
[1], x2
[2])));
95 m_Segments
.resize(segs
.size());
96 qCopy(segs
.begin(), segs
.end(), m_Segments
.begin());
97 m_SegmentTree
.rebuild(m_Segments
.begin(), m_Segments
.end());
98 m_SegmentTree
.accelerate_distance_queries();
101 setName("CgalTriCadInterface");
102 computeSurfaceCurvature();
103 //m_ShootRayImplemented = true;
106 void CgalTriCadInterface::computeSurfaceCurvature()
108 QVector
<double> node_radius(m_BGrid
->GetNumberOfPoints(), EG_LARGE_REAL
);
109 for (vtkIdType id_node
= 0; id_node
< m_BGrid
->GetNumberOfPoints(); ++id_node
) {
111 m_BGrid
->GetPoint(id_node
, x1
.data());
112 vec3_t n1
= m_BPart
.globalNormal(id_node
);
113 for (int i
= 0; i
< m_BPart
.n2nGSize(id_node
); ++i
) {
114 vtkIdType id_neigh
= m_BPart
.n2nGG(id_node
, i
);
115 vec3_t n2
= m_BPart
.globalNormal(id_neigh
);
116 double scal_prod
= max(-1.0, min(1.0, n1
*n2
));
117 double alpha
= max(1e-3, acos(scal_prod
));
120 m_BGrid
->GetPoint(id_neigh
, x2
.data());
121 double a
= (x1
- x2
).abs();
122 node_radius
[id_node
] = min(node_radius
[id_node
], a
/alpha
);
127 // compute weighted (distance) average of radii
128 QVector
<double> R_new(m_BGrid
->GetNumberOfPoints(), 1e99
);
129 for (vtkIdType id_node
= 0; id_node
< m_BGrid
->GetNumberOfPoints(); ++id_node
) {
131 m_BGrid
->GetPoint(id_node
, x1
.data());
132 int N
= m_BPart
.n2nGSize(id_node
);
133 QVector
<double> L(N
);
134 QVector
<double> R(N
, -1);
136 bool average
= false;
137 for (int i
= 0; i
< N
; ++i
) {
138 vtkIdType id_neigh
= m_BPart
.n2nGG(id_node
, i
);
140 m_BGrid
->GetPoint(id_neigh
, x2
.data());
141 L
[i
] = (x2
- x1
).abs();
142 if (node_radius
[id_neigh
] < 1e90
&& L
[i
] > 0) {
143 R
[i
] = node_radius
[id_neigh
];
144 Lmax
= max(Lmax
, L
[i
]);
150 double total_weight
= 0;
151 for (int i
= 0; i
< N
; ++i
) {
153 R_new
[id_node
] += Lmax
/L
[i
] * R
[i
];
154 total_weight
+= Lmax
/L
[i
];
155 //R_new[id_node] += R[i];
156 //total_weight += 1.0;
159 R_new
[id_node
] /= total_weight
;
164 m_Radius
.fill(0, m_BGrid
->GetNumberOfCells());
165 EG_FORALL_CELLS(id_cell
, m_BGrid
) {
166 EG_GET_CELL(id_cell
, m_BGrid
);
167 for (int i
= 0; i
< num_pts
; ++i
) {
168 m_Radius
[id_cell
] = max(m_Radius
[id_cell
], node_radius
[pts
[i
]]);
173 CgalTriCadInterface::Ray
CgalTriCadInterface::createRay(vec3_t x1
, vec3_t v
)
175 Point
p1(x1
[0], x1
[1], x1
[2]);
177 Point
p2(x2
[0], x2
[1], x2
[2]);
181 vec3_t
CgalTriCadInterface::snap(vec3_t x
, bool)
185 Point
p(x
[0], x
[1], x
[2]);
186 TrianglePointAndPrimitiveId result
= m_TriangleTree
.closest_point_and_primitive(p
);
187 Point cp
= result
.first
;
188 Triangle
* T
= result
.second
;
189 int id
= (T
- m_Triangles
.begin());
190 vtkIdType id_face
= m_Tri2Grid
[id
];
191 m_LastNormal
= GeometryTools::cellNormal(m_BGrid
, id_face
);
192 m_LastNormal
.normalise();
193 m_LastRadius
= m_Radius
[id_face
];
194 x_snap
= vec3_t(cp
[0], cp
[1], cp
[2]);
195 } catch (CGAL::Failure_exception
) {
201 vec3_t
CgalTriCadInterface::snapWithNormal(vec3_t x
, vec3_t n_surf
, bool correct_curvature
)
205 x_snap
= snap(x
, correct_curvature
);
206 if (n_surf
*m_LastNormal
< 0) {
208 double dist_min
= EG_LARGE_REAL
;
209 Ray ray
= createRay(x
, n_surf
);
210 std::list
<Intersection
> intersections
;
211 m_TriangleTree
.all_intersections(ray
, std::back_inserter(intersections
));
212 for (std::list
<Intersection
>::iterator i
= intersections
.begin(); i
!= intersections
.end(); ++i
) {
213 int id
= (i
->second
- m_Triangles
.begin());
214 vtkIdType id_face
= m_Tri2Grid
[id
];
215 vec3_t n
= GeometryTools::cellNormal(m_BGrid
, id_face
);
218 if (i
->first
.type() == typeid(Point
)) {
219 Point p
= boost::get
<Point
>(i
->first
);
220 vec3_t
xs(p
[0], p
[1], p
[2]);
221 double dist
= (x
- xs
).abs();
222 if (dist
< dist_min
) {
226 m_LastRadius
= m_Radius
[id_face
];
232 } catch (CGAL::Failure_exception
) {
238 vec3_t
CgalTriCadInterface::snapNode(vtkIdType id_node
, vec3_t x
, bool correct_curvature
)
242 x_snap
= snap(x
, correct_curvature
);
243 vec3_t n_node
= m_FPart
.globalNormal(id_node
);
244 if (n_node
*m_LastNormal
< 0) {
246 double dist_min
= EG_LARGE_REAL
;
247 Ray ray
= createRay(x
, n_node
);
248 std::list
<Intersection
> intersections
;
249 m_TriangleTree
.all_intersections(ray
, std::back_inserter(intersections
));
250 for (std::list
<Intersection
>::iterator i
= intersections
.begin(); i
!= intersections
.end(); ++i
) {
251 int id
= (i
->second
- m_Triangles
.begin());
252 vtkIdType id_face
= m_Tri2Grid
[id
];
253 vec3_t n
= GeometryTools::cellNormal(m_BGrid
, id_face
);
256 if (i
->first
.type() == typeid(Point
)) {
257 Point p
= boost::get
<Point
>(i
->first
);
258 vec3_t
xs(p
[0], p
[1], p
[2]);
259 double dist
= (x
- xs
).abs();
260 if (dist
< dist_min
) {
264 m_LastRadius
= m_Radius
[id_face
];
270 } catch (CGAL::Failure_exception
) {
276 vec3_t
CgalTriCadInterface::snapToEdge(vec3_t x
)
280 Point
p(x
[0], x
[1], x
[2]);
281 Point cp
= m_SegmentTree
.closest_point(p
);
282 x_snap
= vec3_t(cp
[0], cp
[1], cp
[2]);
283 } catch (CGAL::Failure_exception
) {
289 vec3_t
CgalTriCadInterface::snapToCorner(vec3_t
)
295 CgalTriCadInterface::HitType
CgalTriCadInterface::shootRay(vec3_t x
, vec3_t v
, vec3_t
&x_hit
, vec3_t
&n_hit
, double &r
)
298 return CgalTriCadInterface::HitType();
301 void CgalTriCadInterface::computeIntersections(vec3_t x
, vec3_t v
, QVector
<QPair
<vec3_t
, vtkIdType
> > &intersections
)
304 Ray ray
= createRay(x
, v
);
305 std::list
<Intersection
> inters
;
306 m_TriangleTree
.all_intersections(ray
, std::back_inserter(inters
));
307 int i_intersections
= 0;
308 for (std::list
<Intersection
>::iterator i
= inters
.begin(); i
!= inters
.end(); ++i
) {
309 if (i
->first
.type() == typeid(Point
)) {
313 intersections
.resize(i_intersections
);
315 for (std::list
<Intersection
>::iterator i
= inters
.begin(); i
!= inters
.end(); ++i
) {
316 int id
= (i
->second
- m_Triangles
.begin());
317 if (i
->first
.type() == typeid(Point
)) {
318 intersections
[i_intersections
].second
= m_Tri2Grid
[id
];
319 Point p
= boost::get
<Point
>(i
->first
);
320 vec3_t
xs(p
[0], p
[1], p
[2]);
321 intersections
[i_intersections
].first
= xs
;
325 } catch (CGAL::Failure_exception
) {
326 intersections
.clear();