// A generic trait to abstract the rewriting of an element (of the AST). use std::cell::RefCell; use std::rc::Rc; use syntax::parse::ParseSess; use syntax::ptr; use syntax::source_map::{SourceMap, Span}; use crate::config::{Config, IndentStyle}; use crate::shape::Shape; use crate::skip::SkipContext; use crate::visitor::SnippetProvider; use crate::FormatReport; pub(crate) trait Rewrite { /// Rewrite self into shape. fn rewrite(&self, context: &RewriteContext<'_>, shape: Shape) -> Option; } impl Rewrite for ptr::P { fn rewrite(&self, context: &RewriteContext<'_>, shape: Shape) -> Option { (**self).rewrite(context, shape) } } #[derive(Clone)] pub(crate) struct RewriteContext<'a> { pub(crate) parse_session: &'a ParseSess, pub(crate) source_map: &'a SourceMap, pub(crate) config: &'a Config, pub(crate) inside_macro: RefCell, // Force block indent style even if we are using visual indent style. pub(crate) use_block: RefCell, // When `is_if_else_block` is true, unindent the comment on top // of the `else` or `else if`. pub(crate) is_if_else_block: RefCell, // When rewriting chain, veto going multi line except the last element pub(crate) force_one_line_chain: RefCell, pub(crate) snippet_provider: &'a SnippetProvider<'a>, // Used for `format_snippet` pub(crate) macro_rewrite_failure: RefCell, pub(crate) report: FormatReport, pub(crate) skip_context: SkipContext, pub(crate) skipped_range: Rc>>, } impl<'a> RewriteContext<'a> { pub(crate) fn snippet(&self, span: Span) -> &str { self.snippet_provider.span_to_snippet(span).unwrap() } /// Returns `true` if we should use block indent style for rewriting function call. pub(crate) fn use_block_indent(&self) -> bool { self.config.indent_style() == IndentStyle::Block || *self.use_block.borrow() } pub(crate) fn budget(&self, used_width: usize) -> usize { self.config.max_width().saturating_sub(used_width) } pub(crate) fn inside_macro(&self) -> bool { *self.inside_macro.borrow() } pub(crate) fn is_if_else_block(&self) -> bool { *self.is_if_else_block.borrow() } }