ezno-parser 0.0.1

Parser and AST definitions for Ezno
Documentation
use std::borrow::Cow;

use iterator_endiate::EndiateIteratorExt;
use source_map::Span;
use tokenizer_lib::Token;
use visitable_derive::Visitable;

use crate::{
	errors::parse_lexing_error, ASTNode, Expression, ParseSettings, Statement, TSXKeyword, TSXToken,
};

#[derive(Debug, PartialEq, Eq, Clone, Visitable)]
#[cfg_attr(feature = "self-rust-tokenize", derive(self_rust_tokenize::SelfRustTokenize))]
pub struct SwitchStatement {
	pub case: Expression,
	pub branches: Vec<SwitchBranch>,
	pub position: Span,
}

#[derive(Debug, PartialEq, Eq, Clone, Visitable)]
#[cfg_attr(feature = "self-rust-tokenize", derive(self_rust_tokenize::SelfRustTokenize))]
pub enum SwitchBranch {
	Default(Vec<Statement>),
	Case(Expression, Vec<Statement>),
}

impl ASTNode for SwitchStatement {
	fn get_position(&self) -> Cow<Span> {
		Cow::Borrowed(&self.position)
	}

	fn from_reader(
		reader: &mut impl tokenizer_lib::TokenReader<crate::TSXToken, Span>,
		state: &mut crate::ParsingState,
		settings: &ParseSettings,
	) -> Result<Self, crate::ParseError> {
		let start_span = reader.expect_next(TSXToken::Keyword(TSXKeyword::Switch))?;
		reader.expect_next(crate::TSXToken::OpenParentheses)?;
		let case = Expression::from_reader(reader, state, settings)?;
		reader.expect_next(crate::TSXToken::CloseParentheses)?;
		reader.expect_next(crate::TSXToken::OpenBrace)?;
		let mut branches = Vec::new();
		let close_brace_pos: Span;
		loop {
			let Token(token_type, pos) = reader.next().ok_or_else(parse_lexing_error)?;
			let case: Option<Expression> = match token_type {
				TSXToken::Keyword(TSXKeyword::Default) => {
					reader.expect_next(TSXToken::Colon)?;
					None
				}
				TSXToken::Keyword(TSXKeyword::Case) => {
					let case = Expression::from_reader(reader, state, settings)?;
					reader.expect_next(TSXToken::Colon)?;
					Some(case)
				}
				TSXToken::CloseBrace => {
					close_brace_pos = pos;
					break;
				}
				_ => todo!(),
			};
			let mut statements = Vec::new();
			while !matches!(
				&reader.peek().unwrap().0,
				TSXToken::Keyword(TSXKeyword::Case)
					| TSXToken::Keyword(TSXKeyword::Default)
					| TSXToken::CloseBrace
			) {
				statements.push(Statement::from_reader(reader, state, settings)?);
				if let Some(Token(TSXToken::SemiColon, _)) = reader.peek() {
					reader.next();
				}
			}
			if let Some(case) = case {
				branches.push(SwitchBranch::Case(case, statements))
			} else {
				branches.push(SwitchBranch::Default(statements))
			}
		}
		Ok(Self { case, branches, position: start_span.union(&close_brace_pos) })
	}

	fn to_string_from_buffer<T: source_map::ToString>(
		&self,
		buf: &mut T,
		settings: &crate::ToStringSettingsAndData,
		depth: u8,
	) {
		buf.push_str("switch");
		settings.0.add_gap(buf);
		buf.push('(');
		self.case.to_string_from_buffer(buf, settings, depth);
		buf.push(')');
		settings.0.add_gap(buf);
		buf.push('{');
		for branch in self.branches.iter() {
			if settings.0.pretty {
				buf.push_new_line();
				settings.0.add_indent(depth + 1, buf);
			}
			match branch {
				SwitchBranch::Default(statements) => {
					buf.push_str("default:");
					for (at_end, statement) in statements.iter().endiate() {
						if settings.0.pretty {
							buf.push_new_line();
							settings.0.add_indent(depth + 2, buf);
						}
						statement.to_string_from_buffer(buf, settings, depth + 2);
						if settings.0.pretty {
							if !at_end {
								buf.push(';');
							}
						} else {
							buf.push_new_line();
						}
					}
				}
				SwitchBranch::Case(case, statements) => {
					buf.push_str("case ");
					case.to_string_from_buffer(buf, settings, depth);
					buf.push(':');
					for (at_end, statement) in statements.iter().endiate() {
						if settings.0.pretty {
							buf.push_new_line();
							settings.0.add_indent(depth + 2, buf);
						}
						statement.to_string_from_buffer(buf, settings, depth + 2);
						if settings.0.pretty {
							if !at_end {
								buf.push(';');
							}
						} else {
							buf.push_new_line();
						}
					}
				}
			}
		}
		if settings.0.pretty {
			buf.push_new_line();
			settings.0.add_indent(depth, buf);
		}
		buf.push('}');
	}
}