In this tutorial we will show how to use the Radial Basis Functions interpolation technique to deform a cube.
First of all we import the required PyGeM class, we import numpy and we set matplotlib for the notebook.
%matplotlib inline
from pygem import RBF
import numpy as np
import matplotlib.pyplot as plt
Using RBF, we can control the deformation by arranging some control points around the object to deform, then moving these latter to induce the morphing. Within PyGeM, the setting of such parameters can be done by parsing an input text file or manually touching the RBF attributes.
Let's try togheter by using an input file: the first step is the creation of the new object. After this, we can use the read_parameters to set the parameters.
rbf = RBF()
rbf.read_parameters(filename='../tests/test_datasets/parameters_rbf_cube.prm')
The following is the parameters file for this particular case. The Radial Basis Functions section describes the basis functions to use. Here we use Gaussian splines with the distance parameter equal to 0.5 (see the documentation of the RBF class for more details). As control points we consider the 8 vertices of the cube (the first one is not exactly the vertex), and we move 3 of them. In the Control points section there are all the coordinates of the control points.
%cat ../tests/test_datasets/parameters_rbf_cube.prm
[Radial Basis Functions]
# This section describes the radial basis functions shape.
# basis funtion is the name of the basis functions to use in the transformation. The functions
# implemented so far are: gaussian_spline, multi_quadratic_biharmonic_spline,
# inv_multi_quadratic_biharmonic_spline, thin_plate_spline, beckert_wendland_c2_basis, polyharmonic_spline.
# For a comprehensive list with details see the class RBF.
basis function: gaussian_spline
# radius is the scaling parameter r that affects the shape of the basis functions. See the documentation
# of the class RBF for details.
radius: 0.5
# The power parameter k for polyharmonic spline
# See the documentation for details
power: 2
[Control points]
# This section describes the RBF control points.
# original control points collects the coordinates of the interpolation control points before the deformation.
original control points: -.1   0.0   0.0
                         0.0   0.0   1.0
                         0.0   1.0   0.0
                         1.0   0.0   0.0
                         0.0   1     1.0
                         1.0   0.0   1.0
                         1.0   1.0   0
                         1.0   1.0   1.0
# deformed control points collects the coordinates of the interpolation control points after the deformation.
deformed control points: 0.1   0.2   .3
                         0.0   0.0   1.0
                         0.0   1.0   0.0
                         1.0   0.0   0.0
                         0.0   .8   1.0
                         1.0   0.0   1.0
                         1.0   1.0   0.0
                         1.2   1.2   1.2
Here we create a $10 \times10 \times 10$ lattice to mimic a cube.
nx, ny, nz = (10, 10, 10)
mesh = np.zeros((nx * ny * nz, 3))
xv = np.linspace(0, 1, nx)
yv = np.linspace(0, 1, ny)
zv = np.linspace(0, 1, nz)
z, y, x = np.meshgrid(zv, yv, xv)
mesh = np.array([x.ravel(), y.ravel(), z.ravel()])
mesh = mesh.T
Now we plot the points to see what we are doing.
fig = plt.figure(1)
ax = fig.add_subplot(111, projection='3d')
ax.scatter(mesh[:, 0], mesh[:, 1], mesh[:, 2], c='blue', marker='o')
ax.set_xlabel('X axis')
ax.set_ylabel('Y axis')
ax.set_zlabel('Z axis')
plt.show()
We can also plot the original control points and the deformed ones.
rbf.plot_points()
Finally we perform the RBF interpolation using the RBF class.
new_mesh = rbf(mesh)
We can plot the new points in order to see the deformation. Try different basis functions and radius to better fit your specific problem.
fig = plt.figure(2)
ax = fig.add_subplot(111, projection='3d')
ax.scatter(new_mesh[:, 0], new_mesh[:, 1], new_mesh[:, 2], c='red', marker='o')
ax.set_xlabel('X axis')
ax.set_ylabel('Y axis')
ax.set_zlabel('Z axis')
plt.show()