from scipy.spatial import Delaunay
import numpy as np
import itertools
from .reference_elements import ReferenceTriangle, ReferenceInterval
[docs]
class Mesh(object):
"""A one or two dimensional mesh composed of intervals or triangles
respectively."""
def __init__(self, vertex_coords, cell_vertices):
"""
:param vertex_coords: a vertex_count x dim array of the coordinates of
the vertices in the mesh.
:param cell_vertices: a cell_count x (dim+1) array of the
indices of the vertices of which each cell is made up.
"""
self.dim = vertex_coords.shape[1]
"""The geometric and topological dimension of the mesh. Immersed
manifolds are not supported.
"""
if self.dim not in (1, 2):
raise ValueError("Only 1D and 2D meshes are supported")
self.vertex_coords = vertex_coords
"""The coordinates of all the vertices in the mesh."""
self.cell_vertices = np.sort(cell_vertices)
"""The indices of the vertices incident to cell."""
if self.dim == 2:
self.edge_vertices = np.array(
list(
set(
tuple(sorted(e))
for t in cell_vertices
for e in itertools.combinations(t, 2)
)
)
)
"""The indices of the vertices incident to edge (only for 2D
meshes)."""
# Invert self.edge_vertices so that it is possible to look up
# the edge index given the vertex indices.
edge_dict = {
tuple(e): i
for i, e_ in enumerate(self.edge_vertices)
for e in (e_, reversed(e_))
}
# List the local vertex indices associated with
# each local edge index.
local_edge_vertices = np.array([[1, 2], [0, 2], [0, 1]])
self.cell_edges = np.fromiter(
(
edge_dict[tuple(t.take(local_edge_vertices[e]))]
for t in self.cell_vertices
for e in range(3)
),
dtype=np.int32,
count=self.cell_vertices.size,
).reshape((-1, 3))
"""The indices of the edges incident to each cell (only for 2D
meshes)."""
if self.dim == 2:
self.entity_counts = np.array(
(
vertex_coords.shape[0],
self.edge_vertices.shape[0],
self.cell_vertices.shape[0],
)
)
"""The number of entities of each dimension in the mesh. So
:attr:`entity_counts[0]` is the number of vertices in the
mesh."""
else:
self.entity_counts = np.array(
(vertex_coords.shape[0], self.cell_vertices.shape[0])
)
#: The :class:`~.reference_elements.ReferenceCell` of which this
#: :class:`Mesh` is composed.
self.cell = (0, ReferenceInterval, ReferenceTriangle)[self.dim]
[docs]
def adjacency(self, dim1, dim2):
"""Return the set of `dim2` entities adjacent to each `dim1`
entity. For example if `dim1==2` and `dim2==1` then return the list of
edges (1D entities) adjacent to each triangle (2D entity).
The return value is a rank 2 :class:`numpy.array` such that
``adjacency(dim1, dim2)[e1, :]`` is the list of dim2 entities
adjacent to entity ``(dim1, e1)``.
This operation is only defined where `self.dim >= dim1 > dim2`.
This method is simply a more systematic way of accessing
:attr:`edge_vertices`, :attr:`cell_edges` and :attr:`cell_vertices`.
"""
if dim2 >= dim1:
raise ValueError("""dim2 must be less than dim1.""")
if dim2 < 0:
raise ValueError("""dim2 cannot be negative.""")
if dim1 > self.dim:
raise ValueError("""dim1 cannot exceed the mesh dimension.""")
if dim1 == 1:
if self.dim == 1:
return self.cell_vertices
else:
return self.edge_vertices
elif dim1 == 2:
if dim2 == 0:
return self.cell_vertices
else:
return self.cell_edges
[docs]
def jacobian(self, c):
"""Return the Jacobian matrix for the specified cell.
:param c: The number of the cell for which to return the Jacobian.
:result: The Jacobian for cell ``c``.
"""
raise NotImplementedError
[docs]
class UnitIntervalMesh(Mesh):
"""A mesh of the unit interval."""
def __init__(self, nx):
"""
:param nx: The number of cells.
"""
points = np.array(list((x,) for x in np.linspace(0, 1, nx + 1)))
points.shape = (points.shape[0], 1)
cells = np.array(list((a, a + 1) for a in range(nx)))
super(UnitIntervalMesh, self).__init__(points, cells)
[docs]
class UnitSquareMesh(Mesh):
"""A triangulated :class:`Mesh` of the unit square."""
def __init__(self, nx, ny):
"""
:param nx: The number of cells in the x direction.
:param ny: The number of cells in the y direction.
"""
points = list(
(x, y) for x in np.linspace(0, 1, nx + 1) for y in np.linspace(0, 1, ny + 1)
)
mesh = Delaunay(points)
super(UnitSquareMesh, self).__init__(mesh.points, mesh.simplices)