use std::mem;
use crate::nodes::{Block, Expression, Token};
#[derive(Clone, Debug, PartialEq, Eq)]
pub struct IfBranchTokens {
pub elseif: Token,
pub then: Token,
}
impl IfBranchTokens {
pub fn clear_comments(&mut self) {
self.elseif.clear_comments();
self.then.clear_comments();
}
pub fn clear_whitespaces(&mut self) {
self.elseif.clear_whitespaces();
self.then.clear_whitespaces();
}
pub(crate) fn replace_referenced_tokens(&mut self, code: &str) {
self.elseif.replace_referenced_tokens(code);
self.then.replace_referenced_tokens(code);
}
pub(crate) fn shift_token_line(&mut self, amount: usize) {
self.elseif.shift_token_line(amount);
self.then.shift_token_line(amount);
}
}
#[derive(Clone, Debug, PartialEq, Eq)]
pub struct IfBranch {
condition: Expression,
block: Block,
tokens: Option<IfBranchTokens>,
}
impl IfBranch {
pub fn new<E: Into<Expression>, B: Into<Block>>(condition: E, block: B) -> Self {
Self {
condition: condition.into(),
block: block.into(),
tokens: None,
}
}
pub fn empty<E: Into<Expression>>(condition: E) -> Self {
Self {
condition: condition.into(),
block: Block::default(),
tokens: None,
}
}
pub fn with_tokens(mut self, tokens: IfBranchTokens) -> Self {
self.tokens = Some(tokens);
self
}
#[inline]
pub fn set_tokens(&mut self, tokens: IfBranchTokens) {
self.tokens = Some(tokens);
}
#[inline]
pub fn get_tokens(&self) -> Option<&IfBranchTokens> {
self.tokens.as_ref()
}
#[inline]
pub fn get_block(&self) -> &Block {
&self.block
}
#[inline]
pub fn get_condition(&self) -> &Expression {
&self.condition
}
#[inline]
pub fn mutate_block(&mut self) -> &mut Block {
&mut self.block
}
#[inline]
pub fn take_block(&mut self) -> Block {
mem::take(&mut self.block)
}
#[inline]
pub fn mutate_condition(&mut self) -> &mut Expression {
&mut self.condition
}
pub fn clear_comments(&mut self) {
if let Some(tokens) = &mut self.tokens {
tokens.clear_comments();
}
}
pub fn clear_whitespaces(&mut self) {
if let Some(tokens) = &mut self.tokens {
tokens.clear_whitespaces();
}
}
pub(crate) fn replace_referenced_tokens(&mut self, code: &str) {
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);
}
}
}
#[derive(Clone, Debug, PartialEq, Eq)]
pub struct IfStatementTokens {
pub r#if: Token,
pub then: Token,
pub end: Token,
pub r#else: Option<Token>,
}
impl IfStatementTokens {
pub fn clear_comments(&mut self) {
self.r#if.clear_comments();
self.then.clear_comments();
self.end.clear_comments();
if let Some(token) = &mut self.r#else {
token.clear_comments();
}
}
pub fn clear_whitespaces(&mut self) {
self.r#if.clear_whitespaces();
self.then.clear_whitespaces();
self.end.clear_whitespaces();
if let Some(token) = &mut self.r#else {
token.clear_whitespaces();
}
}
pub(crate) fn replace_referenced_tokens(&mut self, code: &str) {
self.r#if.replace_referenced_tokens(code);
self.then.replace_referenced_tokens(code);
self.end.replace_referenced_tokens(code);
if let Some(token) = &mut self.r#else {
token.replace_referenced_tokens(code);
}
}
pub(crate) fn shift_token_line(&mut self, amount: usize) {
self.r#if.shift_token_line(amount);
self.then.shift_token_line(amount);
self.end.shift_token_line(amount);
if let Some(token) = &mut self.r#else {
token.shift_token_line(amount);
}
}
}
#[derive(Clone, Debug, PartialEq, Eq)]
pub struct IfStatement {
branches: Vec<IfBranch>,
else_block: Option<Block>,
tokens: Option<IfStatementTokens>,
}
impl IfStatement {
pub fn new(branches: Vec<IfBranch>, else_block: Option<Block>) -> Self {
Self {
branches,
else_block,
tokens: None,
}
}
pub fn create(condition: impl Into<Expression>, block: impl Into<Block>) -> Self {
Self {
branches: vec![IfBranch::new(condition, block)],
else_block: None,
tokens: None,
}
}
pub fn with_tokens(mut self, tokens: IfStatementTokens) -> Self {
self.tokens = Some(tokens);
self
}
#[inline]
pub fn set_tokens(&mut self, tokens: IfStatementTokens) {
self.tokens = Some(tokens);
}
#[inline]
pub fn get_tokens(&self) -> Option<&IfStatementTokens> {
self.tokens.as_ref()
}
pub fn with_branch(mut self, branch: IfBranch) -> Self {
self.branches.push(branch);
self
}
pub fn with_new_branch(
mut self,
condition: impl Into<Expression>,
block: impl Into<Block>,
) -> Self {
self.branches.push(IfBranch::new(condition, block));
self
}
pub fn with_else_block<B: Into<Block>>(mut self, block: B) -> Self {
self.else_block.replace(block.into());
self
}
pub fn mutate_all_blocks(&mut self) -> Vec<&mut Block> {
let mut blocks: Vec<&mut Block> = self
.branches
.iter_mut()
.map(|branch| branch.mutate_block())
.collect();
if let Some(else_block) = &mut self.else_block {
blocks.push(else_block);
};
blocks
}
#[inline]
pub fn get_branches(&self) -> &Vec<IfBranch> {
&self.branches
}
#[inline]
pub fn iter_branches(&self) -> impl Iterator<Item = &IfBranch> {
self.branches.iter()
}
#[inline]
pub fn branch_count(&self) -> usize {
self.branches.len()
}
#[inline]
pub fn mutate_branches(&mut self) -> &mut Vec<IfBranch> {
&mut self.branches
}
#[inline]
pub fn push_new_branch(&mut self, condition: impl Into<Expression>, block: impl Into<Block>) {
self.branches
.push(IfBranch::new(condition.into(), block.into()));
}
#[inline]
pub fn push_branch(&mut self, branch: IfBranch) {
self.branches.push(branch);
}
#[inline]
pub fn get_else_block(&self) -> Option<&Block> {
self.else_block.as_ref()
}
#[inline]
pub fn mutate_else_block(&mut self) -> &mut Option<Block> {
&mut self.else_block
}
#[inline]
pub fn set_else_block(&mut self, block: impl Into<Block>) {
self.else_block = Some(block.into());
}
#[inline]
pub fn take_else_block(&mut self) -> Option<Block> {
self.else_block.take()
}
pub fn clear_comments(&mut self) {
if let Some(tokens) = &mut self.tokens {
tokens.clear_comments();
}
self.branches.iter_mut().for_each(IfBranch::clear_comments);
}
pub fn clear_whitespaces(&mut self) {
if let Some(tokens) = &mut self.tokens {
tokens.clear_whitespaces();
}
self.branches
.iter_mut()
.for_each(IfBranch::clear_whitespaces);
}
pub(crate) fn replace_referenced_tokens(&mut self, code: &str) {
if let Some(tokens) = &mut self.tokens {
tokens.replace_referenced_tokens(code);
}
for branch in self.branches.iter_mut() {
branch.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);
}
for branch in self.branches.iter_mut() {
branch.shift_token_line(amount);
}
}
pub fn retain_branches_mut(&mut self, filter: impl FnMut(&mut IfBranch) -> bool) -> bool {
self.branches.retain_mut(filter);
if self.branches.is_empty() {
self.branches.push(IfBranch::new(false, Block::default()));
true
} else {
false
}
}
}