organic/src/iter/ast_node_iter.rs

165 lines
5.4 KiB
Rust
Raw Normal View History

use std::marker::PhantomData;
2023-09-27 15:07:26 -04:00
use super::ast_node::AstNode;
2023-09-27 15:47:01 -04:00
use super::macros::children_iter;
use super::macros::empty_iter;
2023-09-27 15:17:56 -04:00
use crate::types::Bold;
use crate::types::Code;
use crate::types::Document;
use crate::types::DocumentElement;
use crate::types::Element;
use crate::types::Heading;
2023-09-27 15:28:43 -04:00
use crate::types::Italic;
use crate::types::Object;
use crate::types::PlainText;
use crate::types::RadioLink;
use crate::types::RegularLink;
use crate::types::Section;
use crate::types::StrikeThrough;
use crate::types::Underline;
2023-09-27 15:47:01 -04:00
use crate::types::Verbatim;
2023-09-27 15:47:01 -04:00
/// Iterator over the AST nodes contained within the starting node.
///
/// This only iterates over the children, not the starting node itself. So an AstNodeIter::PlainList would only return PlainListItems, not the PlainList.
///
/// This only iterates over AST nodes, so an AstNodeIter::Heading would iterate over both the title and section contents, but it would not iterate over simple strings like the TODO keyword or priority.
2023-09-27 15:07:26 -04:00
pub enum AstNodeIter<'r, 's> {
// Document Nodes
Document(DocumentIter<'r, 's>),
Heading(HeadingIter<'r, 's>),
Section(SectionIter<'r, 's>),
// Elements
2023-09-27 15:07:26 -04:00
// Paragraph(ParagraphIter<'r, 's>),
// PlainList(PlainListIter<'r, 's>),
// GreaterBlock(GreaterBlockIter<'r, 's>),
// DynamicBlock(DynamicBlockIter<'r, 's>),
// FootnoteDefinition(FootnoteDefinitionIter<'r, 's>),
// Comment(CommentIter<'r, 's>),
// Drawer(DrawerIter<'r, 's>),
// PropertyDrawer(PropertyDrawerIter<'r, 's>),
// Table(TableIter<'r, 's>),
// VerseBlock(VerseBlockIter<'r, 's>),
// CommentBlock(CommentBlockIter<'r, 's>),
// ExampleBlock(ExampleBlockIter<'r, 's>),
// ExportBlock(ExportBlockIter<'r, 's>),
// SrcBlock(SrcBlockIter<'r, 's>),
// Clock(ClockIter<'r, 's>),
// DiarySexp(DiarySexpIter<'r, 's>),
// Planning(PlanningIter<'r, 's>),
// FixedWidthArea(FixedWidthAreaIter<'r, 's>),
// HorizontalRule(HorizontalRuleIter<'r, 's>),
// Keyword(KeywordIter<'r, 's>),
// BabelCall(BabelCallIter<'r, 's>),
// LatexEnvironment(LatexEnvironmentIter<'r, 's>),
// Objects
2023-09-27 15:07:26 -04:00
Bold(BoldIter<'r, 's>),
2023-09-27 15:28:43 -04:00
Italic(ItalicIter<'r, 's>),
Underline(UnderlineIter<'r, 's>),
StrikeThrough(StrikeThroughIter<'r, 's>),
Code(CodeIter<'r, 's>),
2023-09-27 15:47:01 -04:00
Verbatim(VerbatimIter<'r, 's>),
PlainText(PlainTextIter<'r, 's>),
RegularLink(RegularLinkIter<'r, 's>),
RadioLink(RadioLinkIter<'r, 's>),
2023-09-27 15:07:26 -04:00
// RadioTarget(RadioTargetIter<'r, 's>),
// PlainLink(PlainLinkIter<'r, 's>),
// AngleLink(AngleLinkIter<'r, 's>),
// OrgMacro(OrgMacroIter<'r, 's>),
// Entity(EntityIter<'r, 's>),
// LatexFragment(LatexFragmentIter<'r, 's>),
// ExportSnippet(ExportSnippetIter<'r, 's>),
// FootnoteReference(FootnoteReferenceIter<'r, 's>),
// Citation(CitationIter<'r, 's>),
// CitationReference(CitationReferenceIter<'r, 's>),
// InlineBabelCall(InlineBabelCallIter<'r, 's>),
// InlineSourceBlock(InlineSourceBlockIter<'r, 's>),
// LineBreak(LineBreakIter<'r, 's>),
// Target(TargetIter<'r, 's>),
// StatisticsCookie(StatisticsCookieIter<'r, 's>),
// Subscript(SubscriptIter<'r, 's>),
// Superscript(SuperscriptIter<'r, 's>),
// Timestamp(TimestampIter<'r, 's>),
}
pub struct DocumentIter<'r, 's> {
zeroth_section_next: std::option::Iter<'r, Section<'s>>,
children_next: std::slice::Iter<'r, Heading<'s>>,
}
impl<'r, 's> Iterator for DocumentIter<'r, 's> {
type Item = AstNode<'r, 's>;
fn next(&mut self) -> Option<Self::Item> {
self.zeroth_section_next
.next()
.map(Into::<AstNode>::into)
.or_else(|| self.children_next.next().map(Into::<AstNode>::into))
}
}
impl<'r, 's> IntoIterator for &'r Document<'s> {
type Item = AstNode<'r, 's>;
type IntoIter = DocumentIter<'r, 's>;
fn into_iter(self) -> Self::IntoIter {
DocumentIter {
zeroth_section_next: self.zeroth_section.iter(),
children_next: self.children.iter(),
}
}
}
pub struct HeadingIter<'r, 's> {
title_next: std::slice::Iter<'r, Object<'s>>,
children_next: std::slice::Iter<'r, DocumentElement<'s>>,
2023-09-27 15:07:26 -04:00
}
impl<'r, 's> Iterator for HeadingIter<'r, 's> {
2023-09-27 15:07:26 -04:00
type Item = AstNode<'r, 's>;
fn next(&mut self) -> Option<Self::Item> {
self.title_next
.next()
.map(Into::<AstNode>::into)
.or_else(|| self.children_next.next().map(Into::<AstNode>::into))
2023-09-27 15:17:56 -04:00
}
}
impl<'r, 's> IntoIterator for &'r Heading<'s> {
2023-09-27 15:17:56 -04:00
type Item = AstNode<'r, 's>;
type IntoIter = HeadingIter<'r, 's>;
2023-09-27 15:17:56 -04:00
fn into_iter(self) -> Self::IntoIter {
HeadingIter {
title_next: self.title.iter(),
children_next: self.children.iter(),
2023-09-27 15:17:56 -04:00
}
2023-09-27 15:07:26 -04:00
}
}
2023-09-27 15:28:43 -04:00
children_iter!(Section<'s>, SectionIter, std::slice::Iter<'r, Element<'s>>);
children_iter!(Bold<'s>, BoldIter, std::slice::Iter<'r, Object<'s>>);
2023-09-27 15:47:01 -04:00
children_iter!(Italic<'s>, ItalicIter, std::slice::Iter<'r, Object<'s>>);
children_iter!(
Underline<'s>,
UnderlineIter,
std::slice::Iter<'r, Object<'s>>
);
2023-09-27 15:47:01 -04:00
children_iter!(
StrikeThrough<'s>,
StrikeThroughIter,
std::slice::Iter<'r, Object<'s>>
);
empty_iter!(Code<'s>, CodeIter);
2023-09-27 15:47:01 -04:00
empty_iter!(Verbatim<'s>, VerbatimIter);
empty_iter!(PlainText<'s>, PlainTextIter);
empty_iter!(RegularLink<'s>, RegularLinkIter);
children_iter!(
RadioLink<'s>,
RadioLinkIter,
std::slice::Iter<'r, Object<'s>>
);