use crate::innerlude::*;
use proc_macro2::TokenStream as TokenStream2;
use proc_macro2_diagnostics::SpanDiagnosticExt;
use quote::{quote, ToTokens, TokenStreamExt};
use std::{collections::HashSet, vec};
use syn::{
parse::{Parse, ParseStream},
spanned::Spanned,
token, AngleBracketedGenericArguments, Expr, Ident, PathArguments, Result,
};
#[derive(PartialEq, Eq, Clone, Debug)]
pub struct Component {
pub name: syn::Path,
pub generics: Option<AngleBracketedGenericArguments>,
pub fields: Vec<Attribute>,
pub component_literal_dyn_idx: Vec<DynIdx>,
pub spreads: Vec<Spread>,
pub brace: Option<token::Brace>,
pub children: TemplateBody,
pub dyn_idx: DynIdx,
pub diagnostics: Diagnostics,
}
impl Parse for Component {
fn parse(input: ParseStream) -> Result<Self> {
let mut name = input.parse::<syn::Path>()?;
let generics = normalize_path(&mut name);
if !input.peek(token::Brace) {
return Ok(Self::empty(name, generics));
};
let RsxBlock {
attributes: fields,
children,
brace,
spreads,
diagnostics,
} = input.parse::<RsxBlock>()?;
let literal_properties_count = fields
.iter()
.filter(|attr| matches!(attr.value, AttributeValue::AttrLiteral(_)))
.count();
let component_literal_dyn_idx = vec![DynIdx::default(); literal_properties_count];
let mut component = Self {
dyn_idx: DynIdx::default(),
children: TemplateBody::new(children),
name,
generics,
fields,
brace: Some(brace),
component_literal_dyn_idx,
spreads,
diagnostics,
};
component.validate_component_path();
component.validate_fields();
component.validate_component_spread();
Ok(component)
}
}
impl ToTokens for Component {
fn to_tokens(&self, tokens: &mut TokenStream2) {
let Self { name, generics, .. } = self;
let props = self.create_props();
let diagnostics = &self.diagnostics;
tokens.append_all(quote! {
dioxus_core::DynamicNode::Component({
use dioxus_core::prelude::Properties;
let __comp = ({
#props
}).into_vcomponent(
#name #generics,
);
#diagnostics
__comp
})
})
}
}
impl Component {
fn validate_component_path(&mut self) {
let path = &self.name;
if path.segments.len() == 1 {
let seg = path.segments.first().unwrap();
if seg.ident.to_string().chars().next().unwrap().is_lowercase()
&& !seg.ident.to_string().contains('_')
{
self.diagnostics.push(seg.ident.span().error(
"Component names must be uppercase, contain an underscore, or abe a path.",
));
}
}
if path
.segments
.iter()
.take(path.segments.len() - 1)
.any(|seg| seg.arguments != PathArguments::None)
{
self.diagnostics.push(path.span().error(
"Component names must not have path arguments. Only the last segment is allowed to have one.",
));
}
if !matches!(
path.segments.last().unwrap().arguments,
PathArguments::None | PathArguments::AngleBracketed(_)
) {
self.diagnostics.push(
path.span()
.error("Component names must have no arguments or angle bracketed arguments."),
);
}
}
fn validate_component_spread(&mut self) {
for spread in self.spreads.iter().skip(1) {
self.diagnostics.push(
spread
.expr
.span()
.error("Only one set of manual props is allowed for a component."),
);
}
}
pub fn get_key(&self) -> Option<&AttributeValue> {
self.fields
.iter()
.find(|attr| attr.name.is_likely_key())
.map(|attr| &attr.value)
}
fn validate_fields(&mut self) {
let mut seen = HashSet::new();
for field in self.fields.iter() {
match &field.name {
AttributeName::Custom(_) => {}
AttributeName::BuiltIn(k) => {
if !seen.contains(k) {
seen.insert(k);
} else {
self.diagnostics.push(k.span().error(
"Duplicate prop field found. Only one prop field per name is allowed.",
));
}
}
AttributeName::Spread(_) => {
unreachable!(
"Spread attributes should be handled in the spread validation step."
)
}
}
}
}
fn create_props(&self) -> TokenStream2 {
let manual_props = self.manual_props();
let name = &self.name;
let generics = &self.generics;
let mut tokens = if let Some(props) = manual_props.as_ref() {
quote! { let mut __manual_props = #props; }
} else {
quote! { fc_to_builder(#name #generics) }
};
tokens.append_all(self.add_fields_to_builder(
manual_props.map(|_| Ident::new("__manual_props", proc_macro2::Span::call_site())),
));
if !self.children.is_empty() {
let children = &self.children;
if manual_props.is_some() {
tokens.append_all(quote! { __manual_props.children = { #children }; })
} else {
tokens.append_all(quote! { .children( { #children } ) })
}
}
if manual_props.is_some() {
tokens.append_all(quote! { __manual_props })
} else {
tokens.append_all(quote! { .build() })
}
tokens
}
fn manual_props(&self) -> Option<&Expr> {
self.spreads.first().map(|spread| &spread.expr)
}
pub fn component_props(&self) -> impl Iterator<Item = &Attribute> {
self.fields
.iter()
.filter(move |attr| !attr.name.is_likely_key())
}
fn add_fields_to_builder(&self, manual_props: Option<Ident>) -> TokenStream2 {
let mut dynamic_literal_index = 0;
let mut tokens = TokenStream2::new();
for attribute in self.component_props() {
let release_value = attribute.value.to_token_stream();
let value = if let AttributeValue::AttrLiteral(literal) = &attribute.value {
let idx = self.component_literal_dyn_idx[dynamic_literal_index].get();
dynamic_literal_index += 1;
let debug_value = quote! { __dynamic_literal_pool.component_property(#idx, &*__template_read, #literal) };
quote! {
{
#[cfg(debug_assertions)]
{
#debug_value
}
#[cfg(not(debug_assertions))]
{
#release_value
}
}
}
} else {
release_value
};
match &attribute.name {
AttributeName::BuiltIn(name) => {
if let Some(manual_props) = &manual_props {
tokens.append_all(quote! { #manual_props.#name = #value; })
} else {
tokens.append_all(quote! { .#name(#value) })
}
}
AttributeName::Custom(name) => {
if manual_props.is_some() {
tokens.append_all(name.span().error(
"Custom attributes are not supported for components that are spread",
).emit_as_expr_tokens());
} else {
tokens.append_all(quote! { .push_attribute(#name, None, #value, false) })
}
}
AttributeName::Spread(_) => {}
}
}
tokens
}
fn empty(name: syn::Path, generics: Option<AngleBracketedGenericArguments>) -> Self {
let mut diagnostics = Diagnostics::new();
diagnostics.push(
name.span()
.error("Components must have a body")
.help("Components must have a body, for example `Component {}`"),
);
Component {
name,
generics,
brace: None,
fields: vec![],
spreads: vec![],
children: TemplateBody::new(vec![]),
component_literal_dyn_idx: vec![],
dyn_idx: DynIdx::default(),
diagnostics,
}
}
}
fn normalize_path(name: &mut syn::Path) -> Option<AngleBracketedGenericArguments> {
let seg = name.segments.last_mut()?;
let mut generics = match seg.arguments.clone() {
PathArguments::AngleBracketed(args) => {
seg.arguments = PathArguments::None;
Some(args)
}
_ => None,
};
if let Some(generics) = generics.as_mut() {
use syn::Token;
generics.colon2_token = Some(Token![::](proc_macro2::Span::call_site()));
}
generics
}
#[cfg(test)]
mod tests {
use super::*;
use prettier_please::PrettyUnparse;
use syn::parse_quote;
#[test]
fn parses() {
let input = quote! {
MyComponent {
key: "value {something}",
prop: "value",
..props,
div {
"Hello, world!"
}
}
};
let component: Component = syn::parse2(input).unwrap();
dbg!(component);
let input_without_manual_props = quote! {
MyComponent {
key: "value {something}",
prop: "value",
div { "Hello, world!" }
}
};
let component: Component = syn::parse2(input_without_manual_props).unwrap();
dbg!(component);
}
#[test]
fn rejects() {
let input = quote! {
myComponent {
key: "value",
prop: "value",
prop: "other",
..props,
..other_props,
div {
"Hello, world!"
}
}
};
let component: Component = syn::parse2(input).unwrap();
dbg!(component.diagnostics);
}
#[test]
fn to_tokens_properly() {
let input = quote! {
MyComponent {
key: "value {something}",
prop: "value",
prop: "value",
prop: "value",
prop: "value",
prop: 123,
..props,
div { "Hello, world!" }
}
};
let component: Component = syn::parse2(input).unwrap();
println!("{}", component.to_token_stream());
}
#[test]
fn to_tokens_no_manual_props() {
let input_without_manual_props = quote! {
MyComponent {
key: "value {something}",
named: "value {something}",
prop: "value",
count: 1,
div { "Hello, world!" }
}
};
let component: Component = syn::parse2(input_without_manual_props).unwrap();
println!("{}", component.to_token_stream().pretty_unparse());
}
#[test]
fn generics_params() {
let input_without_children = quote! {
Outlet::<R> {}
};
let component: crate::CallBody = syn::parse2(input_without_children).unwrap();
println!("{}", component.to_token_stream().pretty_unparse());
}
#[test]
fn generics_no_fish() {
let name = quote! { Outlet<R> };
let mut p = syn::parse2::<syn::Path>(name).unwrap();
let generics = normalize_path(&mut p);
assert!(generics.is_some());
let input_without_children = quote! {
div {
Component<Generic> {}
}
};
let component: BodyNode = syn::parse2(input_without_children).unwrap();
println!("{}", component.to_token_stream().pretty_unparse());
}
#[test]
fn fmt_passes_properly() {
let input = quote! {
Link { to: Route::List, class: "pure-button", "Go back" }
};
let component: Component = syn::parse2(input).unwrap();
println!("{}", component.to_token_stream().pretty_unparse());
}
#[test]
fn incomplete_components() {
let input = quote::quote! {
some::cool::Component
};
let _parsed: Component = syn::parse2(input).unwrap();
let input = quote::quote! {
some::cool::C
};
let _parsed: syn::Path = syn::parse2(input).unwrap();
}
#[test]
fn identifies_key() {
let input = quote! {
Link { key: "{value}", to: Route::List, class: "pure-button", "Go back" }
};
let component: Component = syn::parse2(input).unwrap();
assert_eq!(component.get_key(), Some(&parse_quote!("{value}")));
let properties = component
.component_props()
.map(|attr| attr.name.to_string())
.collect::<Vec<_>>();
assert_eq!(properties, ["to", "class"]);
}
}