20#ifndef DUNE_GRIDGLUE_EXTRACTORS_CODIM1EXTRACTOR_HH
21#define DUNE_GRIDGLUE_EXTRACTORS_CODIM1EXTRACTOR_HH
29#include <dune/common/deprecated.hh>
30#include <dune/common/version.hh>
31#include <dune/grid-glue/common/crossproduct.hh>
58 typedef typename GV::Grid::ctype ctype;
59 typedef Dune::FieldVector<ctype, dimworld> Coords;
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;
84 std::cout <<
"This is Codim1Extractor on a <" << dim
85 <<
"," << dimworld <<
"> grid!"
105 void update(
const Predicate& predicate);
111void Codim1Extractor<GV>::update(
const Predicate& predicate)
122 int simplex_index = 0;
123 int vertex_index = 0;
124 IndexType eindex = 0;
131 std::deque<SubEntityInfo> temp_faces;
134 for (
const auto& elmt : elements(this->gv_, Partitions::interior))
136 Dune::GeometryType gt = elmt.type();
139 if (elmt.hasBoundaryIntersections())
143 eindex = this->cellMapper_.index(elmt);
144 this->elmtInfo_.emplace(eindex, ElementInfo(simplex_index, elmt, 0));
148 for (
const auto& in : intersections(this->gv_, elmt))
151 if (!in.boundary() or !predicate(elmt, in.indexInInside()))
154 const auto& refElement = Dune::ReferenceElements<ctype, dim>::general(gt);
156 const int face_corners = refElement.size(in.indexInInside(), 1, dim);
160 switch (face_corners)
168 this->elmtInfo_.at(eindex).faces++;
171 temp_faces.emplace_back(eindex, in.indexInInside(),
172 Dune::GeometryTypes::simplex(dim-codim));
174 std::vector<FieldVector<ctype,dimworld> > cornerCoords(face_corners);
177 for (
int i = 0; i < face_corners; ++i)
180 int vertex_number = refElement.subEntity(in.indexInInside(), 1, i, dim);
183 const Vertex vertex = elmt.template subEntity<dim>(vertex_number);
184 cornerCoords[i] = vertex.geometry().corner(0);
186 IndexType vindex = this->gv_.indexSet().template index<dim>(vertex);
189 temp_faces.back().corners[i].num = vertex_number;
193 typename VertexInfoMap::iterator vimit = this->vtxInfo_.find(vindex);
194 if (vimit == this->vtxInfo_.end())
197 this->vtxInfo_.emplace(vindex, VertexInfo(vertex_index, vertex));
199 temp_faces.back().corners[i].idx = vertex_index;
206 temp_faces.back().corners[i].idx = vimit->second.idx;
214 FieldVector<ctype,dimworld> realNormal = in.centerUnitOuterNormal();
217 FieldVector<ctype,dimworld> reconstructedNormal;
220 reconstructedNormal[0] = cornerCoords[1][1] - cornerCoords[0][1];
221 reconstructedNormal[1] = cornerCoords[0][0] - cornerCoords[1][0];
223 FieldVector<ctype,dimworld> segment1 = cornerCoords[1] - cornerCoords[0];
224 FieldVector<ctype,dimworld> segment2 = cornerCoords[2] - cornerCoords[0];
225 reconstructedNormal = crossProduct(segment1, segment2);
227 reconstructedNormal /= reconstructedNormal.two_norm();
229 if (realNormal * reconstructedNormal < 0.0)
230 std::swap(temp_faces.back().corners[0], temp_faces.back().corners[1]);
238 assert(dim == 3 && cube_corners == 4);
240 std::array<unsigned int, 4> vertex_indices;
241 std::array<unsigned int, 4> vertex_numbers;
244 this->elmtInfo_.at(eindex).faces += 2;
246 std::array<FieldVector<ctype,dimworld>, 4> cornerCoords;
250 for (
int i = 0; i < cube_corners; ++i)
253 vertex_numbers[i] = refElement.subEntity(in.indexInInside(), 1, i, dim);
256 const Vertex vertex = elmt.template subEntity<dim>(vertex_numbers[i]);
257 cornerCoords[i] = vertex.geometry().corner(0);
259 IndexType vindex = this->gv_.indexSet().template index<dim>(vertex);
263 typename VertexInfoMap::iterator vimit = this->vtxInfo_.find(vindex);
264 if (vimit == this->vtxInfo_.end())
267 this->vtxInfo_.emplace(vindex, VertexInfo(vertex_index, vertex));
269 vertex_indices[i] = vertex_index;
276 vertex_indices[i] = vimit->second.idx;
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];
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];
299 FieldVector<ctype,dimworld> realNormal = in.centerUnitOuterNormal();
302 FieldVector<ctype,dimworld> reconstructedNormal = crossProduct(cornerCoords[1] - cornerCoords[0],
303 cornerCoords[2] - cornerCoords[0]);
304 reconstructedNormal /= reconstructedNormal.two_norm();
306 if (realNormal * reconstructedNormal < 0.0)
307 std::swap(temp_faces.back().corners[0], temp_faces.back().corners[1]);
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];
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];
326 reconstructedNormal = crossProduct(cornerCoords[2] - cornerCoords[3],
327 cornerCoords[1] - cornerCoords[3]);
328 reconstructedNormal /= reconstructedNormal.two_norm();
330 if (realNormal * reconstructedNormal < 0.0)
331 std::swap(temp_faces.back().corners[0], temp_faces.back().corners[1]);
337 DUNE_THROW(Dune::NotImplemented,
"the extractor does only work for triangle and quadrilateral faces (" << face_corners <<
" corners)");
344 std::cout <<
"added " << simplex_index <<
" subfaces\n";
347 this->subEntities_.resize(simplex_index);
349 copy(temp_faces.begin(), temp_faces.end(), this->subEntities_.begin());
354 this->coords_.resize(this->vtxInfo_.size());
355 for (
const auto& vinfo : this->vtxInfo_)
358 CoordinateInfo* current = &this->coords_[vinfo.second.idx];
360 current->index = vinfo.second.idx;
362 current->vtxindex = vinfo.first;
365 const auto vtx = this->grid().entity(vinfo.second.p);
366 current->coord = vtx.geometry().corner(0);