#[cfg(test)]
pub mod test;
use crate::{
defeat_message,
math::{
Rank2, Tensor, TensorRank1, TensorRank1List, TensorRank1List2D, TensorRank1RefVec,
TensorRank1Vec, TensorRank1Vec2D, TensorRank2, TensorRank2List, TensorRank2List2D,
TensorRank2Vec, TensorRank2Vec2D, TensorRank4, TensorRank4List, TensorRank4Vec,
},
};
use std::fmt::{self, Debug, Display, Formatter};
pub use crate::math::Scalar;
pub enum DeformationError {
InvalidJacobian(Scalar),
}
impl Debug for DeformationError {
fn fmt(&self, f: &mut Formatter<'_>) -> fmt::Result {
let error = match self {
Self::InvalidJacobian(jacobian) => {
format!("\x1b[1;91mInvalid Jacobian: {jacobian:.6e}.\x1b[0;91m")
}
};
write!(f, "\n{error}\n\x1b[0;2;31m{}\x1b[0m\n", defeat_message())
}
}
impl Display for DeformationError {
fn fmt(&self, f: &mut Formatter<'_>) -> fmt::Result {
let error = match self {
Self::InvalidJacobian(jacobian) => {
format!("\x1b[1;91mInvalid Jacobian: {jacobian:.6e}.\x1b[0;91m")
}
};
write!(f, "{error}\x1b[0m")
}
}
pub trait Deformation<const I: usize, const J: usize> {
fn jacobian(&self) -> Result<Scalar, DeformationError>;
fn left_cauchy_green(&self) -> TensorRank2<3, I, I>;
fn right_cauchy_green(&self) -> TensorRank2<3, J, J>;
}
impl<const I: usize, const J: usize> Deformation<I, J> for DeformationGradientGeneral<I, J> {
fn jacobian(&self) -> Result<Scalar, DeformationError> {
let jacobian = self.determinant();
if jacobian > 0.0 {
Ok(jacobian)
} else {
Err(DeformationError::InvalidJacobian(jacobian))
}
}
fn left_cauchy_green(&self) -> TensorRank2<3, I, I> {
self.iter()
.map(|deformation_gradient_i| {
self.iter()
.map(|deformation_gradient_j| deformation_gradient_i * deformation_gradient_j)
.collect()
})
.collect()
}
fn right_cauchy_green(&self) -> TensorRank2<3, J, J> {
let deformation_gradient_transpose = self.transpose();
deformation_gradient_transpose
.iter()
.map(|deformation_gradient_transpose_i| {
deformation_gradient_transpose
.iter()
.map(|deformation_gradient_transpose_j| {
deformation_gradient_transpose_i * deformation_gradient_transpose_j
})
.collect()
})
.collect()
}
}
pub type Basis = TensorRank1List<3, 1, 3>;
pub type Bases<const N: usize> = TensorRank1List2D<3, 1, 3, N>;
pub type CauchyStress = TensorRank2<3, 1, 1>;
pub type CauchyStresses<const W: usize> = TensorRank2List<3, 1, 1, W>;
pub type CauchyTangentStiffness = TensorRank4<3, 1, 1, 1, 0>;
pub type CauchyTangentStiffness1 = TensorRank4<3, 1, 1, 1, 2>;
pub type CauchyTangentStiffnessElastic = TensorRank4<3, 1, 1, 1, 2>;
pub type CauchyRateTangentStiffness = TensorRank4<3, 1, 1, 1, 0>;
pub type Coordinate<const I: usize> = TensorRank1<3, I>;
pub type CoordinateList<const I: usize, const N: usize> = TensorRank1List<3, I, N>;
pub type Coordinates<const I: usize> = TensorRank1Vec<3, I>;
pub type CoordinatesRef<'a, const I: usize> = TensorRank1RefVec<'a, 3, I>;
pub type CurrentCoordinate = TensorRank1<3, 1>;
pub type CurrentCoordinates<const W: usize> = TensorRank1List<3, 1, W>;
pub type CurrentCoordinatesRef<'a> = TensorRank1RefVec<'a, 3, 1>;
pub type CurrentVelocity = TensorRank1<3, 1>;
pub type DeformationGradient = TensorRank2<3, 1, 0>;
pub type DeformationGradient2 = TensorRank2<3, 2, 0>;
pub type DeformationGradientElastic = TensorRank2<3, 1, 2>;
pub type DeformationGradientGeneral<const I: usize, const J: usize> = TensorRank2<3, I, J>;
pub type DeformationGradientPlastic = TensorRank2<3, 2, 0>;
pub type DeformationGradientRate = TensorRank2<3, 1, 0>;
pub type DeformationGradientRatePlastic = TensorRank2<3, 2, 0>;
pub type DeformationGradientList<const W: usize> = TensorRank2List<3, 1, 0, W>;
pub type DeformationGradientRateList<const W: usize> = TensorRank2List<3, 1, 0, W>;
pub type DeformationGradients = TensorRank2Vec<3, 1, 0>;
pub type DeformationGradientsPlastic = TensorRank2Vec<3, 2, 0>;
pub type DeformationGradientRates = TensorRank2Vec<3, 1, 0>;
pub type DeformationGradientRatesPlastic = TensorRank2Vec<3, 2, 0>;
pub type Displacement = TensorRank1<3, 1>;
pub type FirstPiolaKirchhoffStress = TensorRank2<3, 1, 0>;
pub type FirstPiolaKirchhoffStress1 = TensorRank2<3, 1, 2>;
pub type FirstPiolaKirchhoffStress2 = TensorRank2<3, 2, 0>;
pub type FirstPiolaKirchhoffStressElastic = FirstPiolaKirchhoffStress1;
pub type FirstPiolaKirchhoffStressList<const N: usize> = TensorRank2List<3, 1, 0, N>;
pub type FirstPiolaKirchhoffStresses = TensorRank2Vec<3, 1, 0>;
pub type FirstPiolaKirchhoffTangentStiffness = TensorRank4<3, 1, 0, 1, 0>;
pub type FirstPiolaKirchhoffTangentStiffness1 = TensorRank4<3, 1, 2, 1, 2>;
pub type FirstPiolaKirchhoffTangentStiffness2 = TensorRank4<3, 2, 0, 2, 0>;
pub type FirstPiolaKirchhoffTangentStiffnessElastic = FirstPiolaKirchhoffTangentStiffness1;
pub type FirstPiolaKirchhoffTangentStiffnessList<const N: usize> =
TensorRank4List<3, 1, 0, 1, 0, N>;
pub type FirstPiolaKirchhoffTangentStiffnesses = TensorRank4Vec<3, 1, 0, 1, 0>;
pub type FirstPiolaKirchhoffRateTangentStiffness = TensorRank4<3, 1, 0, 1, 0>;
pub type FirstPiolaKirchhoffRateTangentStiffnesses<const W: usize> =
TensorRank4List<3, 1, 0, 1, 0, W>;
pub type Force = TensorRank1<3, 1>;
pub type ForceList<const N: usize> = TensorRank1List<3, 1, N>;
pub type Forces = TensorRank1Vec<3, 1>;
pub type FrameSpin = TensorRank2<3, 1, 1>;
pub type HeatFlux = TensorRank1<3, 0>;
pub type HeatFluxes<const N: usize> = TensorRank1List<3, 0, N>;
pub type HeatFluxTangent = TensorRank2<3, 0, 0>;
pub type HeatFluxTangents<const N: usize> = TensorRank2List<3, 0, 0, N>;
pub type LeftCauchyGreenDeformation = TensorRank2<3, 1, 1>;
pub type MandelStress = TensorRank2<3, 0, 0>;
pub type MandelStressElastic = TensorRank2<3, 2, 2>;
pub type Normal = TensorRank1<3, 1>;
pub type Normals<const N: usize> = TensorRank1List<3, 1, N>;
pub type NormalGradients<const O: usize, const P: usize> = TensorRank2List2D<3, 1, 1, O, P>;
pub type NormalRate = TensorRank1<3, 1>;
pub type NormalRates<const N: usize> = TensorRank1List<3, 1, N>;
pub type ReferenceCoordinate = TensorRank1<3, 0>;
pub type ReferenceCoordinates<const W: usize> = TensorRank1List<3, 0, W>;
pub type ReferenceNormal = TensorRank1<3, 0>;
pub type ReferenceNormals<const N: usize> = TensorRank1List<3, 0, N>;
pub type RightCauchyGreenDeformation = TensorRank2<3, 0, 0>;
pub type RotationCurrentConfiguration = TensorRank2<3, 1, 1>;
pub type RotationCurrentConfigurationList<const N: usize> = TensorRank2List<3, 1, 1, N>;
pub type RotationRateCurrentConfiguration = TensorRank2<3, 1, 1>;
pub type RotationReferenceConfiguration = TensorRank2<3, 0, 0>;
pub type Separation = Displacement;
pub type SecondPiolaKirchhoffStress = TensorRank2<3, 0, 0>;
pub type SecondPiolaKirchhoffStressElastic = TensorRank2<3, 2, 2>;
pub type SecondPiolaKirchhoffTangentStiffness = TensorRank4<3, 0, 0, 1, 0>;
pub type SecondPiolaKirchhoffTangentStiffnessElastic = TensorRank4<3, 2, 2, 1, 2>;
pub type SecondPiolaKirchhoffRateTangentStiffness = TensorRank4<3, 0, 0, 1, 0>;
pub type Stiffness = TensorRank2<3, 1, 1>;
pub type StiffnessList<const N: usize> = TensorRank2List<3, 1, 1, N>;
pub type StiffnessList2D<const N: usize> = TensorRank2List2D<3, 1, 1, N, N>;
pub type Stiffnesses = TensorRank2Vec2D<3, 1, 1>;
pub type StretchingRate = TensorRank2<3, 1, 1>;
pub type StretchingRatePlastic = TensorRank2<3, 2, 2>;
pub type SurfaceBasis<const I: usize> = TensorRank1List<3, I, 2>;
pub type SurfaceBases<const I: usize, const N: usize> = TensorRank1List2D<3, I, 2, N>;
pub type TemperatureGradient = TensorRank1<3, 0>;
pub type TemperatureGradients<const N: usize> = TensorRank1List<3, 0, N>;
pub type Times = crate::math::Vector;
pub type Traction = TensorRank1<3, 1>;
pub type TractionList<const N: usize> = TensorRank1List<3, 1, N>;
pub type Vector<const I: usize> = TensorRank1<3, I>;
pub type VectorList<const I: usize, const W: usize> = TensorRank1List<3, I, W>;
pub type VectorList2D<const I: usize, const W: usize, const X: usize> =
TensorRank1List2D<3, I, W, X>;
pub type Vectors<const I: usize> = TensorRank1Vec<3, I>;
pub type Vectors2D<const I: usize> = TensorRank1Vec2D<3, I>;