123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220 |
- use std::{
- collections::{HashMap, VecDeque},
- fmt::{Result, Write},
- };
- use dioxus_rsx::{BodyNode, ElementAttr, ElementAttrNamed};
- use proc_macro2::{LineColumn, Span};
- use syn::{spanned::Spanned, Expr};
- #[derive(Default, Debug)]
- pub struct Buffer {
- pub src: Vec<String>,
- pub cached_formats: HashMap<Location, String>,
- pub buf: String,
- pub indent: usize,
- pub comments: VecDeque<usize>,
- }
- #[derive(Clone, Copy, Hash, PartialEq, Eq, Debug)]
- pub struct Location {
- pub line: usize,
- pub col: usize,
- }
- impl Location {
- pub fn new(start: LineColumn) -> Self {
- Self {
- line: start.line,
- col: start.column,
- }
- }
- }
- impl Buffer {
- // Create a new line and tab it to the current tab level
- pub fn tabbed_line(&mut self) -> Result {
- self.new_line()?;
- self.tab()
- }
- // Create a new line and tab it to the current tab level
- pub fn indented_tabbed_line(&mut self) -> Result {
- self.new_line()?;
- self.indented_tab()
- }
- pub fn tab(&mut self) -> Result {
- self.write_tabs(self.indent)
- }
- pub fn indented_tab(&mut self) -> Result {
- self.write_tabs(self.indent + 1)
- }
- pub fn write_tabs(&mut self, num: usize) -> std::fmt::Result {
- for _ in 0..num {
- write!(self.buf, " ")?
- }
- Ok(())
- }
- pub fn new_line(&mut self) -> Result {
- writeln!(self.buf)
- }
- // Expects to be written directly into place
- pub fn write_ident(&mut self, node: &BodyNode) -> Result {
- match node {
- BodyNode::Element(el) => self.write_element(el),
- BodyNode::Component(component) => self.write_component(component),
- BodyNode::Text(text) => self.write_text(text),
- BodyNode::RawExpr(exp) => self.write_raw_expr(exp),
- }
- }
- pub fn write_text(&mut self, text: &syn::LitStr) -> Result {
- write!(self.buf, "\"{}\"", text.value())
- }
- pub fn consume(self) -> Option<String> {
- Some(self.buf)
- }
- pub fn write_comments(&mut self, child: Span) -> Result {
- // collect all comments upwards
- let start = child.start();
- let line_start = start.line - 1;
- for (id, line) in self.src[..line_start].iter().enumerate().rev() {
- if line.trim().starts_with("//") || line.is_empty() {
- if id != 0 {
- self.comments.push_front(id);
- }
- } else {
- break;
- }
- }
- let mut last_was_empty = false;
- while let Some(comment_line) = self.comments.pop_front() {
- let line = &self.src[comment_line];
- if line.is_empty() {
- if !last_was_empty {
- self.new_line()?;
- }
- last_was_empty = true;
- } else {
- last_was_empty = false;
- self.tabbed_line()?;
- write!(self.buf, "{}", self.src[comment_line].trim())?;
- }
- }
- Ok(())
- }
- // Push out the indent level and write each component, line by line
- pub fn write_body_indented(&mut self, children: &[BodyNode]) -> Result {
- self.indent += 1;
- self.write_body_no_indent(children)?;
- self.indent -= 1;
- Ok(())
- }
- pub fn write_body_no_indent(&mut self, children: &[BodyNode]) -> Result {
- let last_child = children.len();
- for (idx, child) in children.iter().enumerate() {
- match child {
- // check if the expr is a short
- BodyNode::RawExpr(_) => {
- self.tabbed_line()?;
- self.write_ident(child)?;
- if idx != last_child - 1 {
- write!(self.buf, ",")?;
- }
- }
- _ => {
- if self.current_span_is_primary(child.span()) {
- self.write_comments(child.span())?;
- }
- self.tabbed_line()?;
- self.write_ident(child)?;
- }
- }
- }
- Ok(())
- }
- pub(crate) fn is_short_attrs(&mut self, attributes: &[ElementAttrNamed]) -> usize {
- let mut total = 0;
- for attr in attributes {
- if self.current_span_is_primary(attr.attr.flart()) {
- 'line: for line in self.src[..attr.attr.flart().start().line - 1].iter().rev() {
- match (line.trim().starts_with("//"), line.is_empty()) {
- (true, _) => return 100000,
- (_, true) => continue 'line,
- _ => break 'line,
- }
- }
- }
- total += match &attr.attr {
- ElementAttr::AttrText { value, name } => {
- value.value().len() + name.span().line_length() + 3
- }
- ElementAttr::AttrExpression { name, value } => {
- value.span().line_length() + name.span().line_length() + 3
- }
- ElementAttr::CustomAttrText { value, name } => {
- value.value().len() + name.value().len() + 3
- }
- ElementAttr::CustomAttrExpression { name, value } => {
- name.value().len() + value.span().line_length() + 3
- }
- ElementAttr::EventTokens { tokens, name } => {
- let location = Location::new(tokens.span().start());
- let len = if let std::collections::hash_map::Entry::Vacant(e) =
- self.cached_formats.entry(location)
- {
- let formatted = prettyplease::unparse_expr(tokens);
- let len = if formatted.contains('\n') {
- 10000
- } else {
- formatted.len()
- };
- e.insert(formatted);
- len
- } else {
- self.cached_formats[&location].len()
- };
- len + name.span().line_length() + 3
- }
- };
- }
- total
- }
- pub fn retrieve_formatted_expr(&mut self, expr: &Expr) -> &str {
- self.cached_formats
- .entry(Location::new(expr.span().start()))
- .or_insert_with(|| prettyplease::unparse_expr(expr))
- .as_str()
- }
- }
- trait SpanLength {
- fn line_length(&self) -> usize;
- }
- impl SpanLength for Span {
- fn line_length(&self) -> usize {
- self.end().line - self.start().line
- }
- }
|