This page was generated from unit-4.3-manualmesh/manualmeshing.ipynb.
4.3 Working with meshes¶
One-dimensional meshes¶
Meshes in one-dimension can be constructed using the netgen.meshing
module. We just have to add segments (Element1D
) and the boundaries (Element0D
).
[1]:
# import netgen.gui
from netgen.meshing import *
First we create a new Mesh
and set the spatial dimension to 1.
[2]:
m = Mesh(dim=1)
Then we define and add the MeshPoint
’s to the mesh. The function m.Add
returns PointId
’s which we store in an array to be able to construct the segments in the next step.
[3]:
N = 10
pnums = []
for i in range(0, N+1):
pnums.append (m.Add (MeshPoint (Pnt(i/N, 0, 0))))
type(pnums[0])
[3]:
netgen.libngpy._meshing.PointId
Now we can loop over the array with the PointId
’s and add one-dimensional elements to the mesh. Further we can set the material for our domain.
[4]:
idx = m.AddRegion("material", dim=1)
for i in range(0,N):
m.Add (Element1D ([pnums[i],pnums[i+1]], index=idx))
Finally we have to add the boundary elements and set boundary conditions.
[5]:
idx_left = m.AddRegion("left", dim=0)
idx_right = m.AddRegion("right", dim=0)
m.Add (Element0D (pnums[0], index=idx_left))
m.Add (Element0D (pnums[N], index=idx_right))
[5]:
1
To be able to visualize one-dimensional meshes and solution activate Show edges
in the menu View > Viewing options > Mesh
.
[6]:
import ngsolve
mesh = ngsolve.Mesh(m)
Two-dimensional meshes¶
As example we mesh a unit square [0,1]x[0,1] using quadrilaterals.
[7]:
from netgen.geom2d import unit_square
We create an empty mesh
[8]:
ngmesh = Mesh(dim=2)
and add all the MeshPoint
’s we will need for the final mesh. Similar to the one-dimensional mesh we store the PointId
’s in the pnums
array.
[9]:
N=5
pnums = []
for i in range(N + 1):
for j in range(N + 1):
pnums.append(ngmesh.Add(MeshPoint(Pnt(i / N, j / N, 0))))
Next, we create a region, and add the quadrilaterals to the mesh.
[10]:
idx_dom = ngmesh.AddRegion("mat", dim=2)
for j in range(N):
for i in range(N):
ngmesh.Add(Element2D(idx_dom, [pnums[i + j * (N + 1)],
pnums[i + (j + 1) * (N + 1)],
pnums[i + 1 + (j + 1) * (N + 1)],
pnums[i + 1 + j * (N + 1)]]))
Finally we have to add boundary elements and set boundary conditions.
[11]:
# horizontal boundaries
for i in range(N):
ngmesh.Add(Element1D([pnums[N + i * (N + 1)],
pnums[N + (i + 1) * (N + 1)]], index=1))
ngmesh.Add(Element1D([pnums[0 + i * (N + 1)], pnums[0 + (i + 1) * (N + 1)]], index=1))
# vertical boundaries
for i in range(N):
ngmesh.Add(Element1D([pnums[i], pnums[i + 1]], index=2))
ngmesh.Add(Element1D([pnums[i + N * (N + 1)], pnums[i + 1 + N * (N + 1)]], index=2))
[12]:
from ngsolve.webgui import Draw
mesh = ngsolve.Mesh(ngmesh)
Draw(mesh)
[12]:
BaseWebGuiScene
Merge three-dimensional meshes¶
In the following example we will merge two surface meshes and generate a unified volume mesh.
[13]:
from netgen.meshing import *
from netgen.csg import *
from ngsolve import ngsglobals
ngsglobals.msg_level = 2
As starting point we create two geometries and mesh them.
[14]:
# generate brick and mesh it
geo1 = CSGeometry()
geo1.Add (OrthoBrick( Pnt(0,0,0), Pnt(1,1,1) ))
m1 = geo1.GenerateMesh (maxh=0.1)
# m1.Refine()
# generate sphere and mesh it
geo2 = CSGeometry()
geo2.Add (Sphere (Pnt(0.5,0.5,0.5), 0.1))
m2 = geo2.GenerateMesh (maxh=0.05)
m2.Refine()
# m2.Refine()
Start Findpoints
Analyze spec points
Find edges
Start Findpoints
Analyze spec points
Find edges
Start Findpoints
Analyze spec points
Find edges
Surface 1 / 6
Optimize Surface
Surface 2 / 6
Optimize Surface
Surface 3 / 6
Optimize Surface
Surface 4 / 6
Optimize Surface
Surface 5 / 6
Optimize Surface
Surface 6 / 6
Optimize Surface
Delaunay meshing
Remove Illegal Elements
Volume Optimization
Start Findpoints
Analyze spec points
Find edges
Start Findpoints
Analyze spec points
Find edges
Start Findpoints
Analyze spec points
Find edges
Surface 1 / 1
Optimize Surface
Delaunay meshing
Remove Illegal Elements
Volume Optimization
Now we start the merging process. Therefore we create an empty mesh and add a FaceDescriptor
for each of the surfaces.
[15]:
# create an empty mesh
ngmesh = Mesh()
# a face-descriptor stores properties associated with a set of surface elements
# bc .. boundary condition marker,
# domin/domout .. domain-number in front/back of surface elements (0 = void),
# surfnr .. number of the surface described by the face-descriptor
fd_outside = ngmesh.Add (FaceDescriptor(bc=1,domin=1,surfnr=1))
fd_inside = ngmesh.Add (FaceDescriptor(bc=2,domin=2,domout=1,surfnr=2))
Since the surface elements stay the same in the merged mesh, we copy the points on the surface and the surface elements to the new mesh.
[16]:
# copy all boundary points from first mesh to new mesh.
# pmap1 maps point-numbers from old to new mesh
pmap1 = { }
for e in m1.Elements2D():
for v in e.vertices:
if (v not in pmap1):
pmap1[v] = ngmesh.Add (m1[v])
# copy surface elements from first mesh to new mesh
# we have to map point-numbers:
for e in m1.Elements2D():
ngmesh.Add (Element2D (fd_outside, [pmap1[v] for v in e.vertices]))
# same for the second mesh:
pmap2 = { }
for e in m2.Elements2D():
for v in e.vertices:
if (v not in pmap2):
pmap2[v] = ngmesh.Add (m2[v])
for e in m2.Elements2D():
ngmesh.Add (Element2D (fd_inside, [pmap2[v] for v in e.vertices]))
Finally we have to generate the new volume mesh.
[17]:
ngmesh.GenerateVolumeMesh()
import ngsolve
mesh = ngsolve.Mesh(ngmesh)
Draw(mesh);
Delaunay meshing
Delaunay meshing
Volume Optimization
[ ]: