use crate::{
config::{items::ItemBraceStyle, user_def::FieldAlignment},
formatter::{shape::LineStyle, *},
utils::{
map::byte_span::{ByteSpan, LeafSpans},
CurlyBrace,
},
};
use std::fmt::Write;
use sway_ast::{keywords::Token, token::Delimiter, ItemStorage, StorageField};
use sway_types::Spanned;
#[cfg(test)]
mod tests;
impl Format for ItemStorage {
fn format(
&self,
formatted_code: &mut FormattedCode,
formatter: &mut Formatter,
) -> Result<(), FormatterError> {
formatter
.shape
.code_line
.update_line_style(LineStyle::Multiline);
write!(formatted_code, "{}", self.storage_token.span().as_str())?;
let fields = self.fields.get();
Self::open_curly_brace(formatted_code, formatter)?;
match formatter.config.structures.field_alignment {
FieldAlignment::AlignFields(storage_field_align_threshold) => {
writeln!(formatted_code)?;
let value_pairs = &fields
.value_separator_pairs
.iter()
.map(|(storage_field, comma_token)| (&storage_field.value, comma_token))
.collect::<Vec<_>>();
let field_length: Vec<usize> = value_pairs
.iter()
.map(|(storage_field, _)| storage_field.name.as_str().len())
.collect();
let mut max_valid_field_length = 0;
field_length.iter().for_each(|length| {
if *length > max_valid_field_length && *length < storage_field_align_threshold {
max_valid_field_length = *length;
}
});
let mut value_pairs_iter = value_pairs.iter().enumerate().peekable();
for (field_index, (storage_field, comma_token)) in value_pairs_iter.clone() {
write!(
formatted_code,
"{}",
&formatter.shape.indent.to_string(&formatter.config)?
)?;
storage_field.name.format(formatted_code, formatter)?;
let current_field_length = field_length[field_index];
if current_field_length < max_valid_field_length {
let mut required_alignment = max_valid_field_length - current_field_length;
while required_alignment != 0 {
write!(formatted_code, " ")?;
required_alignment -= 1;
}
}
write!(
formatted_code,
" {} ",
storage_field.colon_token.ident().as_str(),
)?;
storage_field.ty.format(formatted_code, formatter)?;
write!(
formatted_code,
" {} ",
storage_field.eq_token.ident().as_str()
)?;
storage_field
.initializer
.format(formatted_code, formatter)?;
if value_pairs_iter.peek().is_some() {
writeln!(formatted_code, "{}", comma_token.ident().as_str())?;
} else if let Some(final_value) = &fields.final_value_opt {
final_value.format(formatted_code, formatter)?;
}
}
}
FieldAlignment::Off => fields.format(formatted_code, formatter)?,
}
Self::close_curly_brace(formatted_code, formatter)?;
formatter.shape.reset_line_settings();
Ok(())
}
}
impl CurlyBrace for ItemStorage {
fn open_curly_brace(
line: &mut String,
formatter: &mut Formatter,
) -> Result<(), FormatterError> {
let brace_style = formatter.config.items.item_brace_style;
formatter.shape.block_indent(&formatter.config);
let open_brace = Delimiter::Brace.as_open_char();
match brace_style {
ItemBraceStyle::AlwaysNextLine => {
write!(line, "\n{}", open_brace)?;
}
_ => {
write!(line, " {}", open_brace)?;
}
}
Ok(())
}
fn close_curly_brace(
line: &mut String,
formatter: &mut Formatter,
) -> Result<(), FormatterError> {
formatter.shape.block_unindent(&formatter.config);
write!(
line,
"{}{}",
formatter.shape.indent.to_string(&formatter.config)?,
Delimiter::Brace.as_close_char()
)?;
Ok(())
}
}
impl LeafSpans for ItemStorage {
fn leaf_spans(&self) -> Vec<ByteSpan> {
let mut collected_spans = vec![ByteSpan::from(self.storage_token.span())];
collected_spans.append(&mut self.fields.leaf_spans());
collected_spans
}
}
impl LeafSpans for StorageField {
fn leaf_spans(&self) -> Vec<ByteSpan> {
let mut collected_spans = vec![ByteSpan::from(self.name.span())];
collected_spans.push(ByteSpan::from(self.colon_token.span()));
collected_spans.append(&mut self.ty.leaf_spans());
collected_spans.push(ByteSpan::from(self.eq_token.span()));
collected_spans.append(&mut self.initializer.leaf_spans());
collected_spans
}
}