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 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125
use crate::ast::prelude::*;
/// A struct item.
///
/// # Examples
///
/// ```
/// use rune::{ast, testing};
///
/// testing::roundtrip::<ast::ItemStruct>("struct Foo");
/// testing::roundtrip::<ast::ItemStruct>("struct Foo ( a, b, c )");
/// testing::roundtrip::<ast::ItemStruct>("struct Foo { a, b, c }");
/// testing::roundtrip::<ast::ItemStruct>("struct Foo { #[default_value = 1] a, b, c }");
/// testing::roundtrip::<ast::ItemStruct>("#[alpha] struct Foo ( #[default_value = \"x\" ] a, b, c )");
/// ```
#[derive(Debug, Clone, PartialEq, Eq, Parse, ToTokens, Spanned, Opaque)]
#[rune(parse = "meta_only")]
#[non_exhaustive]
pub struct ItemStruct {
/// Opaque identifier of the struct.
#[rune(id)]
pub(crate) id: Id,
/// The attributes for the struct
#[rune(iter, meta)]
pub attributes: Vec<ast::Attribute>,
/// The visibility of the `struct` item
#[rune(optional, meta)]
pub visibility: ast::Visibility,
/// The `struct` keyword.
pub struct_token: T![struct],
/// The identifier of the struct declaration.
pub ident: ast::Ident,
/// The body of the struct.
#[rune(optional)]
pub body: ItemStructBody,
}
impl ItemStruct {
/// If the struct declaration needs to be terminated with a semicolon.
pub(crate) fn needs_semi_colon(&self) -> bool {
self.body.needs_semi_colon()
}
}
item_parse!(Struct, ItemStruct, "struct item");
/// AST for a struct body.
#[derive(Debug, Clone, PartialEq, Eq, ToTokens, OptionSpanned)]
#[non_exhaustive]
pub enum ItemStructBody {
/// An empty struct declaration.
UnitBody,
/// A tuple struct body.
TupleBody(ast::Parenthesized<Field, T![,]>),
/// A regular struct body.
StructBody(ast::Braced<Field, T![,]>),
}
impl ItemStructBody {
/// If the body needs to be terminated with a semicolon.
fn needs_semi_colon(&self) -> bool {
matches!(self, Self::UnitBody | Self::TupleBody(..))
}
/// Iterate over the fields of the body.
pub(crate) fn fields(&self) -> impl Iterator<Item = &'_ (Field, Option<T![,]>)> {
match self {
ItemStructBody::UnitBody => IntoIterator::into_iter(&[]),
ItemStructBody::TupleBody(body) => body.iter(),
ItemStructBody::StructBody(body) => body.iter(),
}
}
}
/// Parse implementation for a struct body.
///
/// # Examples
///
/// ```
/// use rune::{ast, testing};
///
/// testing::roundtrip::<ast::ItemStructBody>("");
///
/// testing::roundtrip::<ast::ItemStructBody>("{ a, b, c }");
/// testing::roundtrip::<ast::ItemStructBody>("{ #[x] a, #[y] b, #[z] #[w] #[f32] c }");
/// testing::roundtrip::<ast::ItemStructBody>("{ a, #[attribute] b, c }");
///
/// testing::roundtrip::<ast::ItemStructBody>("( a, b, c )");
/// testing::roundtrip::<ast::ItemStructBody>("( #[x] a, b, c )");
/// testing::roundtrip::<ast::ItemStructBody>("( #[x] pub a, b, c )");
/// testing::roundtrip::<ast::ItemStructBody>("( a, b, c )");
/// testing::roundtrip::<ast::ItemStructBody>("()");
/// ```
impl Parse for ItemStructBody {
fn parse(p: &mut Parser<'_>) -> Result<Self, ParseError> {
Ok(match p.nth(0)? {
K!['('] => Self::TupleBody(p.parse()?),
K!['{'] => Self::StructBody(p.parse()?),
_ => Self::UnitBody,
})
}
}
/// A field as part of a struct or a tuple body.
///
/// # Examples
///
/// ```
/// use rune::{ast, testing};
///
/// testing::roundtrip::<ast::Field>("a");
/// testing::roundtrip::<ast::Field>("#[x] a");
/// ```
#[derive(Debug, Clone, PartialEq, Eq, ToTokens, Parse, Spanned)]
#[non_exhaustive]
pub struct Field {
/// Attributes associated with field.
#[rune(iter)]
pub attributes: Vec<ast::Attribute>,
/// The visibility of the field
#[rune(optional)]
pub visibility: ast::Visibility,
/// Name of the field.
pub name: ast::Ident,
}
