Source code for ufl.finiteelement.tensorproductelement
# -*- coding: utf-8 -*-
"This module defines the UFL finite element classes."
# Copyright (C) 2008-2016 Martin Sandve Alnæs
#
# This file is part of UFL (https://www.fenicsproject.org)
#
# SPDX-License-Identifier: LGPL-3.0-or-later
#
# Modified by Kristian B. Oelgaard
# Modified by Marie E. Rognes 2010, 2012
# Modified by Massimiliano Leoni, 2016
from itertools import chain
from ufl.log import error
from ufl.cell import TensorProductCell, as_cell
from ufl.sobolevspace import DirectionalSobolevSpace
from ufl.finiteelement.finiteelementbase import FiniteElementBase
[docs]class TensorProductElement(FiniteElementBase):
r"""The tensor product of :math:`d` element spaces:
.. math:: V = V_1 \otimes V_2 \otimes ... \otimes V_d
Given bases :math:`\{\phi_{j_i}\}` of the spaces :math:`V_i` for :math:`i = 1, ...., d`,
:math:`\{ \phi_{j_1} \otimes \phi_{j_2} \otimes \cdots \otimes \phi_{j_d}
\}` forms a basis for :math:`V`.
"""
__slots__ = ("_sub_elements", "_cell")
def __init__(self, *elements, **kwargs):
"Create TensorProductElement from a given list of elements."
if not elements:
error("Cannot create TensorProductElement from empty list.")
keywords = list(kwargs.keys())
if keywords and keywords != ["cell"]:
raise ValueError("TensorProductElement got an unexpected keyword argument '%s'" % keywords[0])
cell = kwargs.get("cell")
family = "TensorProductElement"
if cell is None:
# Define cell as the product of each elements cell
cell = TensorProductCell(*[e.cell() for e in elements])
else:
cell = as_cell(cell)
# Define polynomial degree as a tuple of sub-degrees
degree = tuple(e.degree() for e in elements)
# No quadrature scheme defined
quad_scheme = None
# match FIAT implementation
value_shape = tuple(chain(*[e.value_shape() for e in elements]))
reference_value_shape = tuple(chain(*[e.reference_value_shape() for e in elements]))
if len(value_shape) > 1:
error("Product of vector-valued elements not supported")
if len(reference_value_shape) > 1:
error("Product of vector-valued elements not supported")
FiniteElementBase.__init__(self, family, cell, degree,
quad_scheme, value_shape,
reference_value_shape)
self._sub_elements = elements
self._cell = cell
self._repr = "TensorProductElement(%s, cell=%s)" % (
", ".join(repr(e) for e in elements), repr(cell))
[docs] def mapping(self):
if all(e.mapping() == "identity" for e in self._sub_elements):
return "identity"
elif all(e.mapping() == "L2 Piola" for e in self._sub_elements):
return "L2 Piola"
else:
return "undefined"
[docs] def sobolev_space(self):
"Return the underlying Sobolev space of the TensorProductElement."
elements = self._sub_elements
if all(e.sobolev_space() == elements[0].sobolev_space()
for e in elements):
return elements[0].sobolev_space()
else:
# Generate a DirectionalSobolevSpace which contains
# continuity information parametrized by spatial index
orders = []
for e in elements:
e_dim = e.cell().geometric_dimension()
e_order = (e.sobolev_space()._order,) * e_dim
orders.extend(e_order)
return DirectionalSobolevSpace(orders)
[docs] def num_sub_elements(self):
"Return number of subelements."
return len(self._sub_elements)
[docs] def sub_elements(self):
"Return subelements (factors)."
return self._sub_elements
[docs] def reconstruct(self, **kwargs):
cell = kwargs.pop("cell", self.cell())
return TensorProductElement(*[e.reconstruct(**kwargs) for e in self.sub_elements()], cell=cell)
def __str__(self):
"Pretty-print."
return "TensorProductElement(%s, cell=%s)" \
% (', '.join([str(e) for e in self._sub_elements]), str(self._cell))
[docs] def shortstr(self):
"Short pretty-print."
return "TensorProductElement(%s, cell=%s)" \
% (', '.join([e.shortstr() for e in self._sub_elements]), str(self._cell))