macro_rules! impl_box_conditional_consumer {
(@let_consumer Consumer, $name:ident, $value:expr) => {
let $name = $value;
};
(@let_consumer ConsumerOnce, $name:ident, $value:expr) => {
let $name = $value;
};
(@let_consumer StatefulConsumer, $name:ident, $value:expr) => {
let mut $name = $value;
};
(@let_consumer BiConsumer, $name:ident, $value:expr) => {
let $name = $value;
};
(@let_consumer BiConsumerOnce, $name:ident, $value:expr) => {
let $name = $value;
};
(@let_consumer StatefulBiConsumer, $name:ident, $value:expr) => {
let mut $name = $value;
};
(
$struct_name:ident<$t:ident>,
$consumer_type:ident,
$consumer_trait:ident
) => {
impl<$t> $struct_name<$t> {
pub fn and_then<C>(self, next: C) -> $consumer_type<$t>
where
$t: 'static,
C: $consumer_trait<$t> + 'static,
{
let predicate = self.predicate;
impl_box_conditional_consumer!(@let_consumer $consumer_trait, consumer, self.consumer);
impl_box_conditional_consumer!(@let_consumer $consumer_trait, next, next);
$consumer_type::new(move |t| {
if predicate.test(t) {
consumer.accept(t);
}
next.accept(t);
})
}
pub fn or_else<C>(self, else_consumer: C) -> $consumer_type<$t>
where
$t: 'static,
C: $consumer_trait<$t> + 'static,
{
let predicate = self.predicate;
impl_box_conditional_consumer!(@let_consumer $consumer_trait, then_consumer, self.consumer);
impl_box_conditional_consumer!(@let_consumer $consumer_trait, else_consumer, else_consumer);
$consumer_type::new(move |t| {
if predicate.test(t) {
then_consumer.accept(t);
} else {
else_consumer.accept(t);
}
})
}
}
};
(
$struct_name:ident<$t:ident, $u:ident>,
$consumer_type:ident,
$consumer_trait:ident
) => {
impl<$t, $u> $struct_name<$t, $u> {
pub fn and_then<C>(self, next: C) -> $consumer_type<$t, $u>
where
$t: 'static,
$u: 'static,
C: $consumer_trait<$t, $u> + 'static,
{
let predicate = self.predicate;
impl_box_conditional_consumer!(@let_consumer $consumer_trait, consumer, self.consumer);
impl_box_conditional_consumer!(@let_consumer $consumer_trait, next, next);
$consumer_type::new(move |t, u| {
if predicate.test(t, u) {
consumer.accept(t, u);
}
next.accept(t, u);
})
}
pub fn or_else<C>(self, else_consumer: C) -> $consumer_type<$t, $u>
where
$t: 'static,
$u: 'static,
C: $consumer_trait<$t, $u> + 'static,
{
let predicate = self.predicate;
impl_box_conditional_consumer!(@let_consumer $consumer_trait, then_consumer, self.consumer);
impl_box_conditional_consumer!(@let_consumer $consumer_trait, else_consumer, else_consumer);
$consumer_type::new(move |t, u| {
if predicate.test(t, u) {
then_consumer.accept(t, u);
} else {
else_consumer.accept(t, u);
}
})
}
}
};
}
pub(crate) use impl_box_conditional_consumer;