https://mooseframework.inl.gov
BSpline.C
Go to the documentation of this file.
1 //* This file is part of the MOOSE framework
2 //* https://mooseframework.inl.gov
3 //*
4 //* All rights reserved, see COPYRIGHT for full restrictions
5 //* https://github.com/idaholab/moose/blob/master/COPYRIGHT
6 //*
7 //* Licensed under LGPL 2.1, please see LICENSE for details
8 //* https://www.gnu.org/licenses/lgpl-2.1.html
9 
10 #include "BSpline.h"
12 #include "MooseError.h"
13 #include "SplineUtils.h"
14 #include "Conversion.h"
15 
16 using namespace libMesh;
17 
18 namespace Moose
19 {
20 BSpline::BSpline(const unsigned int degree,
21  const libMesh::Point & start_point,
22  const libMesh::Point & end_point,
23  const libMesh::RealVectorValue & start_direction,
24  const libMesh::RealVectorValue & end_direction,
25  const unsigned int cps_per_half,
26  const libMesh::Real sharpness)
27  : _degree(degree),
28  _start_point(start_point),
29  _end_point(end_point),
30  _start_dir(start_direction),
31  _end_dir(end_direction),
32  _cps_per_half(cps_per_half),
33  _sharpness(sharpness),
34  _control_points(createControlPoints()),
35  _knot_vector(buildKnotVector())
36 {
37 }
38 
41 {
42  mooseAssert((t >= 0) && (t <= 1), "t should be in [0, 1]. t=" + std::to_string(t));
43  libMesh::Point returnPoint(0, 0, 0);
44  if (t == 1.0)
45  return _control_points.back();
46  for (const auto i : index_range(_control_points))
47  returnPoint += BSpline::CdBBasis(t, i, _degree) * _control_points[i];
48 
49  return returnPoint;
50 }
51 
52 std::vector<libMesh::Point>
54 {
55  std::vector<libMesh::Point> cps;
59  return cps;
60 }
61 
62 std::vector<libMesh::Real>
64 {
65  std::vector<libMesh::Real> knot_vector(_degree, 0); // initialize bottom to zeros
66 
67  const unsigned int num_points_left = _control_points.size() + 1 - _degree;
68  if (_control_points.size() < _degree + 1)
69  mooseError("Number of control points must be greater than or equal to degree + 1!");
70 
71  for (const auto i : make_range(num_points_left))
72  {
73  knot_vector.push_back(i); // count up
74  mooseAssert(i < _control_points.size(),
75  "i must be less than the number of control points because of 0 indexing.");
76  }
77 
78  // Filling with int(degree + 1) 0s at the beginning and 1s at the end makes the B-spline go
79  // through the start and end point Note: the "+1" comes from the loop above which starts at 0 and
80  // ends at num_points_left - 1
81  for (const auto i : make_range(_degree))
82  {
83  libmesh_ignore(i);
84  knot_vector.push_back(num_points_left - 1);
85  }
86 
87  // Normalize values s.t. the the max value is 1.
88  // This must be done to have a constant domain of t to be between 0 and 1 (inclusive).
89  // NOTE: the spacing is uniform, making this a uniform BSpline
90  for (auto & value : knot_vector)
91  {
92  value /= (num_points_left - 1);
93  mooseAssert(value <= 1.0 && value >= 0.0,
94  "knot_vector must be normalized to 1 with minimum value 0.");
95  }
96 
97  return knot_vector;
98 }
99 
101 BSpline::CdBBasis(const libMesh::Real t, const unsigned int i, const unsigned int j) const
102 {
103  mooseAssert(i + 1 < _knot_vector.size(), "Out of bounds access in knot vector");
104  mooseAssert(_knot_vector[i] <= _knot_vector[i + 1],
105  "Knot vector is ill-formatted. Ensure values are increasing." +
107  if (j == 0)
108  return ((t < _knot_vector[i + 1]) && (_knot_vector[i] <= t)) ? 1 : 0;
109  else
110  return BSpline::firstCoeff(t, i, j) * BSpline::CdBBasis(t, i, j - 1) +
111  BSpline::secondCoeff(t, i, j) * BSpline::CdBBasis(t, i + 1, j - 1);
112 }
113 
115 BSpline::firstCoeff(const libMesh::Real t, const unsigned int i, const unsigned int j) const
116 {
117  mooseAssert(i + j < _knot_vector.size(), "Out of bounds access in knot vector");
118  const auto ti = _knot_vector[i];
119  const auto tij = _knot_vector[i + j];
120  if (ti != tij)
121  return (t - ti) / (tij - ti);
122  else
123  return 0;
124 }
125 
127 BSpline::secondCoeff(const libMesh::Real t, const unsigned int i, const unsigned int j) const
128 {
129  mooseAssert(i + j + 1 < _knot_vector.size(), "Out of bounds access in knot vector");
130  libMesh::Real ti1 = _knot_vector[i + 1];
131  libMesh::Real tij1 = _knot_vector[i + j + 1];
132  if (ti1 != tij1)
133  return (tij1 - t) / (tij1 - ti1);
134  else
135  return 0;
136 }
137 }
const libMesh::Real _sharpness
sharpness of the curve
Definition: BSpline.h:96
const unsigned int _cps_per_half
number of control points per half of the spline (vertex is auto-included)
Definition: BSpline.h:94
libMesh::Real secondCoeff(const libMesh::Real t, const unsigned int i, const unsigned int j) const
Submethod used in CdBBasis routine.
Definition: BSpline.C:127
void mooseError(Args &&... args)
Emit an error message with the given stringified, concatenated args and terminate the application...
Definition: MooseError.h:311
const libMesh::RealVectorValue _end_dir
ending direction of spline
Definition: BSpline.h:92
const libMesh::RealVectorValue _start_dir
starting direction of spline
Definition: BSpline.h:90
const libMesh::Point _end_point
ending point of spline
Definition: BSpline.h:88
std::vector< libMesh::Real > buildKnotVector() const
Internal method for building the knot vector given the degree and control points. ...
Definition: BSpline.C:63
The following methods are specializations for using the libMesh::Parallel::packed_range_* routines fo...
libMesh::Real CdBBasis(const libMesh::Real t, const unsigned int i, const unsigned int j) const
Evaluates the the basis function for a B-Spline according to the Cox-de-Boor recursive formula...
Definition: BSpline.C:101
void libmesh_ignore(const Args &...)
Real value(unsigned n, unsigned alpha, unsigned beta, Real x)
std::string stringify(const T &t)
conversion to string
Definition: Conversion.h:64
std::vector< Point > bSplineControlPoints(const libMesh::Point &start_point, const libMesh::Point &end_point, const libMesh::RealVectorValue &start_direction, const libMesh::RealVectorValue &end_direction, const unsigned int cps_per_half, const libMesh::Real sharpness)
Creates control points for an open uniform BSpline.
Definition: SplineUtils.C:67
libMesh::Point getPoint(const libMesh::Real t) const
Evaluate the BSpline interpolation at given value of t.
Definition: BSpline.C:40
const std::vector< libMesh::Point > _control_points
The control points.
Definition: BSpline.h:98
std::vector< libMesh::Point > createControlPoints() const
Creates a vector control points from the SplineUtils set of functions.
Definition: BSpline.C:53
DIE A HORRIBLE DEATH HERE typedef LIBMESH_DEFAULT_SCALAR_TYPE Real
const libMesh::Point _start_point
starting point of spline
Definition: BSpline.h:86
IntRange< T > make_range(T beg, T end)
const std::vector< libMesh::Real > _knot_vector
The knot vector.
Definition: BSpline.h:102
MOOSE now contains C++17 code, so give a reasonable error message stating what the user can do to add...
auto index_range(const T &sizable)
libMesh::Real firstCoeff(const libMesh::Real t, const unsigned int i, const unsigned int j) const
Submethod used in CdBBasis routine.
Definition: BSpline.C:115
const unsigned int _degree
The polynomial degree of the B-Spline.
Definition: BSpline.h:84