conspire 0.6.0

The Rust interface to conspire.
Documentation
use crate::{
    constitutive::solid::hyperviscoelastic::Hyperviscoelastic,
    fem::{
        NodalCoordinates,
        block::{
            Block, FiniteElementBlockError,
            element::solid::hyperviscoelastic::HyperviscoelasticFiniteElement,
            solid::elastic_hyperviscous::ElasticHyperviscousFiniteElementBlock,
        },
    },
    math::Scalar,
};

pub trait HyperviscoelasticFiniteElementBlock<
    C,
    F,
    const G: usize,
    const M: usize,
    const N: usize,
    const P: usize,
> where
    C: Hyperviscoelastic,
    F: HyperviscoelasticFiniteElement<C, G, M, N, P>,
    Self: ElasticHyperviscousFiniteElementBlock<C, F, G, M, N, P>,
{
    fn helmholtz_free_energy(
        &self,
        nodal_coordinates: &NodalCoordinates,
    ) -> Result<Scalar, FiniteElementBlockError>;
}

impl<C, F, const G: usize, const M: usize, const N: usize, const P: usize>
    HyperviscoelasticFiniteElementBlock<C, F, G, M, N, P> for Block<C, F, G, M, N, P>
where
    C: Hyperviscoelastic,
    F: HyperviscoelasticFiniteElement<C, G, M, N, P>,
    Self: ElasticHyperviscousFiniteElementBlock<C, F, G, M, N, P>,
{
    fn helmholtz_free_energy(
        &self,
        nodal_coordinates: &NodalCoordinates,
    ) -> Result<Scalar, FiniteElementBlockError> {
        match self
            .elements()
            .iter()
            .zip(self.connectivity())
            .map(|(element, nodes)| {
                element.helmholtz_free_energy(
                    self.constitutive_model(),
                    &Self::element_coordinates(nodal_coordinates, nodes),
                )
            })
            .sum()
        {
            Ok(helmholtz_free_energy) => Ok(helmholtz_free_energy),
            Err(error) => Err(FiniteElementBlockError::Upstream(
                format!("{error}"),
                format!("{self:?}"),
            )),
        }
    }
}