use crate::nodes::{Expression, Token};
#[derive(Clone, Debug, PartialEq, Eq)]
pub struct ReturnTokens {
pub r#return: Token,
pub commas: Vec<Token>,
}
impl ReturnTokens {
super::impl_token_fns!(
target = [r#return]
iter = [commas]
);
}
#[derive(Clone, Debug, Default, PartialEq, Eq)]
pub struct ReturnStatement {
expressions: Vec<Expression>,
tokens: Option<ReturnTokens>,
}
impl ReturnStatement {
pub fn new(expressions: Vec<Expression>) -> Self {
Self {
expressions,
tokens: None,
}
}
pub fn one<E: Into<Expression>>(expression: E) -> Self {
Self {
expressions: vec![expression.into()],
tokens: None,
}
}
pub fn with_expression<E: Into<Expression>>(mut self, expression: E) -> Self {
self.expressions.push(expression.into());
self
}
#[inline]
pub fn iter_expressions(&self) -> impl Iterator<Item = &Expression> {
self.expressions.iter()
}
#[inline]
pub fn into_iter_expressions(self) -> impl Iterator<Item = Expression> {
self.expressions.into_iter()
}
#[inline]
pub fn iter_mut_expressions(&mut self) -> impl Iterator<Item = &mut Expression> {
self.expressions.iter_mut()
}
#[inline]
pub fn is_empty(&self) -> bool {
self.expressions.is_empty()
}
#[inline]
pub fn len(&self) -> usize {
self.expressions.len()
}
pub fn with_tokens(mut self, tokens: ReturnTokens) -> Self {
self.tokens = Some(tokens);
self
}
#[inline]
pub fn set_tokens(&mut self, tokens: ReturnTokens) {
self.tokens = Some(tokens);
}
#[inline]
pub fn get_tokens(&self) -> Option<&ReturnTokens> {
self.tokens.as_ref()
}
#[inline]
pub fn mutate_tokens(&mut self) -> Option<&mut ReturnTokens> {
self.tokens.as_mut()
}
pub fn mutate_first_token(&mut self) -> &mut Token {
self.set_default_tokens();
&mut self.tokens.as_mut().unwrap().r#return
}
pub fn mutate_last_token(&mut self) -> &mut Token {
if self.is_empty() {
self.set_default_tokens();
&mut self.tokens.as_mut().unwrap().r#return
} else {
self.expressions.last_mut().unwrap().mutate_last_token()
}
}
fn set_default_tokens(&mut self) {
if self.tokens.is_none() {
self.tokens = Some(ReturnTokens {
r#return: Token::from_content("return"),
commas: Vec::new(),
});
}
}
super::impl_token_fns!(iter = [tokens]);
}
#[derive(Clone, Debug, PartialEq, Eq)]
pub enum LastStatement {
Break(Option<Token>),
Continue(Option<Token>),
Return(ReturnStatement),
}
impl LastStatement {
#[inline]
pub fn new_break() -> Self {
Self::Break(None)
}
#[inline]
pub fn new_continue() -> Self {
Self::Continue(None)
}
pub fn mutate_first_token(&mut self) -> &mut Token {
match self {
LastStatement::Break(token_opt) => {
if token_opt.is_none() {
*token_opt = Some(Token::from_content("break"));
}
token_opt.as_mut().unwrap()
}
LastStatement::Continue(token_opt) => {
if token_opt.is_none() {
*token_opt = Some(Token::from_content("continue"));
}
token_opt.as_mut().unwrap()
}
LastStatement::Return(return_stmt) => return_stmt.mutate_first_token(),
}
}
pub fn mutate_last_token(&mut self) -> &mut Token {
match self {
LastStatement::Break(token_opt) => {
if token_opt.is_none() {
*token_opt = Some(Token::from_content("break"));
}
token_opt.as_mut().unwrap()
}
LastStatement::Continue(token_opt) => {
if token_opt.is_none() {
*token_opt = Some(Token::from_content("continue"));
}
token_opt.as_mut().unwrap()
}
LastStatement::Return(return_stmt) => return_stmt.mutate_last_token(),
}
}
}
impl From<ReturnStatement> for LastStatement {
fn from(statement: ReturnStatement) -> Self {
Self::Return(statement)
}
}
#[cfg(test)]
mod test {
use super::*;
#[test]
fn default_return_statement_is_empty() {
assert!(ReturnStatement::default().is_empty())
}
}