#
##
## SPDX-FileCopyrightText: © 2007-2023 Benedict Verhegghe <bverheg@gmail.com>
## SPDX-License-Identifier: GPL-3.0-or-later
##
## This file is part of pyFormex 3.4 (Thu Nov 16 18:07:39 CET 2023)
## pyFormex is a tool for generating, manipulating and transforming 3D
## geometrical models by sequences of mathematical operations.
## Home page: https://pyformex.org
## Project page: https://savannah.nongnu.org/projects/pyformex/
## Development: https://gitlab.com/bverheg/pyformex
## Distributed under the GNU General Public License version 3 or later.
##
## This program is free software: you can redistribute it and/or modify
## it under the terms of the GNU General Public License as published by
## the Free Software Foundation, either version 3 of the License, or
## (at your option) any later version.
##
## This program is distributed in the hope that it will be useful,
## but WITHOUT ANY WARRANTY; without even the implied warranty of
## MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
## GNU General Public License for more details.
##
## You should have received a copy of the GNU General Public License
## along with this program. If not, see http://www.gnu.org/licenses/.
##
"""Isoparametric transformations
"""
import numpy as np
from pyformex import arraytools as at
from pyformex.coords import Coords
from pyformex.plugins.polynomial import Polynomial
[docs]def exponents(n, layout='lag'):
"""Create tuples of polynomial exponents.
Create the exponents of polynomials in 1 to 3 dimensions
which can be used to construct interpolation function over lagrangian,
triangular or serendipity grids.
Parameters
----------
n: tuple of int
A tuple of 1 to 3 integers, specifying the degree of the polynomials
in the x up to z directions. For a lagrangian layout, this is one less
than the number of points in each direction.
layout: str
A string specifying the layout of the grid and the selection of
monomials to be used. Should be one of 'lagrangian', 'triangular',
'serendipity' or 'border'. The string can be abbreviated to its
first 3 characters.
Returns
-------
int array
An integer array with shape (ndim,npoints), where ndim = len(n)
and npoints depends on the layout:
- lagrangian: npoints = prod(n). The point layout is a rectangular
lagrangian grid form by n[i] points in direction i. As an example,
specifying n=(3,2) uses a grid of 3 points in x-direction and 2 points
in y-direction.
- triangular: requires that all values in n are equal. For ndim=2, the
number of points is n*(n+1)//2.
- border: this is like the lagrangian grid with all internal points
removed. For ndim=2, we have npoints = 2 * sum(n) - 4. For ndim=3 we
have npoints = 2 * sum(nx*ny+ny*nz+nz*nx) - 4 * sum(n) + 8.
Thus n=(3,3,3) will yield 2*3*3*3 - 4*(3+3+3) + 8 = 26
- serendipity: tries to use only the corner and edge nodes, but uses
a convex domain of the monomials. This may require some nodes inside
the faces or the volume. Currently works up to (4,4) in 2D or (3,3,3)
in 3D.
See Also
--------
interpoly: Returns the corresponding interpolation polynomial
Examples
--------
Quadratic in x, linear in y:
>>> exponents((2,1))
array([[0, 0],
[1, 0],
[2, 0],
[0, 1],
[1, 1],
[2, 1]])
Linear in x,y,x:
>>> exponents((1,1,1))
array([[0, 0, 0],
[1, 0, 0],
[0, 1, 0],
[1, 1, 0],
[0, 0, 1],
[1, 0, 1],
[0, 1, 1],
[1, 1, 1]])
Quadratic serendipity in x,y:
>>> exponents((2,2), 'ser')
array([[0, 0],
[1, 0],
[2, 0],
[0, 1],
[1, 1],
[2, 1],
[0, 2],
[1, 2]])
"""
n = at.checkArray(n, (-1,), 'i')
ndim = n.shape[0]
if ndim < 1 or ndim > 3:
raise RuntimeError("Expected a 1..3 length tuple")
layout = layout[:3]
if layout in ['tri', 'ser']:
if not (n == n[0]).all():
raise RuntimeError(
"For triangular and serendipity grids, all axes should "
"have the same number of points")
# First create the full lagrangian set
exp = np.indices(n+1).transpose().reshape(-1, ndim)
if layout != 'lag':
if layout == 'tri':
# Select by total maximal degree
ok = exp.sum(axis=-1) <= n[0]
elif layout == 'bor':
# Select if any value is not higher than 1
ok = (exp <= 1).any(axis=-1)
elif layout == 'ser':
if len(n) > 2:
npts = 8 + 12*(n[0] - 1) # minimal number of points
sdeg = n[0] + 1
ok = exp.sum(axis=-1) <= sdeg
if ok.sum() < npts:
sdeg += 1
ok = exp.sum(axis=-1) <= sdeg
ok1 = (exp == n[0]).sum(axis=-1) <= 1
ok = ok*ok1
else:
npts = 4 + 4*(n[0] - 1) # minimal number of points
ok = exp.sum(axis=-1) <= n[0] + 1
if ok.sum() < npts:
raise ValueError("No solution for eltype %s %s" % (layout, n))
else:
raise RuntimeError("Unknown layout %s" % layout)
exp = exp[ok]
return exp
[docs]def interpoly(n, layout='lag'):
"""Create an interpolation polynomial.
This is syntactical sugar for::
Polynomial(exponents(n, layout))
Parameters: see :func:`exponents`.
Returns
-------
Polynomial:
The :class:`Polynomial` constructed from the :func:`exponents`.
"""
return Polynomial(exponents(n, layout))
[docs]class Isopar():
"""A class representing an isoparametric transformation
eltype is one of the keys in Isopar.isodata
coords and oldcoords can be either arrays, Coords or Formex instances,
but should be of equal shape, and match the number of atoms in the
specified transformation type
The following three formulations are equivalent ::
trf = Isopar(eltype,coords,oldcoords)
G = F.isopar(trf)
trf = Isopar(eltype,coords,oldcoords)
G = trf.transform(F)
G = isopar(F,eltype,coords,oldcoords)
"""
# Store the interpolation polynomials corresponding with
# some element
isodata = {
}
isodata_alias = {
'line2': 'lag-1',
'line3': 'lag-2',
'line4': 'lag-3',
'tri3': 'tri-1-1',
'tri6': 'tri-2-2',
'tri10': 'tri-3-3',
'quad4': 'lag-1-1',
'quad9': 'lag-2-2',
'quad16': 'lag-3-3',
'quad8': 'bor-2-2',
'quad12': 'bor-3-3',
'quad13': 'ser-3-3',
'tet4': 'tri-1-1-1',
'tet10': 'tri-2-2-2',
'hex8': 'lag-1-1-1',
'hex20': 'ser-2-2-2',
'hex27': 'lag-2-2-2',
'hex36': 'lag-2-2-3',
'hex64': 'lag-3-3-3',
}
def __init__(self, eltype, coords, oldcoords):
"""Create an isoparametric transformation.
`eltype`: string: either one of the keys in the isodata dictionary,
or a string in the following format: layout-nx-ny-nz
"""
if eltype not in Isopar.isodata:
if eltype in Isopar.isodata_alias:
eltype = Isopar.isodata_alias[eltype]
if eltype not in Isopar.isodata:
s = eltype.split('-')
if s[0] in ['lag', 'tri', 'bor', 'ser']:
n = [int(v) for v in s[1:]]
#
# It might be better to just store (ndim,atoms)
# if we use string atoms to evaluate
#
Isopar.isodata[eltype] = interpoly(n, s[0])
else:
raise RuntimeError("Unknown eltype %s")
poly = Isopar.isodata[eltype]
coords = coords.view().reshape(-1, 3)
oldcoords = oldcoords.view().reshape(-1, 3)
aa = poly.evalAtoms(oldcoords[:, :poly.ndim])
ab = np.linalg.solve(aa, coords)
self.eltype = eltype
self.trf = ab
# End