use nom::branch::alt; use nom::bytes::complete::tag; use nom::character::complete::anychar; use nom::combinator::map; use nom::combinator::peek; use nom::combinator::recognize; use nom::combinator::verify; use nom::multi::many_till; use super::org_source::OrgSource; use super::radio_link::RematchObject; use super::util::exit_matcher_parser; use crate::context::parser_with_context; use crate::context::RefContext; use crate::error::Res; use crate::types::Object; use crate::types::PlainText; pub fn plain_text( end_condition: F, ) -> impl for<'b, 'g, 'r, 's> Fn( RefContext<'b, 'g, 'r, 's>, OrgSource<'s>, ) -> Res, PlainText<'s>> where F: for<'bb, 'gg, 'rr, 'ss> Fn( RefContext<'bb, 'gg, 'rr, 'ss>, OrgSource<'ss>, ) -> Res, ()>, { move |context, input| _plain_text(&end_condition, context, input) } #[cfg_attr( feature = "tracing", tracing::instrument(ret, level = "debug", skip(end_condition)) )] fn _plain_text<'b, 'g, 'r, 's, F>( end_condition: F, context: RefContext<'b, 'g, 'r, 's>, input: OrgSource<'s>, ) -> Res, PlainText<'s>> where F: for<'bb, 'gg, 'rr, 'ss> Fn( RefContext<'bb, 'gg, 'rr, 'ss>, OrgSource<'ss>, ) -> Res, ()>, { let (remaining, source) = recognize(verify( many_till( anychar, peek(alt(( parser_with_context!(exit_matcher_parser)(context), recognize(parser_with_context!(end_condition)(context)), ))), ), |(children, _exit_contents)| !children.is_empty(), ))(input)?; Ok(( remaining, PlainText { source: source.into(), }, )) } impl<'x> RematchObject<'x> for PlainText<'x> { #[cfg_attr(feature = "tracing", tracing::instrument(ret, level = "debug"))] fn rematch_object<'b, 'g, 'r, 's>( &'x self, _context: RefContext<'b, 'g, 'r, 's>, input: OrgSource<'s>, ) -> Res, Object<'s>> { map(tag(self.source), |s| { Object::PlainText(PlainText { source: Into::<&str>::into(s), }) })(input) } } #[cfg(test)] mod tests { use nom::combinator::map; use super::*; use crate::context::Context; use crate::context::ContextElement; use crate::context::GlobalSettings; use crate::context::List; use crate::parser::object_parser::detect_standard_set_object_sans_plain_text; use crate::types::Source; #[test] fn plain_text_simple() { let input = OrgSource::new("foobarbaz"); let global_settings = GlobalSettings::default(); let initial_context = ContextElement::document_context(); let initial_context = Context::new(&global_settings, List::new(&initial_context)); let plain_text_matcher = parser_with_context!(plain_text( detect_standard_set_object_sans_plain_text ))(&initial_context); let (remaining, result) = map(plain_text_matcher, Object::PlainText)(input).unwrap(); assert_eq!(Into::<&str>::into(remaining), ""); assert_eq!(result.get_source(), Into::<&str>::into(input)); } }