use crate::{
comments::{rewrite_with_comments, write_comments},
constants::NEW_LINE,
formatter::*,
utils::{
map::byte_span::{ByteSpan, LeafSpans},
CurlyBrace,
},
};
use std::fmt::Write;
use sway_ast::{
keywords::{AddToken, ColonToken, Keyword, Token, TraitToken},
ItemTrait, ItemTraitItem, PubToken, Traits,
};
use sway_types::{ast::Delimiter, Spanned};
#[cfg(test)]
mod tests;
impl Format for ItemTrait {
fn format(
&self,
formatted_code: &mut FormattedCode,
formatter: &mut Formatter,
) -> Result<(), FormatterError> {
let start_len = formatted_code.len();
if self.visibility.is_some() {
write!(formatted_code, "{} ", PubToken::AS_STR)?;
}
write!(
formatted_code,
"{} {}",
TraitToken::AS_STR,
self.name.as_str(),
)?;
if let Some(generics) = &self.generics {
generics.format(formatted_code, formatter)?;
}
if let Some((_colon_token, traits)) = &self.super_traits {
write!(formatted_code, "{} ", ColonToken::AS_STR)?;
traits.format(formatted_code, formatter)?;
}
if let Some(where_clause) = &self.where_clause_opt {
writeln!(formatted_code)?;
where_clause.format(formatted_code, formatter)?;
} else {
write!(formatted_code, " ")?;
}
Self::open_curly_brace(formatted_code, formatter)?;
let trait_items = self.trait_items.get();
if trait_items.is_empty() {
write_comments(formatted_code, self.trait_items.span().into(), formatter)?;
} else {
for item in trait_items.iter() {
item.format(formatted_code, formatter)?;
write!(formatted_code, "{NEW_LINE}")?;
}
}
Self::close_curly_brace(formatted_code, formatter)?;
if let Some(trait_defs) = &self.trait_defs_opt {
write!(formatted_code, " ")?;
Self::open_curly_brace(formatted_code, formatter)?;
for trait_items in trait_defs.get().iter() {
trait_items.format(formatted_code, formatter)?;
write!(formatted_code, "{NEW_LINE}")?;
}
if trait_defs.get().is_empty() {
write!(formatted_code, "{NEW_LINE}")?;
}
Self::close_curly_brace(formatted_code, formatter)?;
};
rewrite_with_comments::<ItemTrait>(
formatter,
self.span(),
self.leaf_spans(),
formatted_code,
start_len,
)?;
Ok(())
}
}
impl Format for ItemTraitItem {
fn format(
&self,
formatted_code: &mut FormattedCode,
formatter: &mut Formatter,
) -> Result<(), FormatterError> {
match self {
ItemTraitItem::Fn(fn_decl, _) => {
fn_decl.format(formatted_code, formatter)?;
write!(formatted_code, ";")?;
}
ItemTraitItem::Const(const_decl, _) => {
const_decl.format(formatted_code, formatter)?;
}
ItemTraitItem::Type(type_decl, _) => {
type_decl.format(formatted_code, formatter)?;
}
ItemTraitItem::Error(_, _) => {
return Err(FormatterError::SyntaxError);
}
}
Ok(())
}
}
impl CurlyBrace for ItemTrait {
fn open_curly_brace(
line: &mut FormattedCode,
formatter: &mut Formatter,
) -> Result<(), FormatterError> {
formatter.indent();
let open_brace = Delimiter::Brace.as_open_char();
writeln!(line, "{open_brace}")?;
Ok(())
}
fn close_curly_brace(
line: &mut FormattedCode,
formatter: &mut Formatter,
) -> Result<(), FormatterError> {
formatter.unindent();
write!(line, "{}", Delimiter::Brace.as_close_char())?;
Ok(())
}
}
impl Format for Traits {
fn format(
&self,
formatted_code: &mut FormattedCode,
formatter: &mut Formatter,
) -> Result<(), FormatterError> {
self.prefix.format(formatted_code, formatter)?;
for (_add_token, path_type) in self.suffixes.iter() {
write!(formatted_code, " {} ", AddToken::AS_STR)?;
path_type.format(formatted_code, formatter)?;
}
Ok(())
}
}
impl LeafSpans for ItemTrait {
fn leaf_spans(&self) -> Vec<ByteSpan> {
let mut collected_spans = Vec::new();
if let Some(visibility) = &self.visibility {
collected_spans.push(ByteSpan::from(visibility.span()));
}
collected_spans.push(ByteSpan::from(self.trait_token.span()));
collected_spans.push(ByteSpan::from(self.name.span()));
if let Some(super_traits) = &self.super_traits {
collected_spans.append(&mut super_traits.leaf_spans());
}
collected_spans.append(&mut self.trait_items.leaf_spans());
if let Some(trait_defs) = &self.trait_defs_opt {
collected_spans.append(&mut trait_defs.leaf_spans());
}
collected_spans
}
}
impl LeafSpans for ItemTraitItem {
fn leaf_spans(&self) -> Vec<ByteSpan> {
let mut collected_spans = Vec::new();
match &self {
ItemTraitItem::Fn(fn_sig, semicolon) => {
collected_spans.append(&mut fn_sig.leaf_spans());
collected_spans.extend(semicolon.as_ref().into_iter().flat_map(|x| x.leaf_spans()));
}
ItemTraitItem::Const(const_decl, semicolon) => {
collected_spans.append(&mut const_decl.leaf_spans());
collected_spans.extend(semicolon.as_ref().into_iter().flat_map(|x| x.leaf_spans()));
}
ItemTraitItem::Error(spans, _) => {
collected_spans.extend(spans.iter().cloned().map(Into::into));
}
ItemTraitItem::Type(type_decl, _) => {
collected_spans.append(&mut type_decl.leaf_spans())
}
};
collected_spans
}
}
impl LeafSpans for Traits {
fn leaf_spans(&self) -> Vec<ByteSpan> {
let mut collected_spans = self.prefix.leaf_spans();
collected_spans.append(&mut self.suffixes.leaf_spans());
collected_spans
}
}