1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94
/// Helper macro for building a tree in place.
///
/// # Examples
///
/// ```
/// use syntree::{Span, TreeBuilder};
///
/// #[derive(Debug, Clone, Copy, PartialEq, Eq)]
/// enum Syntax {
/// Root,
/// Number,
/// Lit,
/// Whitespace,
/// }
///
/// # fn main() -> anyhow::Result<()> {
/// let mut b = TreeBuilder::new();
///
/// b.open(Syntax::Root);
///
/// b.open(Syntax::Number);
/// b.token(Syntax::Lit, 1);
/// b.close()?;
///
/// b.token(Syntax::Whitespace, 3);
///
/// b.open(Syntax::Number);
/// b.token(Syntax::Lit, 2);
/// b.token(Syntax::Lit, 2);
/// b.close()?;
///
/// b.close()?;
///
/// let tree = b.build()?;
///
/// let expected = syntree::tree! {
/// Syntax::Root => {
/// Syntax::Number => {
/// (Syntax::Lit, 1)
/// },
/// (Syntax::Whitespace, 3),
/// Syntax::Number => {
/// (Syntax::Lit, 2),
/// (Syntax::Lit, 2)
/// }
/// }
/// };
///
/// assert_eq!(expected, tree);
/// # Ok(()) }
/// ```
#[macro_export]
macro_rules! tree {
(@o $b:ident,) => {};
(@o $b:ident, ($expr:expr, $len:expr) $(,)?) => {{
$b.token($expr, $len);
}};
(@o $b:ident, ($expr:expr, $len:expr), $($rest:tt)*) => {{
$b.token($expr, $len);
$crate::tree!(@o $b, $($rest)*);
}};
(@o $b:ident, $expr:expr $(,)?) => {{
$b.open($expr);
$b.close()?;
}};
(@o $b:ident, $expr:expr, $($rest:tt)*) => {{
$b.open($expr);
$b.close()?;
$crate::tree!(@o $b, $($rest)*);
}};
(@o $b:ident, $expr:expr => { $($tt:tt)* } $(,)?) => {{
$b.open($expr);
$crate::tree!(@o $b, $($tt)*);
$b.close()?;
}};
(@o $b:ident, $expr:expr => { $($tt:tt)* }, $($rest:tt)*) => {{
$b.open($expr);
$crate::tree!(@o $b, $($tt)*);
$b.close()?;
$crate::tree!(@o $b, $($rest)*);
}};
($($tt:tt)*) => {{
let mut b = $crate::TreeBuilder::new();
$crate::tree!(@o b, $($tt)*);
b.build()?
}};
}
