use crate::nodes::{Block, Identifier, Token};
#[derive(Clone, Debug, PartialEq, Eq)]
pub struct FunctionExpressionTokens {
pub function: Token,
pub opening_parenthese: Token,
pub closing_parenthese: Token,
pub end: Token,
pub parameter_commas: Vec<Token>,
pub variable_arguments: Option<Token>,
}
impl FunctionExpressionTokens {
pub fn clear_comments(&mut self) {
self.function.clear_comments();
self.opening_parenthese.clear_comments();
self.closing_parenthese.clear_comments();
self.end.clear_comments();
self.parameter_commas
.iter_mut()
.for_each(Token::clear_comments);
if let Some(token) = &mut self.variable_arguments {
token.clear_comments();
}
}
pub fn clear_whitespaces(&mut self) {
self.function.clear_whitespaces();
self.opening_parenthese.clear_whitespaces();
self.closing_parenthese.clear_whitespaces();
self.end.clear_whitespaces();
self.parameter_commas
.iter_mut()
.for_each(Token::clear_whitespaces);
if let Some(token) = &mut self.variable_arguments {
token.clear_whitespaces();
}
}
pub(crate) fn replace_referenced_tokens(&mut self, code: &str) {
self.function.replace_referenced_tokens(code);
self.opening_parenthese.replace_referenced_tokens(code);
self.closing_parenthese.replace_referenced_tokens(code);
self.end.replace_referenced_tokens(code);
self.parameter_commas
.iter_mut()
.for_each(|token| token.replace_referenced_tokens(code));
if let Some(token) = &mut self.variable_arguments {
token.replace_referenced_tokens(code);
}
}
pub(crate) fn shift_token_line(&mut self, amount: usize) {
self.function.shift_token_line(amount);
self.opening_parenthese.shift_token_line(amount);
self.closing_parenthese.shift_token_line(amount);
self.end.shift_token_line(amount);
self.parameter_commas
.iter_mut()
.for_each(|token| token.shift_token_line(amount));
if let Some(token) = &mut self.variable_arguments {
token.shift_token_line(amount);
}
}
}
#[derive(Clone, Debug, Default, PartialEq, Eq)]
pub struct FunctionExpression {
block: Block,
parameters: Vec<Identifier>,
is_variadic: bool,
tokens: Option<Box<FunctionExpressionTokens>>,
}
impl FunctionExpression {
pub fn new(block: Block, parameters: Vec<Identifier>, is_variadic: bool) -> Self {
Self {
block,
parameters,
is_variadic,
tokens: None,
}
}
pub fn from_block<B: Into<Block>>(block: B) -> Self {
Self {
block: block.into(),
parameters: Vec::new(),
is_variadic: false,
tokens: None,
}
}
pub fn with_parameter<P: Into<Identifier>>(mut self, parameter: P) -> Self {
self.parameters.push(parameter.into());
self
}
pub fn variadic(mut self) -> Self {
self.is_variadic = true;
self
}
pub fn set_variadic(&mut self, is_variadic: bool) {
self.is_variadic = is_variadic;
}
pub fn with_tokens(mut self, tokens: FunctionExpressionTokens) -> Self {
self.tokens = Some(tokens.into());
self
}
#[inline]
pub fn set_tokens(&mut self, tokens: FunctionExpressionTokens) {
self.tokens = Some(tokens.into());
}
#[inline]
pub fn get_tokens(&self) -> Option<&FunctionExpressionTokens> {
self.tokens.as_ref().map(|tokens| tokens.as_ref())
}
#[inline]
pub fn get_block(&self) -> &Block {
&self.block
}
#[inline]
pub fn get_parameters(&self) -> &Vec<Identifier> {
&self.parameters
}
#[inline]
pub fn iter_parameters(&self) -> impl Iterator<Item = &Identifier> {
self.parameters.iter()
}
#[inline]
pub fn is_variadic(&self) -> bool {
self.is_variadic
}
#[inline]
pub fn mutate_block(&mut self) -> &mut Block {
&mut self.block
}
#[inline]
pub fn mutate_parameters(&mut self) -> &mut Vec<Identifier> {
&mut self.parameters
}
#[inline]
pub fn parameters_count(&self) -> usize {
self.parameters.len()
}
#[inline]
pub fn has_parameters(&self) -> bool {
!self.parameters.is_empty()
}
pub fn clear_comments(&mut self) {
self.parameters
.iter_mut()
.for_each(Identifier::clear_comments);
if let Some(tokens) = &mut self.tokens {
tokens.clear_comments();
}
}
pub fn clear_whitespaces(&mut self) {
self.parameters
.iter_mut()
.for_each(Identifier::clear_whitespaces);
if let Some(tokens) = &mut self.tokens {
tokens.clear_whitespaces();
}
}
pub(crate) fn replace_referenced_tokens(&mut self, code: &str) {
for parameter in self.parameters.iter_mut() {
parameter.replace_referenced_tokens(code);
}
if let Some(tokens) = &mut self.tokens {
tokens.replace_referenced_tokens(code);
}
}
pub(crate) fn shift_token_line(&mut self, amount: usize) {
if let Some(tokens) = &mut self.tokens {
tokens.shift_token_line(amount);
}
}
}