This page was generated from unit-8.6-mlset_basic/mlset_basic.ipynb.
8.6 Integration over domains described by multiple level sets¶
The ngsxfem
add-on library has the ability to integrate over domains which are defined by muliple level set functions.
We recall that previously, our domain of interest was described by a single level set function, i.e., for a continuous level set function \(\phi\), we considered the domain \(\Omega_{-} := \{x : \phi(x) < 0 \}\).
Now, let us consider the set \(\{\phi_i \}\) of level set functions. With this we define the domain of interest by the intersection \(\Omega := \bigcap_i \{\phi_i < 0\}\).
The libraries are imported as usual:
[1]:
# Import geometry features, NGSolve and xfem
from netgen.geom2d import SplineGeometry
from ngsolve import *
from xfem import *
# Visualisation
from ngsolve.webgui import *
DrawDC = MakeDiscontinuousDraw(Draw)
importing ngsxfem-2.1.2405.dev0
For a better interactive experience set the following parameter to 2 (1 leads to coarse grids). It is set to 1 for the documentation to keep file sizes at a managable level:
[2]:
interactive = 1
We generate a background mesh
[3]:
geo = SplineGeometry()
geo.AddRectangle([-1, -0.5], [1, 1.5], bc=1)
mesh = Mesh (geo.GenerateMesh(maxh=0.3, quad_dominated=False))
Draw(mesh, 0, 0)
[3]:
BaseWebGuiScene
Integrating over regions of co-dimension 0¶
As an example, we shall consider an isosceles triangle, described by the level set functions \(\phi_1 = y - 1\), \(\phi_2 = 2x - y\) and \(\phi_3 = -2x - y\).
As in the single level set setting, ngsxfem
can only handle piecewise linear level set functions. We thefore interpolate each smooth level set into an P1 GridFunction \(\phi^\ast_i \in \mathcal{P}^1(T), T \subset \Omega\). In order to use these more easily later on, we store them in a tuple.
[4]:
level_sets = (y-1, 2*x-y, -2*x-y)
nr_ls = len(level_sets)
level_sets_p1 = tuple(GridFunction(H1(mesh, order=1)) for i in range(nr_ls))
for i, lset_p1 in enumerate(level_sets_p1):
InterpolateToP1(level_sets[i], lset_p1)
DrawDC(lset_p1, -3.5, 2.5, mesh, "lset_p1_{}".format(i))
The domain described where all three level-sets are negavive is then
[5]:
Draw(CoefficientFunction((level_sets_p1[0], level_sets_p1[1], level_sets_p1[2], 0)),
mesh, "NEG-NEG-NEG", min=-1, max=1,
eval_function="value.x<0.0?(value.y<0.0?(value.z<0.0?1.0:-1.0):-1.0):-1.0")
[5]:
BaseWebGuiScene
To integrate, we still use the function Integrate
. In the multiple level set context, the input to the dCut(levelset, domain_type, order)
is very similar to the single level set setting. Here, * the levelset
argument must be a tuple
of the P1-GridFunction level set functions and * the domain_type
can be a * tuple({NEG, POS, IF})
of the same length as the levelset tuple which describes the domain whith respect each level set function * list(tuple)
of such tuples
when the domain is described via (the union of) multiple regions * DomainTypeArray
(see xfem.mlset
convininece layer below). * the remaining (keyword) arguments remain as in the single level set case
[6]:
area = Integrate(CoefficientFunction(1) * dCut(level_sets_p1, (NEG, NEG, NEG), order=0), mesh=mesh)
error = abs(area - 0.5)
print("Result of the integration: {}".format(area))
print("Error of the integration: {:5.3e}".format(error))
Result of the integration: 0.5
Error of the integration: 0.000e+00
Since the smooth level set functions are linear, the integration is exact up to mashine precision.
Integrating over regions of co-dimension > 0¶
We can also integrate over subdomains of higher codimensions. For example, the lenghths of the sides of the triangle can be computed as follows:
[7]:
len_top = Integrate(CoefficientFunction(1) * dCut(level_sets_p1, (IF, NEG, NEG), order=0), mesh)
print("Top side-length of the triangle");
print("Result of the integration: {}".format(len_top))
print("Error of the integration: {:5.3e}".format(abs(len_top - 1)))
len_right = Integrate(CoefficientFunction(1) * dCut(level_sets_p1, (NEG, IF, NEG), order=0), mesh)
print("\nRight side-length of the triangle")
print("Result of the integration: {}".format(len_right))
print("Error of the integration: {:5.3e}".format(abs(len_right - sqrt(5)/2)))
len_left = Integrate(CoefficientFunction(1) * dCut(level_sets_p1, (NEG, NEG, IF), order=0), mesh)
print("\nLeft side-length of the triangle")
print("Result of the integration: {}".format(len_left))
print("Error of the integration: {:5.3e}".format(abs(len_left - sqrt(5)/2)))
Top side-length of the triangle
Result of the integration: 1.0
Error of the integration: 0.000e+00
Right side-length of the triangle
Result of the integration: 1.1180339887498945
Error of the integration: 4.441e-16
Left side-length of the triangle
Result of the integration: 1.1180339887498947
Error of the integration: 2.220e-16
To compute the perimeter of the triangle, we can pass a list containing the tuples describing each side to Integrate
via the domain_type
entry in the levelset_domain
dictionary:
[8]:
domain_list = [(IF, NEG, NEG), (NEG, IF, NEG), (NEG, NEG, IF)]
len_together = Integrate(CoefficientFunction(1) * dCut(level_sets_p1, domain_list, order=0), mesh)
print("Perimeter of the triangle")
print("Result of the integration: {}".format(len_together))
print("Error of the integration: {:5.3e}".format(abs(len_together - 1 - sqrt(5))))
Perimeter of the triangle
Result of the integration: 3.2360679774997894
Error of the integration: 4.441e-16
Similarly, we can also perform point evaluations at (codim 2) intersections of level sets:
[9]:
point_val = Integrate((x**2 + y**2) * dCut(level_sets_p1, (IF, IF, NEG), order=0), mesh)
print("Result of the integration: {}".format(point_val))
print("Error of the integration: {:5.3e}".format(abs(point_val - 1.25)))
Result of the integration: 1.25
Error of the integration: 0.000e+00
Convenience Layer: xfem.mlset
¶
In order to work with more complex domains, described by multiple level sets, we provide a convenience layer in the module xfem.mlset
. At its core, this module provides a container class DomainTypeArray
which allows us to perform operations and manipulate domain regions. A full description of this can be cound in the docstring:
[10]:
from xfem.mlset import *
DomainTypeArray?
Working with DomainTypeArray
s¶
A DomainTypeArray
is initialised either with a tuple({NEG,POS,IF,ANY})
or with a list of such tuples. While integration does not work with COMBINED_DOMAIN_TYPES
, the DomainTypeArray
class accepts ANY
as an entry, and expands this internaly to a list of tuples containing only {IF, POS, NEG}
, such that the result can then be used for integration.
[11]:
domain = DomainTypeArray((ANY, NEG, POS))
The list of regions contained in this can then be acessed via the as_list
attribute
[12]:
domain.as_list
[12]:
[(<DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.POS: 1>),
(<DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.POS: 1>)]
and the codimension of the resulting domain can be accesed via the codim
attribute
[13]:
domain.codim
[13]:
0
Geometrical features¶
The main feature of the DomainTypeArray
class is that we can treat it as a geometrical set and thus can perform logical operations with them. Available operators are:
Operator |
Operation |
Remark |
---|---|---|
|
Returns a |
The result keeps the codimension. |
|
Returns a |
Only possible for domains of the same codimension. |
|
Returns a |
Intersections can increase the codimension. |
Note: - For |
and &
, the tuples in each DomainTypeArray
must have the same length. - The operators &=
and |=
are also available.
[14]:
triangle = DomainTypeArray((NEG,NEG,NEG))
outside_triangle = ~triangle
outside_triangle.as_list
[14]:
[(<DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.POS: 1>),
(<DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.NEG: 0>),
(<DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.NEG: 0>),
(<DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.NEG: 0>),
(<DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.POS: 1>),
(<DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.POS: 1>),
(<DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.POS: 1>)]
[15]:
dta1 = DomainTypeArray((POS, NEG)) | DomainTypeArray([(POS,POS),(NEG,POS)])
dta1.as_list
[15]:
[(<DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.NEG: 0>),
(<DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.POS: 1>),
(<DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.POS: 1>)]
[16]:
dta2 = outside_triangle & DomainTypeArray([(POS, NEG, POS), (POS, POS, POS)])
dta2.as_list
[16]:
[(<DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.POS: 1>),
(<DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.POS: 1>)]
[17]:
dta3 = DomainTypeArray((NEG, IF, POS)) & DomainTypeArray((NEG, POS, IF))
dta3.as_list
[17]:
[(<DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.IF: 2>, <DOMAIN_TYPE.IF: 2>)]
Generating the boundary
In order to obtain the boundary of the region described by a DomainTypeArray
we can use the class .Boundary()
method:
[18]:
boundary = triangle.Boundary()
boundary.as_list
[18]:
[(<DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.IF: 2>, <DOMAIN_TYPE.NEG: 0>),
(<DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.IF: 2>),
(<DOMAIN_TYPE.IF: 2>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.NEG: 0>)]
Since each section in a DomainTypeArray
is considered as a separate domain, we can even go one step further and construct the corners:
[19]:
corners = boundary.Boundary()
corners.as_list
[19]:
[(<DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.IF: 2>, <DOMAIN_TYPE.IF: 2>),
(<DOMAIN_TYPE.IF: 2>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.IF: 2>),
(<DOMAIN_TYPE.IF: 2>, <DOMAIN_TYPE.IF: 2>, <DOMAIN_TYPE.NEG: 0>)]
Visualisation Tools¶
To help visualise the regions descibed by a DomainTypeArray for debugging purposes, we can use the .Indicator(lsets)
method for domains of codim=0 and the IndicatorSmoothed(lsets, eps=0.03)
method for domains of codim>0.
These functions return a CoefficientFunction
which has the value 1 inside the region of interest and is 0 elsewhere. For regions of co-dimension>0, the result of IndicatorSmoothed(lsets, eps)
is a CoefficientFunction
with the value 1 in the eps
region around the domain.
For illustration purposes, we shall consider fine meshes here, in order to render the indicator functions accurately within the NGSolve webgui. Alternaltievly, it is possible to visualise the indicator functions on coarse meshes in the netgen gui or in paraview (using vtk outputs). In these cases, a larger number of subdivisions is then necessary to get a resonalbly sharp representation on elements with multiple cuts.
[20]:
if interactive > 1:
nref = 4
else:
nref = 2
for i in range(nref):
mesh.ngmesh.Refine()
mesh=Mesh(mesh.ngmesh)
for i, lset_p1 in enumerate(level_sets_p1):
lset_p1.Update()
InterpolateToP1(level_sets[i], lset_p1)
[21]:
DrawDC(outside_triangle.Indicator(level_sets_p1), -3.5, 2.5, mesh, "outside_indicator")
[21]:
BaseWebGuiScene
Similarly, DomainTypeArray.IndicatorSmoothed(level_sets, eps)
returns a CoefficientFunction which has the value 1 in the eps
-strip around the interface described in DomainTypeArray
. This assumes that each level set function is approximately a distance function around the interface:
[22]:
DrawDC(boundary.IndicatorSmoothed(level_sets_p1, 0.05/interactive), -3.5, 2.5, mesh,
"boundary_indicator")
[22]:
BaseWebGuiScene
[23]:
DrawDC(corners.IndicatorSmoothed(level_sets_p1, 0.06/interactive), -3.5, 2.5, mesh, "boundary_indicator")
[23]:
BaseWebGuiScene
Removing unnecesary regions¶
These operations take place on a pure logical DOMAIN_TYPE
level, without taking ang geometrical information into account.
Let us again consider the above case of a Zalesak Disk consisting of the 5 regions, marked with a \(+\): "
"
[24]:
geo = SplineGeometry()
geo.AddRectangle((-1.1, -1.1), (1.1, 1.1), bc=1)
mesh = Mesh(geo.GenerateMesh(maxh=0.2/interactive))
level_sets = [x * x + y * y - 1, -x - 1 / 3, x - 1 / 3, y - 0.5]
level_sets_p1 = tuple(GridFunction(H1(mesh, order=1))
for i in range(len(level_sets)))
for i, lsetp1 in enumerate(level_sets_p1):
InterpolateToP1(level_sets[i], lsetp1)
DrawDC(lsetp1, -3.5, 2.5, mesh, "lset_p1_" + str(i))
The easiest way to generate this is to use the DomainTypeArray
functionality:
[25]:
z_disc = DomainTypeArray((NEG,ANY,ANY,ANY)) \
& ~DomainTypeArray((NEG,NEG,NEG,NEG))
z_disc.as_list
[25]:
[(<DOMAIN_TYPE.NEG: 0>,
<DOMAIN_TYPE.NEG: 0>,
<DOMAIN_TYPE.NEG: 0>,
<DOMAIN_TYPE.POS: 1>),
(<DOMAIN_TYPE.NEG: 0>,
<DOMAIN_TYPE.NEG: 0>,
<DOMAIN_TYPE.POS: 1>,
<DOMAIN_TYPE.NEG: 0>),
(<DOMAIN_TYPE.NEG: 0>,
<DOMAIN_TYPE.POS: 1>,
<DOMAIN_TYPE.NEG: 0>,
<DOMAIN_TYPE.POS: 1>),
(<DOMAIN_TYPE.NEG: 0>,
<DOMAIN_TYPE.POS: 1>,
<DOMAIN_TYPE.POS: 1>,
<DOMAIN_TYPE.POS: 1>),
(<DOMAIN_TYPE.NEG: 0>,
<DOMAIN_TYPE.POS: 1>,
<DOMAIN_TYPE.NEG: 0>,
<DOMAIN_TYPE.NEG: 0>),
(<DOMAIN_TYPE.NEG: 0>,
<DOMAIN_TYPE.POS: 1>,
<DOMAIN_TYPE.POS: 1>,
<DOMAIN_TYPE.NEG: 0>),
(<DOMAIN_TYPE.NEG: 0>,
<DOMAIN_TYPE.NEG: 0>,
<DOMAIN_TYPE.POS: 1>,
<DOMAIN_TYPE.POS: 1>)]
[26]:
DrawDC(z_disc.Indicator(level_sets_p1), -3.5, 2.5, mesh, "zdisc")
[26]:
BaseWebGuiScene
However, we note that there are 7 regions, rather than the expected 5. However, on closer inspection, we see that (NEG, POS, POS, POS)
and (NEG, POS, POS, NEG)
are regions that do not exist with respect to our level sets.
We have two options to remove such irrelevant regions from a DomainTypeArray
:
First:
If the object has already been created, we can use the .Compress(lsets, persistent=False)
method. Given a tuple of level set functions lsets
, the .Compress
method removes any domain tuples from the instance, which have zero measure with respect to these level set functions. If persistent=True
is passed to .Compress
, then any DomainTypeArray
s derived from the compressed one will also be compressed with respect to the here given set of level set fucntions.
So that we are able to integrate internally, lsets
needs to be of the type tuple(ngsolve.GridFunction)
, such that we have acess to the mesh.
[27]:
z_disc.as_list
[27]:
[(<DOMAIN_TYPE.NEG: 0>,
<DOMAIN_TYPE.NEG: 0>,
<DOMAIN_TYPE.NEG: 0>,
<DOMAIN_TYPE.POS: 1>),
(<DOMAIN_TYPE.NEG: 0>,
<DOMAIN_TYPE.NEG: 0>,
<DOMAIN_TYPE.POS: 1>,
<DOMAIN_TYPE.NEG: 0>),
(<DOMAIN_TYPE.NEG: 0>,
<DOMAIN_TYPE.POS: 1>,
<DOMAIN_TYPE.NEG: 0>,
<DOMAIN_TYPE.POS: 1>),
(<DOMAIN_TYPE.NEG: 0>,
<DOMAIN_TYPE.POS: 1>,
<DOMAIN_TYPE.POS: 1>,
<DOMAIN_TYPE.POS: 1>),
(<DOMAIN_TYPE.NEG: 0>,
<DOMAIN_TYPE.POS: 1>,
<DOMAIN_TYPE.NEG: 0>,
<DOMAIN_TYPE.NEG: 0>),
(<DOMAIN_TYPE.NEG: 0>,
<DOMAIN_TYPE.POS: 1>,
<DOMAIN_TYPE.POS: 1>,
<DOMAIN_TYPE.NEG: 0>),
(<DOMAIN_TYPE.NEG: 0>,
<DOMAIN_TYPE.NEG: 0>,
<DOMAIN_TYPE.POS: 1>,
<DOMAIN_TYPE.POS: 1>)]
[28]:
z_disc.Compress(level_sets_p1)
z_disc.as_list
[28]:
[(<DOMAIN_TYPE.NEG: 0>,
<DOMAIN_TYPE.NEG: 0>,
<DOMAIN_TYPE.NEG: 0>,
<DOMAIN_TYPE.POS: 1>),
(<DOMAIN_TYPE.NEG: 0>,
<DOMAIN_TYPE.NEG: 0>,
<DOMAIN_TYPE.POS: 1>,
<DOMAIN_TYPE.NEG: 0>),
(<DOMAIN_TYPE.NEG: 0>,
<DOMAIN_TYPE.POS: 1>,
<DOMAIN_TYPE.NEG: 0>,
<DOMAIN_TYPE.POS: 1>),
(<DOMAIN_TYPE.NEG: 0>,
<DOMAIN_TYPE.POS: 1>,
<DOMAIN_TYPE.NEG: 0>,
<DOMAIN_TYPE.NEG: 0>),
(<DOMAIN_TYPE.NEG: 0>,
<DOMAIN_TYPE.NEG: 0>,
<DOMAIN_TYPE.POS: 1>,
<DOMAIN_TYPE.POS: 1>)]
This removes the unnecessary regions (NEG, POS, POS, POS)
and (NEG, POS, POS, NEG)
.
Second:
On initialisation of a DomainTypeArray
, we can also pass a tuple lsets
of level sets for compression at initilisation, and a flag persistent_compress
which indicates, whether the result when using operators schould again be compressed with respect to the same set of level sets
[29]:
tmp1 = DomainTypeArray((NEG,ANY,ANY,ANY),
lsets=level_sets_p1,
persistent_compress=True)
tmp2 = DomainTypeArray((NEG,NEG,NEG,NEG),
lsets=level_sets_p1,
persistent_compress=True)
z_disc2 = tmp1 & ~tmp2
z_disc2.as_list
[29]:
[(<DOMAIN_TYPE.NEG: 0>,
<DOMAIN_TYPE.NEG: 0>,
<DOMAIN_TYPE.NEG: 0>,
<DOMAIN_TYPE.POS: 1>),
(<DOMAIN_TYPE.NEG: 0>,
<DOMAIN_TYPE.NEG: 0>,
<DOMAIN_TYPE.POS: 1>,
<DOMAIN_TYPE.NEG: 0>),
(<DOMAIN_TYPE.NEG: 0>,
<DOMAIN_TYPE.POS: 1>,
<DOMAIN_TYPE.NEG: 0>,
<DOMAIN_TYPE.POS: 1>),
(<DOMAIN_TYPE.NEG: 0>,
<DOMAIN_TYPE.POS: 1>,
<DOMAIN_TYPE.NEG: 0>,
<DOMAIN_TYPE.NEG: 0>),
(<DOMAIN_TYPE.NEG: 0>,
<DOMAIN_TYPE.NEG: 0>,
<DOMAIN_TYPE.POS: 1>,
<DOMAIN_TYPE.POS: 1>)]
The difference between persistent_compress=True/False
becomes more apparent when computing the boundary.
[30]:
z_disc = DomainTypeArray((NEG,ANY,ANY,ANY)) & ~DomainTypeArray((NEG,NEG,NEG,NEG))
bnd1 = z_disc.Boundary()
DrawDC(bnd1.IndicatorSmoothed(level_sets_p1,0.08/interactive), -3.5, 2.5, mesh, "bnd1")
bnd1.as_list
[30]:
[(<DOMAIN_TYPE.IF: 2>,
<DOMAIN_TYPE.POS: 1>,
<DOMAIN_TYPE.POS: 1>,
<DOMAIN_TYPE.NEG: 0>),
(<DOMAIN_TYPE.IF: 2>,
<DOMAIN_TYPE.NEG: 0>,
<DOMAIN_TYPE.NEG: 0>,
<DOMAIN_TYPE.POS: 1>),
(<DOMAIN_TYPE.IF: 2>,
<DOMAIN_TYPE.POS: 1>,
<DOMAIN_TYPE.NEG: 0>,
<DOMAIN_TYPE.POS: 1>),
(<DOMAIN_TYPE.IF: 2>,
<DOMAIN_TYPE.NEG: 0>,
<DOMAIN_TYPE.POS: 1>,
<DOMAIN_TYPE.POS: 1>),
(<DOMAIN_TYPE.NEG: 0>,
<DOMAIN_TYPE.NEG: 0>,
<DOMAIN_TYPE.IF: 2>,
<DOMAIN_TYPE.NEG: 0>),
(<DOMAIN_TYPE.IF: 2>,
<DOMAIN_TYPE.POS: 1>,
<DOMAIN_TYPE.NEG: 0>,
<DOMAIN_TYPE.NEG: 0>),
(<DOMAIN_TYPE.NEG: 0>,
<DOMAIN_TYPE.IF: 2>,
<DOMAIN_TYPE.NEG: 0>,
<DOMAIN_TYPE.NEG: 0>),
(<DOMAIN_TYPE.NEG: 0>,
<DOMAIN_TYPE.NEG: 0>,
<DOMAIN_TYPE.NEG: 0>,
<DOMAIN_TYPE.IF: 2>),
(<DOMAIN_TYPE.IF: 2>,
<DOMAIN_TYPE.POS: 1>,
<DOMAIN_TYPE.POS: 1>,
<DOMAIN_TYPE.POS: 1>),
(<DOMAIN_TYPE.IF: 2>,
<DOMAIN_TYPE.NEG: 0>,
<DOMAIN_TYPE.POS: 1>,
<DOMAIN_TYPE.NEG: 0>)]
[31]:
bnd2 = z_disc2.Boundary()
DrawDC(bnd2.IndicatorSmoothed(level_sets_p1,0.08/interactive), -3.5, 2.5, mesh, "bnd2")
bnd2.as_list
[31]:
[(<DOMAIN_TYPE.IF: 2>,
<DOMAIN_TYPE.NEG: 0>,
<DOMAIN_TYPE.NEG: 0>,
<DOMAIN_TYPE.POS: 1>),
(<DOMAIN_TYPE.IF: 2>,
<DOMAIN_TYPE.POS: 1>,
<DOMAIN_TYPE.NEG: 0>,
<DOMAIN_TYPE.POS: 1>),
(<DOMAIN_TYPE.IF: 2>,
<DOMAIN_TYPE.NEG: 0>,
<DOMAIN_TYPE.POS: 1>,
<DOMAIN_TYPE.POS: 1>),
(<DOMAIN_TYPE.NEG: 0>,
<DOMAIN_TYPE.NEG: 0>,
<DOMAIN_TYPE.IF: 2>,
<DOMAIN_TYPE.NEG: 0>),
(<DOMAIN_TYPE.IF: 2>,
<DOMAIN_TYPE.POS: 1>,
<DOMAIN_TYPE.NEG: 0>,
<DOMAIN_TYPE.NEG: 0>),
(<DOMAIN_TYPE.NEG: 0>,
<DOMAIN_TYPE.IF: 2>,
<DOMAIN_TYPE.NEG: 0>,
<DOMAIN_TYPE.NEG: 0>),
(<DOMAIN_TYPE.NEG: 0>,
<DOMAIN_TYPE.NEG: 0>,
<DOMAIN_TYPE.NEG: 0>,
<DOMAIN_TYPE.IF: 2>),
(<DOMAIN_TYPE.IF: 2>,
<DOMAIN_TYPE.NEG: 0>,
<DOMAIN_TYPE.POS: 1>,
<DOMAIN_TYPE.NEG: 0>)]
Generating outward pointing unit normal vectors¶
To implement boundary conditions using Nitsche’s method, for example in a CutFEM problem, the outward pointing unit normal vectors are needed. Since in the multiple level set setting the boundary can consist of many sections (8 in the case of the Zalesak disk), we have a member function to compute the outward pointing unit normal vector on each section of the boundary of a codim=0 domain.
[32]:
normals = z_disc2.GetOuterNormals(level_sets_p1)
normals
[32]:
{(<DOMAIN_TYPE.IF: 2>,
<DOMAIN_TYPE.NEG: 0>,
<DOMAIN_TYPE.NEG: 0>,
<DOMAIN_TYPE.POS: 1>): <ngsolve.fem.CoefficientFunction at 0x7fda955226f0>,
(<DOMAIN_TYPE.NEG: 0>,
<DOMAIN_TYPE.NEG: 0>,
<DOMAIN_TYPE.NEG: 0>,
<DOMAIN_TYPE.IF: 2>): <ngsolve.fem.CoefficientFunction at 0x7fda95521eb0>,
(<DOMAIN_TYPE.IF: 2>,
<DOMAIN_TYPE.NEG: 0>,
<DOMAIN_TYPE.POS: 1>,
<DOMAIN_TYPE.NEG: 0>): <ngsolve.fem.CoefficientFunction at 0x7fda95522810>,
(<DOMAIN_TYPE.NEG: 0>,
<DOMAIN_TYPE.NEG: 0>,
<DOMAIN_TYPE.IF: 2>,
<DOMAIN_TYPE.NEG: 0>): <ngsolve.fem.CoefficientFunction at 0x7fda95522150>,
(<DOMAIN_TYPE.IF: 2>,
<DOMAIN_TYPE.POS: 1>,
<DOMAIN_TYPE.NEG: 0>,
<DOMAIN_TYPE.POS: 1>): <ngsolve.fem.CoefficientFunction at 0x7fda95521790>,
(<DOMAIN_TYPE.IF: 2>,
<DOMAIN_TYPE.POS: 1>,
<DOMAIN_TYPE.NEG: 0>,
<DOMAIN_TYPE.NEG: 0>): <ngsolve.fem.CoefficientFunction at 0x7fda95523890>,
(<DOMAIN_TYPE.NEG: 0>,
<DOMAIN_TYPE.IF: 2>,
<DOMAIN_TYPE.NEG: 0>,
<DOMAIN_TYPE.NEG: 0>): <ngsolve.fem.CoefficientFunction at 0x7fda94bc23f0>,
(<DOMAIN_TYPE.IF: 2>,
<DOMAIN_TYPE.NEG: 0>,
<DOMAIN_TYPE.POS: 1>,
<DOMAIN_TYPE.POS: 1>): <ngsolve.fem.CoefficientFunction at 0x7fda955238f0>}
[33]:
ngsolve.internal.visoptions.showsurfacesolution=1
Draw(normals[(IF,NEG,NEG,POS)], mesh, "normal_1", vectors={'grid_size': 14})
DrawDC(DomainTypeArray((IF,NEG,NEG,POS)).IndicatorSmoothed(level_sets_p1, eps=0.08/interactive), -3.5, 2.5, mesh, "boundary_1")
[33]:
BaseWebGuiScene
[34]:
Draw(normals[(NEG,NEG,NEG,IF)], mesh, "normal_2", vectors={'grid_size': 14})
DrawDC(DomainTypeArray((NEG,NEG,NEG,IF)).IndicatorSmoothed(level_sets_p1, eps=0.08/interactive), -3.5, 2.5, mesh, "boundary_2")
[34]:
BaseWebGuiScene
[35]:
Draw(normals[(IF,NEG,POS,NEG)], mesh, "normal_3", vectors={'grid_size': 14})
DrawDC(DomainTypeArray((IF,NEG,POS,NEG)).IndicatorSmoothed(level_sets_p1, eps=0.08/interactive), -3.5, 2.5, mesh, "boundary_3")
[35]:
BaseWebGuiScene
[36]:
Draw(normals[(NEG,NEG,IF,NEG)], mesh, "normal_4", vectors={'grid_size': 14})
DrawDC(DomainTypeArray((NEG,NEG,IF,NEG)).IndicatorSmoothed(level_sets_p1, eps=0.08/interactive), -3.5, 2.5, mesh, "boundary_4")
[36]:
BaseWebGuiScene
[37]:
Draw(normals[(IF,POS,NEG,POS)], mesh, "normal_5", vectors={'grid_size': 14})
DrawDC(DomainTypeArray((IF,POS,NEG,POS)).IndicatorSmoothed(level_sets_p1, eps=0.08/interactive), -3.5, 2.5, mesh, "boundary_5")
[37]:
BaseWebGuiScene
[38]:
Draw(normals[(IF,POS,NEG,NEG)], mesh, "normal_6", vectors={'grid_size': 14})
DrawDC(DomainTypeArray((IF,POS,NEG,NEG)).IndicatorSmoothed(level_sets_p1, eps=0.08/interactive), -3.5, 2.5, mesh, "boundary_6")
[38]:
BaseWebGuiScene
[39]:
Draw(normals[(NEG,IF,NEG,NEG)], mesh, "normal_7", vectors={'grid_size': 14})
DrawDC(DomainTypeArray((NEG,IF,NEG,NEG)).IndicatorSmoothed(level_sets_p1, eps=0.08/interactive), -3.5, 2.5, mesh, "boundary_7")
[39]:
BaseWebGuiScene
[40]:
Draw(normals[(IF,NEG,POS,POS)], mesh, "normal_8", vectors={'grid_size': 14})
DrawDC(DomainTypeArray((IF,NEG,POS,POS)).IndicatorSmoothed(level_sets_p1, eps=0.08/interactive), -3.5, 2.5, mesh, "boundary_8")
[40]:
BaseWebGuiScene
Integrating using DomainTypeArray
s¶
In order to use DomainTypeArrays for integration purposes we can simply pass the DomainTypeArray
to the domain_type
argumet of the SymbolicDifferentialSymbol dCut
:
[41]:
from math import pi
area_zd = pi - 2 / 3 * (0.5 + sqrt(2) * 2 / 3)
area_zd += - (2 * asin(1 / 3) - sin(2 * asin(1 / 3))) / 2
area = Integrate(CoefficientFunction(1) * dCut(level_sets_p1, z_disc2, order=0), mesh)
print("Area 2 = {:12.10f}".format(area))
print("Error: {:4.2e}".format(abs(area - area_zd)))
Area 2 = 2.1260431413
Error: 2.81e-02
Since the smooth level sets are not all piecewise linear, integration is only accurate up to \(\mathcal{O}(h^2)\)
Constructing new domais from ``DomainTypeArrays`` defined with respect to different level sets
Some complicated gemetries consist of the union or the intersection of geometrical objects which are easily descriped be a small number of level sets. For this purpose xfem.mlset
provides the two functions TensorUnion
and TensorIntersection
, which generate the DomainTypeArray
description from the union or intersection of different DomainTypeArray
s, each defined with respect to different level sets.
For example, consider the follwoing two triangular domains:
[42]:
geo = SplineGeometry()
geo.AddRectangle((-2.5, -1), (1, 2.5), bc=1)
mesh = Mesh(geo.GenerateMesh(maxh=0.3/interactive))
level_sets1 = [-2*x + y - 1, 2*x + y - 1, -y + 2]
dta1 = DomainTypeArray((POS, POS, POS))
DrawDC(dta1.Indicator(level_sets1), -3.5, 2.5, mesh, "dta1")
level_sets2 = [-2*x + y - 4, 2*x + y + 2, -y]
dta2 = DomainTypeArray((NEG, NEG, NEG))
DrawDC(dta2.Indicator(level_sets2), -3.5, 2.5, mesh, "dta2")
[42]:
BaseWebGuiScene
Then we can construct the Union of these two domains as
[43]:
dta_union = TensorUnion(dta1, dta2)
DrawDC(dta_union.Indicator(level_sets1 + level_sets2), -3.5, 2.5, mesh, "union")
[43]:
BaseWebGuiScene
Note, that a the same result would have been achieved "by hand" with
[44]:
dta3 = DomainTypeArray([(POS, POS, POS, ANY, ANY, ANY),
(ANY, ANY, ANY, NEG, NEG, NEG)])
DrawDC(dta3.Indicator(level_sets1 + level_sets2), -3.5, 2.5, mesh, "union")
[44]:
BaseWebGuiScene
Unsing TensorIntersection
we can also generate the domain description of "outside both triangles"
[45]:
dta_outside = TensorIntersection(~dta1, ~dta2)
DrawDC(dta_outside.Indicator(level_sets1 + level_sets2), -3.5, 2.5, mesh, "outside")
[45]:
BaseWebGuiScene
Notehowever, that the description can be very long and contain non existent regions. This is because the DomainTypeArray
does not have any information on the actual level sets used to define the domains.
[46]:
print(dta_outside)
print("Number of regions in DomainTypeArray:", len(dta_outside))
DomainTypeArray:
Co-dim : 0
Domains: [(<DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.NEG: 0>), (<DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.NEG: 0>), (<DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.POS: 1>), (<DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.NEG: 0>), (<DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.NEG: 0>), (<DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.POS: 1>), (<DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.POS: 1>), (<DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.NEG: 0>), (<DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.POS: 1>), (<DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.POS: 1>), (<DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.POS: 1>), (<DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.NEG: 0>), (<DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.POS: 1>), (<DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.NEG: 0>), (<DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.POS: 1>), (<DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.POS: 1>), (<DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.POS: 1>), (<DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.POS: 1>), (<DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.NEG: 0>), (<DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.POS: 1>), (<DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.POS: 1>), (<DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.POS: 1>), (<DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.NEG: 0>), (<DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.POS: 1>), (<DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.NEG: 0>), (<DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.POS: 1>), (<DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.POS: 1>), (<DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.POS: 1>), (<DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.NEG: 0>), (<DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.POS: 1>), (<DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.POS: 1>), (<DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.NEG: 0>), (<DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.POS: 1>), (<DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.POS: 1>), (<DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.NEG: 0>), (<DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.NEG: 0>), (<DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.POS: 1>), (<DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.NEG: 0>), (<DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.NEG: 0>), (<DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.POS: 1>), (<DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.POS: 1>), (<DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.NEG: 0>), (<DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.NEG: 0>), (<DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.NEG: 0>), (<DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.NEG: 0>), (<DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.POS: 1>), (<DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.POS: 1>), (<DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.POS: 1>), (<DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.POS: 1>, <DOMAIN_TYPE.NEG: 0>, <DOMAIN_TYPE.NEG: 0>)]
Number of regions in DomainTypeArray: 49
In such cases, using the .Compress()
memeber of DomainTypeArray
can make sense, to remove regions which do not contibute to the domain for the given level sets:
[47]:
P1 = H1(mesh, order=1)
levelsets_p1 = tuple([GridFunction(P1) for _l in level_sets1 + level_sets2])
for lsetp1, lset in zip(levelsets_p1, level_sets1 + level_sets2):
InterpolateToP1(lset, lsetp1)
dta_outside.Compress(levelsets_p1)
print("Number of regions in DomainTypeArray after compression:", len(dta_outside))
Number of regions in DomainTypeArray after compression: 16
[48]:
DrawDC(dta_outside.Indicator(levelsets_p1), -3.5, 2.5, mesh, "outside")
[48]:
BaseWebGuiScene