use crate::matrix;
use crate::{
algebra::{
abstr::Real,
linear::{matrix::General, vector::vector::Vector},
},
analysis::differential_equation::ordinary::{ExplicitODE, ImplicitODE},
};
pub struct Euler<T>
where
T: Real,
{
i1: T,
i2: T,
i3: T,
}
impl<T> Default for Euler<T>
where
T: Real,
{
fn default() -> Euler<T> {
Euler {
i1: T::from_f64(0.5),
i2: T::from_f64(2.0),
i3: T::from_f64(3.0),
}
}
}
impl<T> ExplicitODE<T> for Euler<T>
where
T: Real,
{
fn ode(&self, x: &T, y: &Vector<T>) -> Vector<T>
where
T: Real,
{
let y_1s: T = ((self.i2 - self.i3) / self.i1) * (y[1] * y[2]);
let y_2s: T = ((self.i3 - self.i1) / self.i2) * (y[2] * y[0]);
let f: T = if *x >= T::from_f64(3.0) * T::pi() && *x <= T::from_f64(4.0) * T::pi() {
T::from_f64(0.25) * x.sin() * x.sin()
} else {
T::zero()
};
let y_3s: T = ((self.i1 - self.i2) / self.i3) * (y[0] * y[1]) + f;
vector![y_1s; y_2s; y_3s]
}
}
impl<T> ImplicitODE<T> for Euler<T>
where
T: Real,
{
fn ode(&self, x: &T, y: &Vector<T>) -> Vector<T> {
let a: T = (self.i2 - self.i3) / self.i1;
let b: T = (self.i3 - self.i1) / self.i2;
let c: T = (self.i1 - self.i2) / self.i3;
let y_1s: T = a * (y[1] * y[2]);
let y_2s: T = b * (y[2] * y[0]);
let f: T = if *x >= T::from_f64(3.0) * T::pi() && *x <= T::from_f64(4.0) * T::pi() {
T::from_f64(0.25) * x.sin() * x.sin()
} else {
T::zero()
};
let y_3s: T = c * (y[0] * y[1]) + f;
vector![y_1s; y_2s; y_3s]
}
fn jacobian(&self, _x: &T, y: &Vector<T>) -> General<T> {
let a: T = (self.i2 - self.i3) / self.i1;
let b: T = (self.i3 - self.i1) / self.i2;
let c: T = (self.i1 - self.i2) / self.i3;
matrix![T::zero(), a * y[2], a * y[1];
b * y[2], T::zero(), b * y[0];
c * y[1], c * y[0], T::zero()]
}
}