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 // ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
21 #include "laplacesmoother.h"
22 #include <vtkCellLocator.h>
23 #include <vtkSignedCharArray.h>
24 #include <vtkGenericCell.h>
26 #include "guimainwindow.h"
27 #include "localnodegraphinterface.h"
28 #include "checkerboardgraphiterator.h"
29 #include "snaptofeatures.h"
31 using namespace GeometryTools
;
33 LaplaceSmoother::LaplaceSmoother() : SurfaceOperation()
37 m_UseProjection
= true;
38 // m_UseNormalCorrection = false;
39 getSet("surface meshing", "under relaxation for smoothing", 0.5, m_UnderRelaxation
);
40 getSet("surface meshing", "feature magic", 1.0, m_FeatureMagic
);
41 getSet("surface meshing", "smoothing limiter", 1.0, m_Limit
);
42 getSet("surface meshing", "use uniform smoothing", false, m_UniformSnapPoints
);
43 m_Limit
= min(1.0, max(0.0, m_Limit
));
45 m_ProjectionIterations
= 50;
46 m_AllowedCellTypes
.clear();
47 m_AllowedCellTypes
.insert(VTK_TRIANGLE
);
48 //m_StrictFeatureSnap = false;
51 bool LaplaceSmoother::setNewPosition(vtkIdType id_node
, vec3_t x_new
)
53 using namespace GeometryTools
;
56 m_Grid
->GetPoint(id_node
, x_old
.data());
57 EG_VTKDCN(vtkCharArray_t
, node_type
, m_Grid
, "node_type");
60 m_Grid
->GetPoints()->SetPoint(id_node
, x_new
.data());
63 QVector
<vec3_t
> old_cell_normals(m_Part
.n2cGSize(id_node
));
64 EG_VTKDCC(vtkIntArray
, cell_code
, m_Grid
, "cell_code");
65 for (int i
= 0; i
< m_Part
.n2cGSize(id_node
); ++i
) {
66 old_cell_normals
[i
] = GeometryTools::cellNormal(m_Grid
, m_Part
.n2cGG(id_node
, i
));
68 m_Grid
->GetPoints()->SetPoint(id_node
, x_new
.data());
70 for (int i
= 0; i
< m_Part
.n2cGSize(id_node
); ++i
) {
71 vec3_t n
= GeometryTools::cellNormal(m_Grid
, m_Part
.n2cGG(id_node
, i
));
72 if (n
*old_cell_normals
[i
] < 0.2*old_cell_normals
[i
].abs2()) {
78 m_Grid
->GetPoints()->SetPoint(id_node
, x_old
.data());
83 void LaplaceSmoother::featureCorrection(vtkIdType id_node
, CadInterface
*cad_interface
, vec3_t
&x_new
)
85 EG_VTKDCN(vtkCharArray_t
, node_type
, m_Grid
, "node_type");
86 char type
= node_type
->GetValue(id_node
);
87 if (type
== EG_FEATURE_EDGE_VERTEX
|| type
== EG_BOUNDARY_EDGE_VERTEX
) {
88 x_new
= cad_interface
->snapToEdge(x_new
);
92 bool LaplaceSmoother::moveNode(vtkIdType id_node
, vec3_t
&Dx
)
94 if (!checkVector(Dx
)) {
97 EG_VTKDCN(vtkDoubleArray
, cl
, m_Grid
, "node_meshdensity_desired");
99 m_Grid
->GetPoint(id_node
, x_old
.data());
102 for (int i_relaxation
= 0; i_relaxation
< 10; ++i_relaxation
) {
103 vec3_t x_new
= x_old
+ Dx
;
104 if (m_UseProjection
) {
105 int i_nodes
= m_Part
.localNode(id_node
);
106 int bc
= m_NodeToBc
[i_nodes
][0];
107 x_new
= GuiMainWindow::pointer()->getCadInterface(bc
)->snapNode(id_node
, x_new
, m_CorrectCurvature
);
108 featureCorrection(id_node
, GuiMainWindow::pointer()->getCadInterface(bc
), x_new
);
111 // compute the minimal length of any edge adjacent to this node
112 // .. This will be used to limit the node movement.
113 // .. Hopefully jammed topologies can be avoided this way.
115 EG_VTKDCN(vtkDoubleArray
, cl
, m_Grid
, "node_meshdensity_desired");
116 double L_min
= cl
->GetValue(id_node
);
117 for (int i
= 0; i
< m_Part
.n2nGSize(id_node
); ++i
) {
118 vtkIdType id_neigh
= m_Part
.n2nGG(id_node
, i
);
120 m_Grid
->GetPoint(id_neigh
, x_neigh
.data());
121 L_min
= min(L_min
, (x_old
- x_neigh
).abs());
124 if (setNewPosition(id_node
, x_new
)) {
125 m_Grid
->GetPoints()->SetPoint(id_node
, x_new
.data());
135 void LaplaceSmoother::fixNodes(const QVector
<bool> &fixnodes
)
137 if (fixnodes
.size() != m_Grid
->GetNumberOfPoints()) {
143 void LaplaceSmoother::operate()
145 m_NoCheck
= false; // !!!!!!!!!!!!!!
146 if (m_Fixed
.size() != m_Grid
->GetNumberOfPoints()) {
147 m_Fixed
.fill(false, m_Grid
->GetNumberOfPoints());
150 GuiMainWindow::pointer()->getAllBoundaryCodes(bcs
);
151 if (m_UseProjection
) {
152 foreach (int bc
, bcs
) {
153 GuiMainWindow::pointer()->getCadInterface(bc
)->setForegroundGrid(m_Grid
);
157 EG_VTKDCC(vtkIntArray
, cell_code
, m_Grid
, "cell_code");
158 EG_VTKDCN(vtkCharArray_t
, node_type
, m_Grid
, "node_type" );
159 EG_VTKDCN(vtkDoubleArray
, cl
, m_Grid
, "node_meshdensity_desired");
160 QVector
<vtkIdType
> smooth_node(m_Grid
->GetNumberOfPoints(), false);
162 l2g_t nodes
= m_Part
.getNodes();
163 foreach (vtkIdType id_node
, nodes
) {
164 smooth_node
[id_node
] = true;
167 setAllSurfaceCells();
168 l2g_t nodes
= m_Part
.getNodes();
169 m_NodeToBc
.resize(nodes
.size());
170 for (int i_nodes
= 0; i_nodes
< nodes
.size(); ++i_nodes
) {
172 for (int j
= 0; j
< m_Part
.n2cLSize(i_nodes
); ++j
) {
173 bcs
.insert(cell_code
->GetValue(m_Part
.n2cLG(i_nodes
, j
)));
175 m_NodeToBc
[i_nodes
].resize(bcs
.size());
176 qCopy(bcs
.begin(), bcs
.end(), m_NodeToBc
[i_nodes
].begin());
179 QVector
<vec3_t
> x_new(nodes
.size());
181 QVector
<bool> blocked(nodes
.size(), false);
182 for (int i_nodes
= 0; i_nodes
< nodes
.size(); ++i_nodes
) {
183 if (node_type
->GetValue(nodes
[i_nodes
]) == EG_FEATURE_CORNER_VERTEX
) {
184 blocked
[i_nodes
] = true;
186 for (int i
= 0; i
< m_Part
.n2cLSize(i_nodes
); ++i
) {
187 vtkIdType type
= m_Grid
->GetCellType(m_Part
.n2cLG(i_nodes
, i
));
188 if (!m_AllowedCellTypes
.contains(type
)) {
189 blocked
[i_nodes
] = true;
196 for (int i_iter
= 0; i_iter
< m_NumberOfIterations
; ++i_iter
) {
200 for (int i_nodes
= 0; i_nodes
< nodes
.size(); ++i_nodes
) {
201 vtkIdType id_node
= nodes
[i_nodes
];
202 m_Grid
->GetPoint(id_node
, x_new
[i_nodes
].data());
203 if (!m_Fixed
[id_node
] && !blocked
[i_nodes
]) {
204 if (smooth_node
[id_node
] && node_type
->GetValue(id_node
) != EG_FIXED_VERTEX
) {
205 if (node_type
->GetValue(id_node
) != EG_FIXED_VERTEX
) {
206 QVector
<vtkIdType
> snap_points
= getPotentialSnapPoints(id_node
);
208 vec3_t x_old
= x_new
[i_nodes
];
210 if (snap_points
.size() > 0) {
212 x_new
[i_nodes
] = vec3_t(0,0,0);
215 foreach (vtkIdType id_snap_node
, snap_points
) {
216 m_Grid
->GetPoint(id_snap_node
, x
.data());
219 x_new
[i_nodes
] += w
*x
;
220 n
+= m_NodeNormal
[id_snap_node
];
221 double L
= (x
- x_old
).abs();
222 L_min
= min(L
, L_min
);
225 x_new
[i_nodes
] *= 1.0/w_tot
;
227 x_new
[i_nodes
] = x_old
;
230 if (m_UseNormalCorrection
) {
231 vec3_t dx
= x_new
[i_nodes
] - x_old
;
232 double scal
= dx
*m_NodeNormal
[id_node
];
233 x_new
[i_nodes
] += scal
*m_NodeNormal
[id_node
];
235 vec3_t Dx
= x_new
[i_nodes
] - x_old
;
236 //Dx *= m_UnderRelaxation;
237 if (moveNode(id_node
, Dx
)) {
238 x_new
[i_nodes
] = x_old
+ m_UnderRelaxation
*Dx
;
240 x_new
[i_nodes
] = x_old
;
243 m_Grid
->GetPoints()->SetPoint(id_node
, x_old
.data());
248 for (int i_nodes
= 0; i_nodes
< nodes
.size(); ++i_nodes
) {
249 vtkIdType id_node
= nodes
[i_nodes
];
250 m_Grid
->GetPoints()->SetPoint(id_node
, x_new
[i_nodes
].data());