Loading [MathJax]/extensions/MathMenu.js

DUNE-GRID-GLUE (2.10)

codim1extractor.hh
Go to the documentation of this file.
1// -*- tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 2 -*-
2// vi: set et ts=4 sw=2 sts=2:
3// SPDX-FileCopyrightInfo: Copyright © DUNE Project contributors, see file LICENSE.md in module root
4// SPDX-License-Identifier: LGPL-3.0-or-later OR LicenseRef-GPL-2.0-only-with-dune-grid-glue-exception
5/*
6 * Filename: codim1extractor.hh
7 * Version: 1.0
8 * Created on: Jun 23, 2009
9 * Author: Oliver Sander, Christian Engwer
10 * ---------------------------------
11 * Project: dune-grid-glue
12 * Description: class for grid extractors extracting surface grids
13 *
14 */
20#ifndef DUNE_GRIDGLUE_EXTRACTORS_CODIM1EXTRACTOR_HH
21#define DUNE_GRIDGLUE_EXTRACTORS_CODIM1EXTRACTOR_HH
22
23#include "extractor.hh"
24
25#include <array>
26#include <deque>
27#include <functional>
28
29#include <dune/common/deprecated.hh>
30#include <dune/common/version.hh>
31#include <dune/grid-glue/common/crossproduct.hh>
32
33namespace Dune {
34
35 namespace GridGlue {
36
40template<typename GV>
41class Codim1Extractor : public Extractor<GV,1>
42{
43public:
44
45 /* E X P O R T E D T Y P E S A N D C O N S T A N T S */
46
47 using Extractor<GV,1>::dimworld;
48 using Extractor<GV,1>::dim;
49 using Extractor<GV,1>::codim;
50 using Extractor<GV,1>::cube_corners;
51 typedef typename Extractor<GV,1>::IndexType IndexType;
52
54 static constexpr int simplex_corners = dim;
55
56 typedef GV GridView;
57
58 typedef typename GV::Grid::ctype ctype;
59 typedef Dune::FieldVector<ctype, dimworld> Coords;
60
61 typedef typename GV::Traits::template Codim<dim>::Entity Vertex;
62 typedef typename GV::Traits::template Codim<0>::Entity Element;
63 typedef std::function<bool(const Element&, unsigned int subentity)> Predicate;
64
65 // import typedefs from base class
66 typedef typename Extractor<GV,1>::SubEntityInfo SubEntityInfo;
67 typedef typename Extractor<GV,1>::ElementInfo ElementInfo;
68 typedef typename Extractor<GV,1>::VertexInfo VertexInfo;
69 typedef typename Extractor<GV,1>::CoordinateInfo CoordinateInfo;
70 typedef typename Extractor<GV,1>::VertexInfoMap VertexInfoMap;
71
72public:
73
74 /* C O N S T R U C T O R S A N D D E S T R U C T O R S */
75
81 Codim1Extractor(const GV& gv, const Predicate& predicate)
82 : Extractor<GV,1>(gv)
83 {
84 std::cout << "This is Codim1Extractor on a <" << dim
85 << "," << dimworld << "> grid!"
86 << std::endl;
87 update(predicate);
88 }
89
90private:
91
105 void update(const Predicate& predicate);
106
107};
108
109
110template<typename GV>
111void Codim1Extractor<GV>::update(const Predicate& predicate)
112{
113 // free everything there is in this object
114 this->clear();
115
116 // In this first pass iterate over all entities of codim 0.
117 // For each codim 1 intersection check if it is part of the boundary and if so,
118 // get its corner vertices, find resp. store them together with their associated index,
119 // and remember the indices of the boundary faces' corners.
120 {
121 // several counter for consecutive indexing are needed
122 int simplex_index = 0;
123 int vertex_index = 0;
124 IndexType eindex = 0; // suppress warning
125
126 // needed later for insertion into a std::set which only
127 // works with const references
128
129 // a temporary container where newly acquired face
130 // information can be stored at first
131 std::deque<SubEntityInfo> temp_faces;
132
133 // iterate over interior codim 0 elements on the grid
134 for (const auto& elmt : elements(this->gv_, Partitions::interior))
135 {
136 Dune::GeometryType gt = elmt.type();
137
138 // if some face is part of the surface add it!
139 if (elmt.hasBoundaryIntersections())
140 {
141 // add an entry to the element info map, the index will be set properly later,
142 // whereas the number of faces is already known
143 eindex = this->cellMapper_.index(elmt);
144 this->elmtInfo_.emplace(eindex, ElementInfo(simplex_index, elmt, 0));
145
146 // now add the faces in ascending order of their indices
147 // (we are only talking about 1-4 faces here, so O(n^2) is ok!)
148 for (const auto& in : intersections(this->gv_, elmt))
149 {
150 // Stop only at selected boundary faces
151 if (!in.boundary() or !predicate(elmt, in.indexInInside()))
152 continue;
153
154 const auto& refElement = Dune::ReferenceElements<ctype, dim>::general(gt);
155 // get the corner count of this face
156 const int face_corners = refElement.size(in.indexInInside(), 1, dim);
157
158 // now we only have to care about the 3D case, i.e. a triangle face can be
159 // inserted directly whereas a quadrilateral face has to be divided into two triangles
160 switch (face_corners)
161 {
162 case 2 :
163 case 3:
164 {
165 // we have a simplex here
166
167 // register the additional face(s)
168 this->elmtInfo_.at(eindex).faces++;
169
170 // add a new face to the temporary collection
171 temp_faces.emplace_back(eindex, in.indexInInside(),
172 Dune::GeometryTypes::simplex(dim-codim));
173
174 std::vector<FieldVector<ctype,dimworld> > cornerCoords(face_corners);
175
176 // try for each of the faces vertices whether it is already inserted or not
177 for (int i = 0; i < face_corners; ++i)
178 {
179 // get the number of the vertex in the parent element
180 int vertex_number = refElement.subEntity(in.indexInInside(), 1, i, dim);
181
182 // get the vertex pointer and the index from the index set
183 const Vertex vertex = elmt.template subEntity<dim>(vertex_number);
184 cornerCoords[i] = vertex.geometry().corner(0);
185
186 IndexType vindex = this->gv_.indexSet().template index<dim>(vertex);
187
188 // remember the vertex' number in parent element's vertices
189 temp_faces.back().corners[i].num = vertex_number;
190
191 // if the vertex is not yet inserted in the vertex info map
192 // it is a new one -> it will be inserted now!
193 typename VertexInfoMap::iterator vimit = this->vtxInfo_.find(vindex);
194 if (vimit == this->vtxInfo_.end())
195 {
196 // insert into the map
197 this->vtxInfo_.emplace(vindex, VertexInfo(vertex_index, vertex));
198 // remember the vertex as a corner of the current face in temp_faces
199 temp_faces.back().corners[i].idx = vertex_index;
200 // increase the current index
201 vertex_index++;
202 }
203 else
204 {
205 // only insert the index into the simplices array
206 temp_faces.back().corners[i].idx = vimit->second.idx;
207 }
208 }
209
210 // Now we have the correct vertices in the last entries of temp_faces, but they may
211 // have the wrong orientation. We want them to be oriented such that all boundary edges
212 // point in the counterclockwise direction. Therefore, we check the orientation of the
213 // new face and possibly switch the two vertices.
214 FieldVector<ctype,dimworld> realNormal = in.centerUnitOuterNormal();
215
216 // Compute segment normal
217 FieldVector<ctype,dimworld> reconstructedNormal;
218 if (dim==2) // boundary face is a line segment
219 {
220 reconstructedNormal[0] = cornerCoords[1][1] - cornerCoords[0][1];
221 reconstructedNormal[1] = cornerCoords[0][0] - cornerCoords[1][0];
222 } else { // boundary face is a triangle
223 FieldVector<ctype,dimworld> segment1 = cornerCoords[1] - cornerCoords[0];
224 FieldVector<ctype,dimworld> segment2 = cornerCoords[2] - cornerCoords[0];
225 reconstructedNormal = crossProduct(segment1, segment2);
226 }
227 reconstructedNormal /= reconstructedNormal.two_norm();
228
229 if (realNormal * reconstructedNormal < 0.0)
230 std::swap(temp_faces.back().corners[0], temp_faces.back().corners[1]);
231
232 // now increase the current face index
233 simplex_index++;
234 break;
235 }
236 case 4 :
237 {
238 assert(dim == 3 && cube_corners == 4);
239 // we have a quadrilateral here
240 std::array<unsigned int, 4> vertex_indices;
241 std::array<unsigned int, 4> vertex_numbers;
242
243 // register the additional face(s) (2 simplices)
244 this->elmtInfo_.at(eindex).faces += 2;
245
246 std::array<FieldVector<ctype,dimworld>, 4> cornerCoords;
247
248 // get the vertex pointers for the quadrilateral's corner vertices
249 // and try for each of them whether it is already inserted or not
250 for (int i = 0; i < cube_corners; ++i)
251 {
252 // get the number of the vertex in the parent element
253 vertex_numbers[i] = refElement.subEntity(in.indexInInside(), 1, i, dim);
254
255 // get the vertex pointer and the index from the index set
256 const Vertex vertex = elmt.template subEntity<dim>(vertex_numbers[i]);
257 cornerCoords[i] = vertex.geometry().corner(0);
258
259 IndexType vindex = this->gv_.indexSet().template index<dim>(vertex);
260
261 // if the vertex is not yet inserted in the vertex info map
262 // it is a new one -> it will be inserted now!
263 typename VertexInfoMap::iterator vimit = this->vtxInfo_.find(vindex);
264 if (vimit == this->vtxInfo_.end())
265 {
266 // insert into the map
267 this->vtxInfo_.emplace(vindex, VertexInfo(vertex_index, vertex));
268 // remember this vertex' index
269 vertex_indices[i] = vertex_index;
270 // increase the current index
271 vertex_index++;
272 }
273 else
274 {
275 // only remember the vertex' index
276 vertex_indices[i] = vimit->second.idx;
277 }
278 }
279
280 // now introduce the two triangles subdividing the quadrilateral
281 // ATTENTION: the order of vertices given by "orientedSubface" corresponds to the order
282 // of a Dune quadrilateral, i.e. the triangles are given by 0 1 2 and 3 2 1
283
284 // add a new face to the temporary collection for the first tri
285 temp_faces.emplace_back(eindex, in.indexInInside(),
286 Dune::GeometryTypes::simplex(dim-codim));
287 temp_faces.back().corners[0].idx = vertex_indices[0];
288 temp_faces.back().corners[1].idx = vertex_indices[1];
289 temp_faces.back().corners[2].idx = vertex_indices[2];
290 // remember the vertices' numbers in parent element's vertices
291 temp_faces.back().corners[0].num = vertex_numbers[0];
292 temp_faces.back().corners[1].num = vertex_numbers[1];
293 temp_faces.back().corners[2].num = vertex_numbers[2];
294
295 // Now we have the correct vertices in the last entries of temp_faces, but they may
296 // have the wrong orientation. We want the triangle vertices on counterclockwise order,
297 // when viewed from the outside of the grid. Therefore, we check the orientation of the
298 // new face and possibly switch two vertices.
299 FieldVector<ctype,dimworld> realNormal = in.centerUnitOuterNormal();
300
301 // Compute segment normal
302 FieldVector<ctype,dimworld> reconstructedNormal = crossProduct(cornerCoords[1] - cornerCoords[0],
303 cornerCoords[2] - cornerCoords[0]);
304 reconstructedNormal /= reconstructedNormal.two_norm();
305
306 if (realNormal * reconstructedNormal < 0.0)
307 std::swap(temp_faces.back().corners[0], temp_faces.back().corners[1]);
308
309
310 // add a new face to the temporary collection for the second tri
311 temp_faces.emplace_back(eindex, in.indexInInside(),
312 Dune::GeometryTypes::simplex(dim-codim));
313 temp_faces.back().corners[0].idx = vertex_indices[3];
314 temp_faces.back().corners[1].idx = vertex_indices[2];
315 temp_faces.back().corners[2].idx = vertex_indices[1];
316 // remember the vertices' numbers in parent element's vertices
317 temp_faces.back().corners[0].num = vertex_numbers[3];
318 temp_faces.back().corners[1].num = vertex_numbers[2];
319 temp_faces.back().corners[2].num = vertex_numbers[1];
320
321 // Now we have the correct vertices in the last entries of temp_faces, but they may
322 // have the wrong orientation. We want the triangle vertices on counterclockwise order,
323 // when viewed from the outside of the grid. Therefore, we check the orientation of the
324 // new face and possibly switch two vertices.
325 // Compute segment normal
326 reconstructedNormal = crossProduct(cornerCoords[2] - cornerCoords[3],
327 cornerCoords[1] - cornerCoords[3]);
328 reconstructedNormal /= reconstructedNormal.two_norm();
329
330 if (realNormal * reconstructedNormal < 0.0)
331 std::swap(temp_faces.back().corners[0], temp_faces.back().corners[1]);
332
333 simplex_index+=2;
334 break;
335 }
336 default :
337 DUNE_THROW(Dune::NotImplemented, "the extractor does only work for triangle and quadrilateral faces (" << face_corners << " corners)");
338 break;
339 }
340 } // end loop over found surface parts
341 }
342 } // end loop over elements
343
344 std::cout << "added " << simplex_index << " subfaces\n";
345
346 // allocate the array for the face specific information...
347 this->subEntities_.resize(simplex_index);
348 // ...and fill in the data from the temporary containers
349 copy(temp_faces.begin(), temp_faces.end(), this->subEntities_.begin());
350 }
351
352
353 // now first write the array with the coordinates...
354 this->coords_.resize(this->vtxInfo_.size());
355 for (const auto& vinfo : this->vtxInfo_)
356 {
357 // get a pointer to the associated info object
358 CoordinateInfo* current = &this->coords_[vinfo.second.idx];
359 // store this coordinates index // NEEDED?
360 current->index = vinfo.second.idx;
361 // store the vertex' index for the index2vertex mapping
362 current->vtxindex = vinfo.first;
363 // store the vertex' coordinates under the associated index
364 // in the coordinates array
365 const auto vtx = this->grid().entity(vinfo.second.p);
366 current->coord = vtx.geometry().corner(0);
367 }
368
369}
370
371} // namespace GridGlue
372
373} // namespace Dune
374
375#endif // DUNE_GRIDGLUE_EXTRACTORS_CODIM1EXTRACTOR_HH
Definition: codim1extractor.hh:42
Codim1Extractor(const GV &gv, const Predicate &predicate)
Constructor.
Definition: codim1extractor.hh:81
static constexpr int simplex_corners
compile time number of corners of surface simplices
Definition: codim1extractor.hh:54
Provides codimension-independent methods for grid extraction.
Definition: extractor.hh:46
extractor base class
Creative Commons License   |  Legal Statements / Impressum  |  Hosted by TU Dresden & Uni Heidelberg  |  generated with Hugo v0.111.3 (Apr 3, 22:46, 2025)