use std::collections::{HashSet, VecDeque};
use anyhow::format_err;
use ndarray::Array2;
use num_complex::Complex;
use crate::group::GroupProperties;
use crate::permutation::Permutation;
use crate::symmetry::symmetry_element::SymmetryOperation;
use crate::symmetry::symmetry_transformation::{
ComplexConjugationTransformable, DefaultTimeReversalTransformable, SpatialUnitaryTransformable,
SpinUnitaryTransformable, SymmetryTransformable, TimeReversalTransformable,
TransformationError,
};
use crate::target::noci::basis::{EagerBasis, OrbitBasis};
impl<'g, G, I> SpatialUnitaryTransformable for OrbitBasis<'g, G, I>
where
G: GroupProperties + Clone,
I: SpatialUnitaryTransformable,
{
fn transform_spatial_mut(
&mut self,
_: &Array2<f64>,
_: Option<&Permutation<usize>>,
) -> Result<&mut Self, TransformationError> {
Err(TransformationError("Transforming an orbit basis by an arbitrary spatial transformation matrix is not supported.".to_string()))
}
}
impl<'g, G, I> SpinUnitaryTransformable for OrbitBasis<'g, G, I>
where
G: GroupProperties + Clone,
I: SpinUnitaryTransformable,
{
fn transform_spin_mut(
&mut self,
_: &Array2<Complex<f64>>,
) -> Result<&mut Self, TransformationError> {
Err(TransformationError("Transforming an orbit basis by an arbitrary spin transformation matrix is not supported.".to_string()))
}
}
impl<'g, G, I> ComplexConjugationTransformable for OrbitBasis<'g, G, I>
where
G: GroupProperties + Clone,
I: ComplexConjugationTransformable,
{
fn transform_cc_mut(&mut self) -> Result<&mut Self, TransformationError> {
Err(TransformationError(
"Transforming an orbit basis by an explicit complex conjugation is not supported."
.to_string(),
))
}
}
impl<'g, G, I> DefaultTimeReversalTransformable for OrbitBasis<'g, G, I>
where
G: GroupProperties + Clone,
I: DefaultTimeReversalTransformable,
{
}
impl<'g, G, I> SymmetryTransformable for OrbitBasis<'g, G, I>
where
G: GroupProperties<GroupElement = SymmetryOperation> + Clone,
I: SymmetryTransformable,
OrbitBasis<'g, G, I>: TimeReversalTransformable,
{
fn sym_permute_sites_spatial(
&self,
symop: &SymmetryOperation,
) -> Result<Permutation<usize>, TransformationError> {
let mut perms = self
.origins
.iter()
.map(|origin| origin.sym_permute_sites_spatial(symop))
.collect::<Result<HashSet<_>, _>>()?;
if perms.len() == 1 {
perms.drain().next().ok_or(TransformationError(
"Unable to retrieve the site permutation.".to_string(),
))
} else {
Err(TransformationError(
"Mismatched site permutations across the origins.".to_string(),
))
}
}
fn sym_transform_spatial_mut(
&mut self,
symop: &SymmetryOperation,
) -> Result<&mut Self, TransformationError> {
if let Some(prefactors) = self.prefactors.as_mut() {
prefactors.push_front((symop.clone(), |g, item| {
item.sym_transform_spatial(g)
.map_err(|err| format_err!(err))
}));
} else {
let mut v = VecDeque::<(
G::GroupElement,
fn(&G::GroupElement, &I) -> Result<I, anyhow::Error>,
)>::new();
v.push_front((symop.clone(), |g, item| {
item.sym_transform_spatial(g)
.map_err(|err| format_err!(err))
}));
self.prefactors = Some(v);
}
Ok(self)
}
fn sym_transform_spatial_with_spintimerev_mut(
&mut self,
symop: &SymmetryOperation,
) -> Result<&mut Self, TransformationError> {
if let Some(prefactors) = self.prefactors.as_mut() {
prefactors.push_front((symop.clone(), |g, item| {
item.sym_transform_spatial_with_spintimerev(g)
.map_err(|err| format_err!(err))
}));
} else {
let mut v = VecDeque::<(
G::GroupElement,
fn(&G::GroupElement, &I) -> Result<I, anyhow::Error>,
)>::new();
v.push_front((symop.clone(), |g, item| {
item.sym_transform_spatial_with_spintimerev(g)
.map_err(|err| format_err!(err))
}));
self.prefactors = Some(v);
}
Ok(self)
}
fn sym_transform_spin_mut(
&mut self,
symop: &SymmetryOperation,
) -> Result<&mut Self, TransformationError> {
if let Some(prefactors) = self.prefactors.as_mut() {
prefactors.push_front((symop.clone(), |g, item| {
item.sym_transform_spin(g).map_err(|err| format_err!(err))
}));
} else {
let mut v = VecDeque::<(
G::GroupElement,
fn(&G::GroupElement, &I) -> Result<I, anyhow::Error>,
)>::new();
v.push_front((symop.clone(), |g, item| {
item.sym_transform_spin(g).map_err(|err| format_err!(err))
}));
self.prefactors = Some(v);
}
Ok(self)
}
fn sym_transform_spin_spatial_mut(
&mut self,
symop: &SymmetryOperation,
) -> Result<&mut Self, TransformationError> {
if let Some(prefactors) = self.prefactors.as_mut() {
prefactors.push_front((symop.clone(), |g, item| {
item.sym_transform_spin_spatial(g)
.map_err(|err| format_err!(err))
}));
} else {
let mut v = VecDeque::<(
G::GroupElement,
fn(&G::GroupElement, &I) -> Result<I, anyhow::Error>,
)>::new();
v.push_front((symop.clone(), |g, item| {
item.sym_transform_spin_spatial(g)
.map_err(|err| format_err!(err))
}));
self.prefactors = Some(v);
}
Ok(self)
}
}
impl<I> SpatialUnitaryTransformable for EagerBasis<I>
where
I: SpatialUnitaryTransformable,
{
fn transform_spatial_mut(
&mut self,
rmat: &Array2<f64>,
perm: Option<&Permutation<usize>>,
) -> Result<&mut Self, TransformationError> {
self.elements
.iter_mut()
.map(|origin| origin.transform_spatial_mut(rmat, perm))
.collect::<Result<Vec<_>, _>>()?;
Ok(self)
}
}
impl<I> SpinUnitaryTransformable for EagerBasis<I>
where
I: SpinUnitaryTransformable,
{
fn transform_spin_mut(
&mut self,
dmat: &Array2<Complex<f64>>,
) -> Result<&mut Self, TransformationError> {
self.elements
.iter_mut()
.map(|origin| origin.transform_spin_mut(dmat))
.collect::<Result<Vec<_>, _>>()?;
Ok(self)
}
}
impl<I> ComplexConjugationTransformable for EagerBasis<I>
where
I: ComplexConjugationTransformable,
{
fn transform_cc_mut(&mut self) -> Result<&mut Self, TransformationError> {
for origin in self.elements.iter_mut() {
origin.transform_cc_mut()?;
}
Ok(self)
}
}
impl<I> DefaultTimeReversalTransformable for EagerBasis<I> where
I: DefaultTimeReversalTransformable + Clone
{
}
impl<I> SymmetryTransformable for EagerBasis<I>
where
I: SymmetryTransformable,
EagerBasis<I>: TimeReversalTransformable,
{
fn sym_permute_sites_spatial(
&self,
symop: &SymmetryOperation,
) -> Result<Permutation<usize>, TransformationError> {
let mut perms = self
.elements
.iter()
.map(|origin| origin.sym_permute_sites_spatial(symop))
.collect::<Result<HashSet<_>, _>>()?;
if perms.len() == 1 {
perms.drain().next().ok_or(TransformationError(
"Unable to retrieve the site permutation.".to_string(),
))
} else {
Err(TransformationError(
"Mismatched site permutations across the elements in the basis.".to_string(),
))
}
}
}