mirror of
https://github.com/roc-lang/roc.git
synced 2025-09-26 21:39:07 +00:00
147 lines
4 KiB
Rust
147 lines
4 KiB
Rust
use bumpalo::{collections::Vec, Bump};
|
|
use roc_parse::ast::{CommentOrNewline, Spaces, TypeAnnotation};
|
|
|
|
use crate::{
|
|
annotation::{Formattable, Newlines, Parens},
|
|
collection::Braces,
|
|
spaces::fmt_spaces,
|
|
Buf,
|
|
};
|
|
|
|
#[derive(Copy, Clone, Debug)]
|
|
pub struct Sp<'a> {
|
|
default_space: bool, // if true and comments is empty, use a space (' ')
|
|
comments: &'a [CommentOrNewline<'a>],
|
|
}
|
|
|
|
impl<'a> Sp<'a> {
|
|
pub fn empty() -> Sp<'a> {
|
|
Sp {
|
|
default_space: false,
|
|
comments: &[],
|
|
}
|
|
}
|
|
pub fn space() -> Sp<'a> {
|
|
Sp {
|
|
default_space: true,
|
|
comments: &[],
|
|
}
|
|
}
|
|
|
|
pub fn with_space(sp: &'a [CommentOrNewline<'a>]) -> Self {
|
|
Sp {
|
|
default_space: true,
|
|
comments: sp,
|
|
}
|
|
}
|
|
}
|
|
|
|
impl<'a> From<&'a [CommentOrNewline<'a>]> for Sp<'a> {
|
|
fn from(comments: &'a [CommentOrNewline<'a>]) -> Self {
|
|
Sp {
|
|
default_space: false,
|
|
comments,
|
|
}
|
|
}
|
|
}
|
|
|
|
#[derive(Copy, Clone, Debug)]
|
|
pub enum Node<'a> {
|
|
Literal(&'a str),
|
|
Sequence(&'a Node<'a>, &'a [(Sp<'a>, Node<'a>)]),
|
|
DelimitedSequence(Braces, &'a [(Sp<'a>, Node<'a>)], Sp<'a>),
|
|
TypeAnnotation(TypeAnnotation<'a>),
|
|
}
|
|
|
|
pub trait Nodify<'a> {
|
|
fn to_node<'b>(&'a self, arena: &'b Bump, parens: Parens) -> Spaces<'b, Node<'b>>
|
|
where
|
|
'a: 'b;
|
|
}
|
|
|
|
fn fmt_sp(buf: &mut Buf, sp: Sp<'_>, indent: u16) {
|
|
if !sp.comments.is_empty() {
|
|
fmt_spaces(buf, sp.comments.iter(), indent);
|
|
} else if sp.default_space {
|
|
buf.spaces(1);
|
|
}
|
|
}
|
|
|
|
impl<'a> Formattable for Node<'a> {
|
|
fn is_multiline(&self) -> bool {
|
|
match self {
|
|
Node::DelimitedSequence(_braces, lefts, right) => {
|
|
right.comments.is_empty()
|
|
&& lefts
|
|
.iter()
|
|
.any(|(sp, l)| l.is_multiline() || !sp.comments.is_empty())
|
|
}
|
|
Node::Sequence(first, rest) => {
|
|
first.is_multiline()
|
|
|| rest
|
|
.iter()
|
|
.any(|(sp, l)| l.is_multiline() || !sp.comments.is_empty())
|
|
}
|
|
Node::TypeAnnotation(type_annotation) => type_annotation.is_multiline(),
|
|
Node::Literal(_) => false,
|
|
}
|
|
}
|
|
|
|
fn format_with_options(&self, buf: &mut Buf, parens: Parens, newlines: Newlines, indent: u16) {
|
|
match self {
|
|
Node::DelimitedSequence(braces, lefts, right) => {
|
|
buf.indent(indent);
|
|
buf.push(braces.start());
|
|
|
|
for (sp, l) in *lefts {
|
|
fmt_sp(buf, *sp, indent);
|
|
l.format_with_options(buf, parens, newlines, indent);
|
|
}
|
|
fmt_sp(buf, *right, indent);
|
|
|
|
buf.indent(indent);
|
|
buf.push(braces.end());
|
|
}
|
|
Node::Sequence(first, rest) => {
|
|
first.format_with_options(buf, parens, newlines, indent);
|
|
|
|
for (sp, l) in *rest {
|
|
fmt_sp(buf, *sp, indent);
|
|
l.format_with_options(buf, parens, newlines, indent);
|
|
}
|
|
}
|
|
Node::TypeAnnotation(type_annotation) => {
|
|
type_annotation.format_with_options(buf, parens, newlines, indent);
|
|
}
|
|
Node::Literal(text) => {
|
|
buf.indent(indent);
|
|
buf.push_str(text);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
pub struct NodeSequenceBuilder<'a> {
|
|
first: Node<'a>,
|
|
rest: Vec<'a, (Sp<'a>, Node<'a>)>,
|
|
}
|
|
|
|
impl<'a> NodeSequenceBuilder<'a> {
|
|
pub fn new(arena: &'a Bump, first: Node<'a>, capacity: usize) -> Self {
|
|
Self {
|
|
first,
|
|
rest: Vec::with_capacity_in(capacity, arena),
|
|
}
|
|
}
|
|
|
|
pub fn push(&mut self, sp: Sp<'a>, literal: Node<'a>) {
|
|
self.rest.push((sp, literal));
|
|
}
|
|
|
|
pub fn build(self) -> Node<'a> {
|
|
Node::Sequence(
|
|
self.rest.bump().alloc(self.first),
|
|
self.rest.into_bump_slice(),
|
|
)
|
|
}
|
|
}
|