use super::*;
pub type MFbSyn = MotionSyn<MNormFourBar, 6, 2>;
pub type MotionSyn<M, const N: usize, const D: usize> = Syn<efd::PosedEfd<D>, M, N, D>;
impl<M, const N: usize, const D: usize> MotionSyn<M, N, D>
where
efd::U<D>: efd::EfdDim<D>,
{
pub fn from_series<C1, C2>(curve1: C1, curve2: C2, mode: Mode) -> Self
where
C1: efd::Curve<D>,
C2: efd::Curve<D>,
{
let efd = efd::PosedEfd::from_series(curve1, curve2, mode.is_target_open());
Self::from_efd(efd, mode)
}
pub fn from_uvec<C, V>(curve: C, vectors: V, mode: Mode) -> Self
where
C: efd::Curve<D>,
V: efd::Curve<D>,
{
let efd = efd::PosedEfd::from_uvec(curve, vectors, mode.is_target_open());
Self::from_efd(efd, mode)
}
pub fn from_efd(efd: efd::PosedEfd<D>, mode: Mode) -> Self {
Self::new(efd, mode)
}
pub fn harmonic(&self) -> usize {
self.tar.harmonic()
}
}
impl<M, const N: usize, const D: usize> mh::Bounded for MotionSyn<M, N, D>
where
Self: mh::ObjFunc,
M: mech::FromVectorized<N>,
efd::U<D>: efd::EfdDim<D>,
{
#[inline]
fn bound(&self) -> &[[f64; 2]] {
if self.mode == Mode::Partial {
M::BOUND_PARTIAL
} else {
&M::BOUND
}
}
}
#[doc(hidden)]
#[derive(Clone)]
pub struct MOFit {
pub curve: f64,
pub pose: f64,
pub center: f64,
pub unit: f64,
}
impl mh::Fitness for MOFit {
type Best<T: mh::Fitness> = mh::pareto::Pareto<T>;
type Eval = f64;
fn is_dominated(&self, rhs: &Self) -> bool {
self.curve <= rhs.curve
&& self.pose <= rhs.pose
&& self.center <= rhs.center
&& self.unit <= rhs.unit
}
fn eval(&self) -> Self::Eval {
self.curve + self.pose + self.center + self.unit
}
}
impl Infeasible for MOFit {
fn infeasible() -> Self {
Self {
curve: infeasible(),
pose: infeasible(),
center: infeasible(),
unit: infeasible(),
}
}
}
impl<M, const N: usize, const D: usize> mh::ObjFunc for MotionSyn<M, N, D>
where
M: SynBound<N> + mech::Normalized<D> + mech::PoseGen<D>,
M::De: Default + Clone + Sync + Send + 'static,
efd::Efd<D>: Sync + Send,
efd::U<D>: efd::EfdDim<D>,
{
type Ys = mh::WithProduct<MOFit, M::De>;
fn fitness(&self, xs: &[f64]) -> Self::Ys {
let is_open = self.mode.is_target_open();
let get_series = |fb: &M, start, end| {
let (curve, pose) = fb.pose_in(start, end, self.res);
(curve.len() > 2).then_some((curve, pose))
};
impl_fitness(self.mode, xs, get_series, |((c, v), fb)| {
let efd = efd::PosedEfd::from_uvec_harmonic(c, v, is_open, self.harmonic());
let geo = efd.as_curve().as_geo().to(self.tar.as_curve().as_geo());
let fb = fb.clone().trans_denorm(&geo);
let err = MOFit {
curve: efd.as_curve().err(self.tar.as_curve()),
pose: efd.as_pose().err(self.tar.as_pose()),
center: {
use efd::Distance as _;
let me = efd.as_pose().as_geo().trans();
let tar = self.tar.as_pose().as_geo().trans();
me.l2_err(tar)
},
unit: self.unit_err(&geo),
};
mh::WithProduct::new(err, fb)
})
}
}