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, 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_map(|attr| match &attr.name {
AttributeName::BuiltIn(key) if key == "key" => Some(&attr.value),
_ => None,
})
}
fn validate_fields(&mut self) {
let mut seen = HashSet::new();
for field in self.fields.iter() {
match &field.name {
AttributeName::Custom(name) => self.diagnostics.push(
name.span()
.error("Custom attributes are not supported for Components. Only known attributes are allowed."),
),
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) }
};
for (name, value) in self.make_field_idents() {
if manual_props.is_some() {
tokens.append_all(quote! { __manual_props.#name = #value; })
} else {
tokens.append_all(quote! { .#name(#value) })
}
}
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)
}
fn make_field_idents(&self) -> Vec<(TokenStream2, TokenStream2)> {
let mut dynamic_literal_index = 0;
self.fields
.iter()
.filter_map(move |attr| {
let Attribute { name, value, .. } = attr;
let attr = match name {
AttributeName::BuiltIn(k) => {
if k == "key" {
return None;
}
quote! { #k }
}
AttributeName::Custom(_) => return None,
AttributeName::Spread(_) => return None,
};
let release_value = value.to_token_stream();
let value = if let AttributeValue::AttrLiteral(literal) = &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
};
Some((attr, value))
})
.collect()
}
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));
}
generics
}
#[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();
}