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
use crate::ast::{Argument, FromTokens, IsMatch, ParseError, Tokens};
use crate::lexical_token::PunctuatorType;
use crate::{HasSpan, Span};
use bluejay_core::AsIter;

#[derive(Debug)]
pub struct Arguments<'a, const CONST: bool> {
    arguments: Vec<Argument<'a, CONST>>,
    span: Span,
}

pub type VariableArguments<'a> = Arguments<'a, false>;

impl<'a, const CONST: bool> FromTokens<'a> for Arguments<'a, CONST> {
    fn from_tokens(tokens: &mut impl Tokens<'a>) -> Result<Self, ParseError> {
        let open_span = tokens.expect_punctuator(PunctuatorType::OpenRoundBracket)?;
        let mut arguments: Vec<Argument<CONST>> = Vec::new();
        let close_span = loop {
            arguments.push(Argument::from_tokens(tokens)?);
            if let Some(close_span) = tokens.next_if_punctuator(PunctuatorType::CloseRoundBracket) {
                break close_span;
            }
        };
        let span = open_span.merge(&close_span);
        Ok(Self { arguments, span })
    }
}

impl<'a, const CONST: bool> IsMatch<'a> for Arguments<'a, CONST> {
    fn is_match(tokens: &mut impl Tokens<'a>) -> bool {
        tokens.peek_punctuator_matches(0, PunctuatorType::OpenRoundBracket)
    }
}

impl<'a, const CONST: bool> HasSpan for Arguments<'a, CONST> {
    fn span(&self) -> &Span {
        &self.span
    }
}

impl<'a, const CONST: bool> bluejay_core::Arguments<CONST> for Arguments<'a, CONST> {
    type Argument = Argument<'a, CONST>;
}

impl<'a, const CONST: bool> AsIter for Arguments<'a, CONST> {
    type Item = Argument<'a, CONST>;
    type Iterator<'b> = std::slice::Iter<'b, Self::Item> where 'a: 'b;

    fn iter(&self) -> Self::Iterator<'_> {
        self.arguments.iter()
    }
}