Loading [MathJax]/extensions/tex2jax.js

DUNE PDELab (unstable)

printgrid.hh
1// SPDX-FileCopyrightText: Copyright © DUNE Project contributors, see file LICENSE.md in module root
2// SPDX-License-Identifier: LicenseRef-GPL-2.0-only-with-DUNE-exception
3// -*- tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 2 -*-
4// vi: set et ts=4 sw=2 sts=2:
5#ifndef DUNE_PRINTGRID_HH
6#define DUNE_PRINTGRID_HH
7
8#include <fstream>
9#include <string>
10
14
15namespace Dune {
16
17 namespace {
18
19 template<int dim>
20 struct ElementDataLayout
21 {
23 {
24 return gt.dim()==dim;
25 }
26 };
27
28 template<int dim>
29 struct NodeDataLayout
30 {
32 {
33 return gt.dim()==0;
34 }
35 };
36
37 // Move a point closer to basegeo's center by factor scale (used for drawing relative to the element)
38 template <typename B, typename C>
39 C centrify (const B& basegeo, const C& coords, const double scale) {
40 C ret = coords;
41 ret -= basegeo.center();
42 ret *= scale;
43 ret += basegeo.center();
44 return ret;
45 }
46
47 // Add a line to the plotfile from p1 to p2
48 template <typename Coord>
49 void draw_line (std::ofstream &plotfile, const Coord &p1, const Coord &p2, std::string options) {
50 plotfile << "set object poly from ";
51 plotfile << p1[0] << "," << p1[1] << " to ";
52 plotfile << p2[0] << "," << p2[1] << " to ";
53 plotfile << p1[0] << "," << p1[1];
54 plotfile << " " << options << std::endl;
55 }
56
57 }
58
71 template <typename GridType>
72 void printGrid (const GridType& grid, std::string output_file,
73 int size = 2000, bool execute_plot = true, bool png = true, bool local_corner_indices = true,
74 bool local_intersection_indices = true, bool outer_normals = true)
75 {
76 // Create output file
77 std::string plot_file_name = output_file + ".gnuplot";
78 std::ofstream plotfile (plot_file_name, std::ios::out | std::ios::trunc);
79 if (!plotfile.is_open()) {
80 DUNE_THROW(Dune::IOError, "Could not create plot file " << output_file << "!");
81 return;
82 }
83
84 // Basic plot settings
85 plotfile << "set size ratio -1" << std::endl;
86 if (png) {
87 plotfile << "set terminal png size " << size << "," << size << std::endl;
88 plotfile << "set output '" << output_file << ".png'" << std::endl;
89 } else {
90 plotfile << "set terminal svg size " << size << "," << size << " enhanced background rgb 'white'" << std::endl;
91 plotfile << "set output '" << output_file << ".svg'" << std::endl;
92 }
93
94 // Get GridView
95 typedef typename GridType::LeafGridView GV;
96 const GV gv = grid.leafGridView();
97
98 // Create mappers used to retrieve indices
100 const Mapper elementmapper(gv, mcmgElementLayout());
101 const Mapper nodemapper(gv, mcmgVertexLayout());
102
103 // Create iterators
104 typedef typename GV::template Codim<0 >::Iterator LeafIterator;
105 typedef typename GV::IntersectionIterator IntersectionIterator;
106
107 LeafIterator it = gv.template begin<0>();
108
109 // Will contain min/max coordinates. Needed for scaling of the plot
110 Dune::FieldVector<typename GridType::ctype,2> max_coord (it->geometry().center()), min_coord (max_coord);
111
112 // Iterate over elements
113 for (; it != gv.template end<0>(); ++it) {
114
115 const auto& entity = *it;
116 auto geo = entity.geometry();
117
118 // Plot element index
119 int element_id = elementmapper.index(entity);
120 plotfile << "set label at " << geo.center()[0] << "," << geo.center()[1] << " '"
121 << element_id << "' center" << std::endl;
122
123 for (int i = 0; i < geo.corners(); ++i) {
124 // Plot corner indices
125 const int globalNodeNumber1 = nodemapper.subIndex(entity, i, 2);
126 auto labelpos = centrify (geo, geo.corner(i), 0.7);
127 plotfile << "set label at " << labelpos[0] << "," << labelpos[1] << " '" << globalNodeNumber1;
128 if (local_corner_indices)
129 plotfile << "(" << i << ")";
130 plotfile << "' center" << std::endl;
131
132 // Adapt min / max coordinates
133 for (int dim = 0; dim < 2; ++dim) {
134 if (geo.corner(i)[dim] < min_coord[dim])
135 min_coord[dim] = geo.corner(i)[dim];
136 else if (geo.corner(i)[dim] > max_coord[dim])
137 max_coord[dim] = geo.corner(i)[dim];
138 }
139 }
140
141 // Iterate over intersections
142 for (IntersectionIterator is = gv.ibegin(entity); is != gv.iend(entity); ++is) {
143
144 const auto& intersection = *is;
145 auto igeo = intersection.geometry();
146
147 // Draw intersection line
148 draw_line (plotfile, igeo.corner(0), igeo.corner(1), "fs empty border 1");
149
150 // Plot local intersection index
151 if (local_intersection_indices) {
152 auto label_pos = centrify (geo, igeo.center(), 0.8);
153 plotfile << "set label at " << label_pos[0] << "," << label_pos[1]
154 << " '" << intersection.indexInInside() << "' center" << std::endl;
155 }
156
157 // Plot outer normal
158 if (outer_normals) {
159 auto intersection_pos = igeo.center();
160 auto normal = intersection.centerUnitOuterNormal();
161 normal *= 0.15 * igeo.volume();
162 auto normal_end = intersection_pos + normal;
163 plotfile << "set arrow from " << intersection_pos[0] << "," << intersection_pos[1]
164 << " to " << normal_end[0] << "," << normal_end[1] << " lt rgb \"gray\"" << std::endl;
165 }
166
167 // Get corners for inner intersection representation
168 auto inner_corner1 = centrify (geo, igeo.corner(0), 0.5);
169 auto inner_corner2 = centrify (geo, igeo.corner(1), 0.5);
170
171 // Thick line in case of boundary()
172 if (intersection.boundary())
173 draw_line (plotfile, inner_corner1, inner_corner2, "fs empty border 3 lw 4");
174
175 // Thin line with color according to neighbor()
176 if (intersection.neighbor())
177 draw_line (plotfile, inner_corner1, inner_corner2, "fs empty border 2");
178 else
179 draw_line (plotfile, inner_corner1, inner_corner2, "fs empty border 1");
180 }
181
182 }
183
184 // Finish plot, pass extend of the grid
185 Dune::FieldVector<typename GridType::ctype,2> extend (max_coord - min_coord);
186
187 extend *= 0.2;
188 min_coord -= extend;
189 max_coord += extend;
190 plotfile << "plot [" << min_coord[0] << ":" << max_coord[0] << "] [" << min_coord[1]
191 << ":" << max_coord[1] << "] NaN notitle" << std::endl;
192 plotfile.close();
193
194 if (execute_plot) {
195 std::string cmd = "gnuplot -p '" + plot_file_name + "'";
196 if (std::system (cmd.c_str()) != 0)
197 DUNE_THROW(Dune::Exception,"Error running GNUPlot: " << cmd);
198 }
199 }
200
201
215 template <typename GridType>
216 void printGrid (const GridType& grid, const Dune::MPIHelper& helper, std::string output_file = "printgrid",
217 int size = 2000, bool execute_plot = true, bool png = true, bool local_corner_indices = true,
218 bool local_intersection_indices = true, bool outer_normals = true)
219 {
220 printGrid(grid, output_file + "_" + std::to_string(helper.rank()), size, execute_plot, png, local_corner_indices, local_intersection_indices, outer_normals);
221 }
222}
223
224#endif // #ifndef DUNE_PRINTGRID_HH
Base class for Dune-Exceptions.
Definition: exceptions.hh:96
vector space out of a tensor product of fields.
Definition: fvector.hh:92
Unique label for each type of entities that can occur in DUNE grids.
Definition: type.hh:114
Default exception class for I/O errors.
Definition: exceptions.hh:323
Mesh entities of codimension 0 ("elements") allow to visit all intersections with "neighboring" eleme...
Definition: intersectioniterator.hh:83
A real mpi helper.
Definition: mpihelper.hh:181
int rank() const
return rank of process
Definition: mpihelper.hh:294
Mapper interface.
Definition: mapper.hh:110
Index index(const EntityType &e) const
Map entity to array index.
Definition: mapper.hh:122
Index subIndex(const typename G::Traits::template Codim< 0 >::Entity &e, int i, unsigned int codim) const
Map subentity i of codim cc of a codim 0 entity to array index.
Definition: mapper.hh:136
Implementation class for a multiple codim and multiple geometry type mapper.
Definition: mcmgmapper.hh:129
A few common exception classes.
#define DUNE_THROW(E,...)
Definition: exceptions.hh:312
I trunc(const T &val, typename EpsilonType< T >::Type epsilon)
truncate using epsilon
Definition: float_cmp.cc:407
bool gt(const T &first, const T &second, typename EpsilonType< T >::Type epsilon)
test if first greater than second
Definition: float_cmp.cc:158
MCMGLayout mcmgElementLayout()
layout for elements (codim-0 entities)
Definition: mcmgmapper.hh:97
MCMGLayout mcmgVertexLayout()
layout for vertices (dim-0 entities)
Definition: mcmgmapper.hh:107
Mapper for multiple codim and multiple geometry types.
Helpers for dealing with MPI.
Dune namespace.
Definition: alignedallocator.hh:13
constexpr std::integral_constant< std::size_t, sizeof...(II)> size(std::integer_sequence< T, II... >)
Return the size of the sequence.
Definition: integersequence.hh:75
constexpr std::bool_constant<((II==value)||...)> contains(std::integer_sequence< T, II... >, std::integral_constant< T, value >)
Checks whether or not a given sequence contains a value.
Definition: integersequence.hh:137
void printGrid(const GridType &grid, std::string output_file, int size=2000, bool execute_plot=true, bool png=true, bool local_corner_indices=true, bool local_intersection_indices=true, bool outer_normals=true)
Print a grid as a gnuplot for testing and development.
Definition: printgrid.hh:72
Static tag representing a codimension.
Definition: dimension.hh:24
Creative Commons License   |  Legal Statements / Impressum  |  Hosted by TU Dresden & Uni Heidelberg  |  generated with Hugo v0.111.3 (Apr 2, 23:03, 2025)