Program Listing for File decasteljau.h

Return to documentation for file (manif/algorithms/decasteljau.h)

#ifndef _MANIF_MANIF_DECASTELJAU_H_
#define _MANIF_MANIF_DECASTELJAU_H_

#include "manif/impl/lie_group_base.h"

#include <vector>

namespace manif {

template <typename LieGroup>
std::vector<typename LieGroup::LieGroup>
decasteljau(const std::vector<LieGroup>& trajectory,
            const unsigned int degree,
            const unsigned int k_interp,
            const bool closed_curve = false)
{
  MANIF_CHECK(trajectory.size() > 2,
    "Input trajectory must have more than two points!");
  MANIF_CHECK(degree <= trajectory.size(),
    "Degree must be less or equal to the number of input points!");
  MANIF_CHECK(k_interp > 0,
    "k_interp must be greater than zero!");

  // Number of connected, non-overlapping segments
  const unsigned int n_segments = static_cast<unsigned int>(
      std::floor(double(trajectory.size()-degree)/double((degree-1)+1))
  );

  std::vector<std::vector<const LieGroup*>> segments_control_points;
  for (unsigned int t=0; t<n_segments; ++t)
  {
    segments_control_points.emplace_back(std::vector<const LieGroup*>());

    // Retrieve control points of the current segment
    for (unsigned int n=0; n<degree; ++n)
    {
      segments_control_points.back().push_back( &trajectory[t*(degree-1)+n] );
    }
  }

  // Close the curve if there are left-over points
  if (closed_curve && (n_segments*(degree-1)) <= trajectory.size()-1)
  {
    const unsigned int last_pts_idx = n_segments*(degree-1);
    const unsigned int left_over = trajectory.size()-1-last_pts_idx;
    segments_control_points.emplace_back(std::vector<const LieGroup*>());

    // Get the left-over points
    for (unsigned int p=last_pts_idx; p<trajectory.size(); ++p)
    {
      segments_control_points.back().push_back( &trajectory[p] );
    }
    // Add a extra points from the beginning of the trajectory
    for (unsigned int p=0; p<degree-left_over-1; ++p)
    {
      segments_control_points.back().push_back( &trajectory[p] );
    }
  }

  const unsigned int segment_k_interp = (degree == 2) ?
        k_interp : k_interp * degree;

  // Actual curve fitting
  std::vector<LieGroup> curve;
  for (unsigned int s=0; s<segments_control_points.size(); ++s)
  {
    for (unsigned int t=1; t<=segment_k_interp; ++t)
    {
      // t in [0,1]
      const double t_01 = static_cast<double>(t)/(segment_k_interp);

      std::vector<LieGroup> Qs, Qs_tmp;

      for (const auto m : segments_control_points[s])
        Qs.emplace_back(*m);

      // recursive chunk of the algo,
      // compute tmp control points.
      for (unsigned int i=0; i<degree-1; ++i)
      {
        for (unsigned int q=0; q<Qs.size()-1; ++q)
        {
          Qs_tmp.push_back( Qs[q].rplus(Qs[q+1].rminus(Qs[q]) * t_01) );
        }

        Qs = Qs_tmp;
        Qs_tmp.clear();
      }

      curve.push_back(Qs[0]);
    }
  }

  return curve;
}

} /* namespace manif */

#endif /* _MANIF_MANIF_DECASTELJAU_H_ */