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 "insertpoints.h"
24 #include "guimainwindow.h"
26 #include <vtkSignedCharArray.h>
30 InsertPoints::InsertPoints() : SurfaceOperation()
33 getSet("surface meshing", "point insertion threshold", 1, m_Threshold
);
36 void InsertPoints::operate()
38 int N1
= m_Grid
->GetNumberOfPoints();
40 int N2
= m_Grid
->GetNumberOfPoints();
41 m_NumInserted
= N2
- N1
;
44 ///\todo Adapt this code for multiple volumes.
45 int InsertPoints::insertPoints()
48 l2g_t cells
= getPartCells();
49 g2l_t _cells
= getPartLocalCells();
53 EG_VTKDCC(vtkIntArray
, cell_code
, m_Grid
, "cell_code");
54 EG_VTKDCN(vtkDoubleArray
, characteristic_length_desired
, m_Grid
, "node_meshdensity_desired");
59 QVector
<int> marked_cells(cells
.size(), 0);
60 QVector
<stencil_t
> stencil_vector(cells
.size());
62 // find potential edges for splitting
64 for (int i
= 0; i
< cells
.size(); ++i
) {
65 vtkIdType id_cell
= cells
[i
];
67 // if cell is selected and a triangle
68 if (m_BoundaryCodes
.contains(cell_code
->GetValue(id_cell
)) && (m_Grid
->GetCellType(id_cell
) == VTK_TRIANGLE
)) {
71 EG_GET_CELL(id_cell
, m_Grid
);
73 //find best side to split (longest)
74 for (int j
= 0; j
< 3; ++j
) {
76 //check if neighbour cell on this side is also selected
77 stencil_t S
= getStencil(id_cell
, j
);
78 bool selected_edge
= true;
79 for (int i_cell_neighbour
=1;i_cell_neighbour
<S
.id_cell
.size();i_cell_neighbour
++) {
80 vtkIdType id_cell_neighbour
= S
.id_cell
[i_cell_neighbour
];
81 if (!m_BoundaryCodes
.contains(cell_code
->GetValue(id_cell_neighbour
)) || S
.type_cell
[i_cell_neighbour
] != VTK_TRIANGLE
) {
84 } // end of loop through neighbour cells
87 vtkIdType id_node1
= pts
[j
];
88 vtkIdType id_node2
= pts
[(j
+1)%num_pts
];
89 double L
= distance(m_Grid
, id_node1
, id_node2
);
90 double L1
= characteristic_length_desired
->GetValue(id_node1
);
91 double L2
= characteristic_length_desired
->GetValue(id_node2
);
92 if (L
> m_Threshold
*min(L1
,L2
)) {
99 } // end of loop through sides
101 stencil_t S
= getStencil(id_cell
, j_split
);
104 E
.L1
= characteristic_length_desired
->GetValue(S
.p1
);
105 E
.L2
= characteristic_length_desired
->GetValue(S
.p2
);
106 E
.L12
= distance(m_Grid
, S
.p1
, S
.p2
);
115 foreach (edge_t E
, edges
) {
116 int i_cells1
= _cells
[E
.S
.id_cell
[0]];
117 bool all_unmarked
= true;
118 for (int i
= 0; i
< E
.S
.id_cell
.size(); i
++) {
119 int i_cells
= _cells
[E
.S
.id_cell
[i
]];
120 if (marked_cells
[i_cells
]) {
125 stencil_vector
[i_cells1
] = E
.S
;
126 marked_cells
[i_cells1
] = 1;
127 for (int i
= 1; i
< E
.S
.id_cell
.size(); i
++) {
128 int i_cells
= _cells
[E
.S
.id_cell
[i
]];
129 marked_cells
[i_cells
] = 2;
132 num_newcells
+= E
.S
.id_cell
.size();
136 //initialize grid_tmp
137 int l_N_points
=m_Grid
->GetNumberOfPoints();
138 int l_N_cells
=m_Grid
->GetNumberOfCells();
139 EG_VTKSP(vtkUnstructuredGrid
,grid_tmp
);
140 allocateGrid(grid_tmp
, l_N_cells
+ num_newcells
, l_N_points
+ num_newpoints
);
141 makeCopyNoAlloc(m_Grid
, grid_tmp
);
143 //initialize new node counter
144 vtkIdType id_new_node
= l_N_points
;
147 for (int i
= 0; i
< cells
.size(); i
++) {
148 if (marked_cells
[i
] == 1) {
149 stencil_t S
= stencil_vector
[i
];
153 grid_tmp
->GetPoint(S
.p1
,A
.data());
154 grid_tmp
->GetPoint(S
.p2
,B
.data());
158 grid_tmp
->GetPoints()->SetPoint(id_new_node
, M
.data());
159 copyNodeData(grid_tmp
,S
.p1
,grid_tmp
,id_new_node
); ///\todo maybe trouble
161 // inserted edge point = type of the edge on which it is inserted
162 EG_VTKDCN(vtkCharArray_t
, node_type
, grid_tmp
, "node_type");
163 node_type
->SetValue(id_new_node
, getNewNodeType(S
) );
167 int N
= S
.id_cell
.size();
168 QVector
< QVector
<vtkIdType
> > pts_triangle(2*N
,QVector
<vtkIdType
>(3));
170 for(int i_triangle
= 0; i_triangle
< N
; i_triangle
++) {
171 EG_GET_CELL(S
.id_cell
[i_triangle
], grid_tmp
);
174 for(int i_pts
= 0; i_pts
< num_pts
; i_pts
++) {
175 if (pts
[i_pts
] == S
.p1
) {
176 if (pts
[(i_pts
+1)%num_pts
] == S
.p2
) {
184 pts_triangle
[i_triangle
][0] = S
.p1
;
185 pts_triangle
[i_triangle
][1] = id_new_node
;
186 pts_triangle
[i_triangle
][2] = S
.id_node
[i_triangle
];
188 pts_triangle
[i_triangle
+N
][0] = id_new_node
;
189 pts_triangle
[i_triangle
+N
][1] = S
.p2
;
190 pts_triangle
[i_triangle
+N
][2] = S
.id_node
[i_triangle
];
192 pts_triangle
[i_triangle
][0] = S
.p2
;
193 pts_triangle
[i_triangle
][1] = id_new_node
;
194 pts_triangle
[i_triangle
][2] = S
.id_node
[i_triangle
];
196 pts_triangle
[i_triangle
+N
][0] = id_new_node
;
197 pts_triangle
[i_triangle
+N
][1] = S
.p1
;
198 pts_triangle
[i_triangle
+N
][2] = S
.id_node
[i_triangle
];
201 grid_tmp
->ReplaceCell(S
.id_cell
[i_triangle
] , 3, pts_triangle
[i_triangle
].data());
202 auto new_pts
= idListFromVector(pts_triangle
[i_triangle
+N
]);
203 vtkIdType newCellId
= grid_tmp
->InsertNextCell(VTK_TRIANGLE
, new_pts
);
204 //vtkIdType newCellId = grid_tmp->InsertNextCell(VTK_TRIANGLE,3,pts_triangle[i_triangle+N].data());
205 copyCellData(grid_tmp
,S
.id_cell
[i_triangle
],grid_tmp
,newCellId
);
214 makeCopy(grid_tmp
,m_Grid
);
219 char InsertPoints::getNewNodeType(stencil_t S
)
221 vtkIdType id_node1
= S
.p1
;
222 vtkIdType id_node2
= S
.p2
;
224 EG_VTKDCN(vtkCharArray_t
, node_type
, m_Grid
, "node_type");
225 EG_VTKDCC(vtkIntArray
, cell_code
, m_Grid
, "cell_code");
226 char type1
= node_type
->GetValue(id_node1
);
227 char type2
= node_type
->GetValue(id_node2
);
228 if (type1
== EG_SIMPLE_VERTEX
|| type2
== EG_SIMPLE_VERTEX
) {
229 return EG_SIMPLE_VERTEX
;
231 QVector
<vtkIdType
> PSP1
= getPotentialSnapPoints(id_node1
);
232 QVector
<vtkIdType
> PSP2
= getPotentialSnapPoints(id_node2
);
233 bool complex = false;
234 if (PSP1
.contains(id_node2
) || PSP2
.contains(id_node1
)) {
236 } else if (type1
>= EG_BOUNDARY_EDGE_VERTEX
&& type2
>= EG_BOUNDARY_EDGE_VERTEX
) {
237 if (!PSP1
.contains(id_node2
) && !PSP2
.contains(id_node1
) && S
.id_cell
.size() == 2) {
238 int bc1
= cell_code
->GetValue(S
.id_cell
[0]);
239 int bc2
= cell_code
->GetValue(S
.id_cell
[1]);
241 return EG_BOUNDARY_EDGE_VERTEX
;
243 vec3_t n1
= cellNormal(m_Grid
, S
.id_cell
[0]);
244 vec3_t n2
= cellNormal(m_Grid
, S
.id_cell
[1]);
245 if (GeometryTools::angle(n1
, n2
) > m_FeatureAngle
) {
246 return EG_FEATURE_EDGE_VERTEX
;
251 if (S
.id_cell
.size() < 1) {
252 return EG_BOUNDARY_EDGE_VERTEX
;
253 } else if (S
.id_cell
.size() == 1) {
254 EG_ERR_RETURN("Invalid surface mesh. Check this with 'Tools -> Check surface integrity'.");
256 if (cell_code
->GetValue(S
.id_cell
[0]) != cell_code
->GetValue(S
.id_cell
[1])) {
257 return EG_BOUNDARY_EDGE_VERTEX
;
259 if (isFeatureNode(id_node1
) || isFeatureNode(id_node2
)) {
260 return EG_FEATURE_EDGE_VERTEX
;
262 return EG_SIMPLE_VERTEX
;
267 return EG_SIMPLE_VERTEX
;