use std::collections::HashSet; use proc_macro::TokenStream; use proc_macro2_diagnostics::Diagnostic; use quote::{quote, quote_spanned, ToTokens}; use rstml::{ node::{KeyedAttribute, Node, NodeAttribute, NodeElement, NodeName}, Infallible, Parser, ParserConfig }; use syn::punctuated::Punctuated; use syn::{parse::Parse, parse_quote, spanned::Spanned, Expr, ExprLit, FnArg, ItemStruct, Token}; #[proc_macro] pub fn html(tokens: TokenStream) -> TokenStream { html_inner(tokens, false) } #[proc_macro] pub fn html_ide(tokens: TokenStream) -> TokenStream { html_inner(tokens, true) } fn is_empty_element(name: &str) -> bool { // https://developer.mozilla.org/en-US/docs/Glossary/Empty_element match name { "img" | "input" | "meta" | "link" | "hr" | "br" | "source" | "track" | "wbr" | "area" | "base" | "col" | "embed" | "param" => true, _ => false, } } fn empty_elements_set() -> HashSet<&'static str> { [ "area", "base", "br", "col", "embed", "hr", "img", "input", "link", "meta", "param", "source", "track", "wbr", ] .into_iter() .collect() } fn html_inner(tokens: TokenStream, ide_helper: bool) -> TokenStream { let config = ParserConfig::new() .recover_block(true) .element_close_use_default_wildcard_ident(true) .always_self_closed_elements(empty_elements_set()); let parser = Parser::new(config); let (nodes, errors) = parser.parse_recoverable(tokens).split_vec(); process_nodes(ide_helper, &nodes, errors).into() } fn process_nodes<'n>( ide_helper: bool, nodes: &'n Vec, errors: Vec, ) -> proc_macro2::TokenStream { let WalkNodesOutput { static_format: html_string, values, collected_elements: elements, diagnostics, } = walk_nodes(&nodes); let docs = if ide_helper { generate_tags_docs(elements) } else { vec![] }; let errors = errors .into_iter() .map(|e| e.emit_as_expr_tokens()) .chain(diagnostics); quote! { { // Make sure that "compile_error!(..);" can be used in this context. #(#errors;)* // Make sure that "enum x{};" and "let _x = crate::element;" can be used in this context #(#docs;)* format!(#html_string, #(vespid::FormatRender::new(#values)),*) } } } fn generate_tags_docs(elements: Vec<&NodeName>) -> Vec { // Mark some of elements as type, // and other as elements as fn in crate::docs, // to give an example how to link tag with docs. let elements_as_type: HashSet<&'static str> = vec!["html", "head", "meta", "link", "body", "div"] .into_iter() .collect(); elements .into_iter() .map(|e| { if elements_as_type.contains(&*e.to_string()) { let element = quote_spanned!(e.span() => enum); quote!({#element X{}}) } else { // let _ = crate::docs::element; let element = quote_spanned!(e.span() => element); quote!(let _ = crate::docs::#element) } }) .collect() } #[derive(Default)] struct WalkNodesOutput<'a> { static_format: String, // Use proc_macro2::TokenStream instead of syn::Expr // to provide more errors to the end user. values: Vec, // Additional diagnostic messages. diagnostics: Vec, // Collect elements to provide semantic highlight based on element tag. // No differences between open tag and closed tag. // Also multiple tags with same name can be present, // because we need to mark each of them. collected_elements: Vec<&'a NodeName>, } impl<'a> WalkNodesOutput<'a> { fn extend(&mut self, other: WalkNodesOutput<'a>) { self.static_format.push_str(&other.static_format); self.values.extend(other.values); self.diagnostics.extend(other.diagnostics); self.collected_elements.extend(other.collected_elements); } } fn walk_nodes<'a>(nodes: &'a Vec) -> WalkNodesOutput<'a> { let mut out = WalkNodesOutput::default(); for node in nodes { match node { Node::Doctype(doctype) => { let value = &doctype.value.to_token_stream_string(); out.static_format.push_str(&format!("", value)); } Node::Element(element) => { let name = element.name().to_string(); if !is_component_tag_name(&name) { match element.name() { NodeName::Block(block) => { out.static_format.push_str("<{}"); out.values.push(block.to_token_stream()); } _ => { out.static_format.push_str(&format!("<{}", name)); out.collected_elements.push(&element.open_tag.name); if let Some(e) = &element.close_tag { out.collected_elements.push(&e.name) } } } // attributes for attribute in element.attributes() { match attribute { NodeAttribute::Block(block) => { // If the nodes parent is an attribute we prefix with whitespace out.static_format.push(' '); out.static_format.push_str("{}"); out.values.push(block.to_token_stream()); } NodeAttribute::Attribute(attribute) => { let (static_format, value) = walk_attribute(attribute); out.static_format.push_str(&static_format); if let Some(value) = value { out.values.push(value); } } } } // Ignore childs of special Empty elements if is_empty_element(element.open_tag.name.to_string().as_str()) { out.static_format.push_str(" />"); if !element.children.is_empty() { let warning = proc_macro2_diagnostics::Diagnostic::spanned( element.open_tag.name.span(), proc_macro2_diagnostics::Level::Warning, "Element is processed as empty, and cannot have any child", ); out.diagnostics.push(warning.emit_as_expr_tokens()) } continue; } out.static_format.push('>'); // children let other_output = walk_nodes(&element.children); out.extend(other_output); match element.name() { NodeName::Block(block) => { out.static_format.push_str(""); out.values.push(block.to_token_stream()); } _ => { out.static_format.push_str(&format!("", name)); } } } else { // custom elements out.static_format.push_str("{}"); out.values .push(CustomElement::new(element).to_token_stream()); } } Node::Text(text) => { out.static_format.push_str(&text.value_string()); } Node::RawText(text) => { out.static_format.push_str(&text.to_string_best()); } Node::Fragment(fragment) => { let other_output = walk_nodes(&fragment.children); out.extend(other_output) } Node::Comment(comment) => { out.static_format.push_str(""); out.values.push(comment.value.to_token_stream()); } Node::Block(block) => { let block = block.try_block().unwrap(); let stmts = &block.stmts; out.static_format.push_str("{}"); out.values.push(quote!(#(#stmts)*)); } Node::Custom(_) => {} } } out } fn walk_attribute(attribute: &KeyedAttribute) -> (String, Option) { let mut static_format = String::new(); let mut format_value = None; let key = match attribute.key.to_string().as_str() { "as_" => "as".to_string(), _ => attribute.key.to_string(), }; static_format.push_str(&format!(" {}", key)); match attribute.value() { Some(Expr::Lit(ExprLit { lit: syn::Lit::Str(value), .. })) => { static_format.push_str(&format!( r#"="{}""#, html_escape::encode_unquoted_attribute(&value.value()) )); } Some(Expr::Lit(ExprLit { lit: syn::Lit::Bool(value), .. })) => { static_format.push_str(&format!(r#"="{}""#, value.value())); } Some(Expr::Lit(ExprLit { lit: syn::Lit::Int(value), .. })) => { static_format.push_str(&format!(r#"="{}""#, value.token())); } Some(Expr::Lit(ExprLit { lit: syn::Lit::Float(value), .. })) => { static_format.push_str(&format!(r#"="{}""#, value.token())); } Some(value) => { static_format.push_str(r#"="{}""#); format_value = Some( quote! {{ // (#value).escape_attribute() ::vespid::EscapeAttribute::escape_attribute(&#value) }} .into_token_stream(), ); } None => {} } (static_format, format_value) } fn is_component_tag_name(name: &str) -> bool { name.starts_with(|c: char| c.is_ascii_uppercase()) } struct CustomElement<'e> { e: &'e NodeElement, } impl<'e> CustomElement<'e> { fn new(e: &'e NodeElement) -> Self { CustomElement { e } } } impl<'e> ToTokens for CustomElement<'_> { fn to_tokens(&self, tokens: &mut proc_macro2::TokenStream) { let name = self.e.name(); let mut chain = vec![quote! { ::vespid::support::props_builder(&#name) }]; let children = &self.e.children; if !children.is_empty() { let c = process_nodes(false, children, vec![]); chain.push(quote! { .children(#c) }); } chain.push({ self.e .attributes() .iter() .map(|a| match a { NodeAttribute::Block(block) => { quote! { .push_attr( #[allow(unused_braces)] #block ) } } NodeAttribute::Attribute(attribute) => { let key = &attribute.key; let value = attribute.value().unwrap(); quote! { .#key(#value) } } }) .collect::() }); chain.push(quote! { .build() }); tokens.extend(quote! { #name(#(#chain)*).await }); } } fn props(input: proc_macro2::TokenStream) -> proc_macro2::TokenStream { let props = syn::parse2::(input).unwrap(); quote! { #props }.to_token_stream() } struct PropsStruct { name: syn::Ident, item: ItemStruct, } impl Parse for PropsStruct { fn parse(input: syn::parse::ParseStream) -> syn::Result { let item = input.parse::()?; let name = item.ident.clone(); Ok(PropsStruct { name, item }) } } impl ToTokens for PropsStruct { fn to_tokens(&self, tokens: &mut proc_macro2::TokenStream) { let name = &self.name; let item = &self.item; let builder_name = syn::Ident::new(&format!("{}Builder", name), proc_macro2::Span::call_site()); tokens.extend(quote! { #[derive(::vespid::typed_builder::TypedBuilder)] #[builder(doc, crate_module_path=::vespid::typed_builder)] #item impl ::vespid::support::Props for #name { type Builder = #builder_name; fn builder() -> Self::Builder { #name::builder() } } }); let has_attributes = item .fields .iter() .any(|field| field.ident.as_ref().unwrap().to_string() == "attributes"); if has_attributes { tokens.extend(quote! { impl #builder_name { pub fn push_attr(mut self, attr: A) -> Self { self.props.attributes.push_str(&format!("{} ", attr)); self } } }); } } } #[proc_macro_attribute] pub fn component(_attr: TokenStream, input: TokenStream) -> TokenStream { let comp = syn::parse_macro_input!(input as ComponentFn); quote! { #comp }.to_token_stream().into() } struct ComponentFn { item: syn::ItemFn, } impl Parse for ComponentFn { fn parse(input: syn::parse::ParseStream) -> syn::Result { let item = input.parse::()?; Ok(ComponentFn { item }) } } impl ToTokens for ComponentFn { fn to_tokens(&self, tokens: &mut proc_macro2::TokenStream) { let item = &self.item; let name = &item.sig.ident; let (defs, args) = match item.sig.inputs.len() { 0 => { // generate empty props let props_name = syn::Ident::new(&format!("{}Props", name), proc_macro2::Span::call_site()); ( props(quote! { pub struct #props_name{} }), quote! { _props: #props_name }, ) } // match if there is a single arg of type #nameProps 1 if matches!(item.sig.inputs.first().unwrap(), syn::FnArg::Typed(arg) if matches!(arg.ty.as_ref(), syn::Type::Path(p) if p.path.segments.last().unwrap().ident.to_string() == format!("{}Props", name))) => { let props = item.sig.inputs.first().unwrap(); (quote! {}, props.to_token_stream()) } _ => { let field_defs = &item .sig .inputs .clone() .into_iter() .map(|i| match i { FnArg::Receiver(_) => { panic!("receiver arguments unsupported"); } FnArg::Typed(mut t) => { if t.attrs.is_empty() { t.attrs.push(parse_quote! { #[builder(setter(into))] }); } t } }) .collect::>(); let field_names = item .sig .inputs .iter() .map(|i| match i { FnArg::Receiver(_) => { panic!("receiver arguments unsupported"); } FnArg::Typed(t) => &t.pat, }) .collect::>(); let props_name = syn::Ident::new(&format!("{}Props", name), proc_macro2::Span::call_site()); ( props(quote! { pub struct #props_name { #field_defs } }), quote! { #props_name { #field_names }: #props_name }, ) } }; let body = &item.block; let output = &item.sig.output; let vis = &item.vis; tokens.extend(quote! { #defs #[allow(non_snake_case)] #vis async fn #name(#args) #output { #body } }); } }