use std::ops::Not;
use super::{
ALWAYS_FALSE_NAME,
ALWAYS_TRUE_NAME,
Arc,
BoxStatefulBiPredicate,
Mutex,
RcStatefulBiPredicate,
StatefulBiPredicate,
impl_arc_conversions,
impl_closure_trait,
impl_predicate_clone,
impl_predicate_common_methods,
impl_predicate_debug_display,
};
type ArcStatefulBiPredicateFn<T, U> = Arc<Mutex<dyn FnMut(&T, &U) -> bool + Send + 'static>>;
pub struct ArcStatefulBiPredicate<T, U> {
pub(super) function: ArcStatefulBiPredicateFn<T, U>,
pub(super) name: Option<String>,
}
impl<T, U> ArcStatefulBiPredicate<T, U> {
impl_predicate_common_methods!(
ArcStatefulBiPredicate<T, U>,
(FnMut(&T, &U) -> bool + Send + 'static),
|f| Arc::new(Mutex::new(f))
);
#[inline]
pub fn and<P>(&self, mut other: P) -> ArcStatefulBiPredicate<T, U>
where
P: StatefulBiPredicate<T, U> + Send + 'static,
T: 'static,
U: 'static,
{
let self_fn = self.function.clone();
ArcStatefulBiPredicate::new(move |first, second| {
let matched = (self_fn.lock())(first, second);
matched && other.test(first, second)
})
}
#[inline]
pub fn or<P>(&self, mut other: P) -> ArcStatefulBiPredicate<T, U>
where
P: StatefulBiPredicate<T, U> + Send + 'static,
T: 'static,
U: 'static,
{
let self_fn = self.function.clone();
ArcStatefulBiPredicate::new(move |first, second| {
let matched = (self_fn.lock())(first, second);
matched || other.test(first, second)
})
}
#[inline]
pub fn nand<P>(&self, mut other: P) -> ArcStatefulBiPredicate<T, U>
where
P: StatefulBiPredicate<T, U> + Send + 'static,
T: 'static,
U: 'static,
{
let self_fn = self.function.clone();
ArcStatefulBiPredicate::new(move |first, second| {
let matched = (self_fn.lock())(first, second);
!(matched && other.test(first, second))
})
}
#[inline]
pub fn xor<P>(&self, mut other: P) -> ArcStatefulBiPredicate<T, U>
where
P: StatefulBiPredicate<T, U> + Send + 'static,
T: 'static,
U: 'static,
{
let self_fn = self.function.clone();
ArcStatefulBiPredicate::new(move |first, second| {
let matched = (self_fn.lock())(first, second);
matched ^ other.test(first, second)
})
}
#[inline]
pub fn nor<P>(&self, mut other: P) -> ArcStatefulBiPredicate<T, U>
where
P: StatefulBiPredicate<T, U> + Send + 'static,
T: 'static,
U: 'static,
{
let self_fn = self.function.clone();
ArcStatefulBiPredicate::new(move |first, second| {
let matched = (self_fn.lock())(first, second);
!(matched || other.test(first, second))
})
}
}
impl<T, U> Not for ArcStatefulBiPredicate<T, U>
where
T: 'static,
U: 'static,
{
type Output = ArcStatefulBiPredicate<T, U>;
fn not(self) -> Self::Output {
let function = self.function;
ArcStatefulBiPredicate::new(move |first, second| !((function.lock())(first, second)))
}
}
impl<T, U> Not for &ArcStatefulBiPredicate<T, U>
where
T: 'static,
U: 'static,
{
type Output = ArcStatefulBiPredicate<T, U>;
fn not(self) -> Self::Output {
let function = self.function.clone();
ArcStatefulBiPredicate::new(move |first, second| !((function.lock())(first, second)))
}
}
impl_predicate_clone!(ArcStatefulBiPredicate<T, U>);
impl_predicate_debug_display!(ArcStatefulBiPredicate<T, U>);
impl<T, U> StatefulBiPredicate<T, U> for ArcStatefulBiPredicate<T, U> {
fn test(&mut self, first: &T, second: &U) -> bool {
(self.function.lock())(first, second)
}
impl_arc_conversions!(
ArcStatefulBiPredicate<T, U>,
BoxStatefulBiPredicate,
RcStatefulBiPredicate,
FnMut(first: &T, second: &U) -> bool
);
}
impl_closure_trait!(
StatefulBiPredicate<T, U>,
test,
FnMut(first: &T, second: &U) -> bool
);