mirror of
https://github.com/DioxusLabs/dioxus
synced 2024-12-19 00:53:12 +00:00
212 lines
6.6 KiB
Rust
212 lines
6.6 KiB
Rust
|
use proc_macro2::TokenStream;
|
||
|
use quote::{quote, quote_spanned, ToTokens};
|
||
|
use syn::parse::{Parse, ParseStream};
|
||
|
use syn::spanned::Spanned;
|
||
|
use syn::{
|
||
|
parse_macro_input, Attribute, Block, FnArg, Ident, Item, ItemFn, ReturnType, Type, Visibility,
|
||
|
};
|
||
|
|
||
|
struct FunctionComponent {
|
||
|
block: Box<Block>,
|
||
|
props_type: Box<Type>,
|
||
|
arg: FnArg,
|
||
|
vis: Visibility,
|
||
|
attrs: Vec<Attribute>,
|
||
|
name: Ident,
|
||
|
return_type: Box<Type>,
|
||
|
}
|
||
|
|
||
|
impl Parse for FunctionComponent {
|
||
|
fn parse(input: ParseStream) -> syn::Result<Self> {
|
||
|
let parsed: Item = input.parse()?;
|
||
|
|
||
|
match parsed {
|
||
|
Item::Fn(func) => {
|
||
|
let ItemFn {
|
||
|
attrs,
|
||
|
vis,
|
||
|
sig,
|
||
|
block,
|
||
|
} = func;
|
||
|
|
||
|
if !sig.generics.params.is_empty() {
|
||
|
return Err(syn::Error::new_spanned(
|
||
|
sig.generics,
|
||
|
"function components can't contain generics",
|
||
|
));
|
||
|
}
|
||
|
|
||
|
if sig.asyncness.is_some() {
|
||
|
return Err(syn::Error::new_spanned(
|
||
|
sig.asyncness,
|
||
|
"function components can't be async",
|
||
|
));
|
||
|
}
|
||
|
|
||
|
if sig.constness.is_some() {
|
||
|
return Err(syn::Error::new_spanned(
|
||
|
sig.constness,
|
||
|
"const functions can't be function components",
|
||
|
));
|
||
|
}
|
||
|
|
||
|
if sig.abi.is_some() {
|
||
|
return Err(syn::Error::new_spanned(
|
||
|
sig.abi,
|
||
|
"extern functions can't be function components",
|
||
|
));
|
||
|
}
|
||
|
|
||
|
let return_type = match sig.output {
|
||
|
ReturnType::Default => {
|
||
|
return Err(syn::Error::new_spanned(
|
||
|
sig,
|
||
|
"function components must return `yew::Html`",
|
||
|
))
|
||
|
}
|
||
|
ReturnType::Type(_, ty) => ty,
|
||
|
};
|
||
|
|
||
|
let mut inputs = sig.inputs.into_iter();
|
||
|
let arg: FnArg = inputs
|
||
|
.next()
|
||
|
.unwrap_or_else(|| syn::parse_quote! { _: &() });
|
||
|
|
||
|
let ty = match &arg {
|
||
|
FnArg::Typed(arg) => match &*arg.ty {
|
||
|
Type::Reference(ty) => {
|
||
|
if ty.lifetime.is_some() {
|
||
|
return Err(syn::Error::new_spanned(
|
||
|
&ty.lifetime,
|
||
|
"reference must not have a lifetime",
|
||
|
));
|
||
|
}
|
||
|
|
||
|
if ty.mutability.is_some() {
|
||
|
return Err(syn::Error::new_spanned(
|
||
|
&ty.mutability,
|
||
|
"reference must not be mutable",
|
||
|
));
|
||
|
}
|
||
|
|
||
|
ty.elem.clone()
|
||
|
}
|
||
|
ty => {
|
||
|
let msg = format!(
|
||
|
"expected a reference to a `Properties` type (try: `&{}`)",
|
||
|
ty.to_token_stream()
|
||
|
);
|
||
|
return Err(syn::Error::new_spanned(ty, msg));
|
||
|
}
|
||
|
},
|
||
|
|
||
|
FnArg::Receiver(_) => {
|
||
|
return Err(syn::Error::new_spanned(
|
||
|
arg,
|
||
|
"function components can't accept a receiver",
|
||
|
));
|
||
|
}
|
||
|
};
|
||
|
|
||
|
// Checking after param parsing may make it a little inefficient
|
||
|
// but that's a requirement for better error messages in case of receivers
|
||
|
// `>0` because first one is already consumed.
|
||
|
if inputs.len() > 0 {
|
||
|
let params: TokenStream = inputs.map(|it| it.to_token_stream()).collect();
|
||
|
return Err(syn::Error::new_spanned(
|
||
|
params,
|
||
|
"function components can accept at most one parameter for the props",
|
||
|
));
|
||
|
}
|
||
|
|
||
|
Ok(Self {
|
||
|
props_type: ty,
|
||
|
block,
|
||
|
arg,
|
||
|
vis,
|
||
|
attrs,
|
||
|
name: sig.ident,
|
||
|
return_type,
|
||
|
})
|
||
|
}
|
||
|
item => Err(syn::Error::new_spanned(
|
||
|
item,
|
||
|
"`function_component` attribute can only be applied to functions",
|
||
|
)),
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
struct FunctionComponentName {
|
||
|
component_name: Ident,
|
||
|
}
|
||
|
|
||
|
impl Parse for FunctionComponentName {
|
||
|
fn parse(input: ParseStream) -> syn::Result<Self> {
|
||
|
if input.is_empty() {
|
||
|
return Err(input.error("expected identifier for the component"));
|
||
|
}
|
||
|
|
||
|
let component_name = input.parse()?;
|
||
|
|
||
|
Ok(Self { component_name })
|
||
|
}
|
||
|
}
|
||
|
|
||
|
#[proc_macro_attribute]
|
||
|
pub fn function_component(
|
||
|
attr: proc_macro::TokenStream,
|
||
|
item: proc_macro::TokenStream,
|
||
|
) -> proc_macro::TokenStream {
|
||
|
let item = parse_macro_input!(item as FunctionComponent);
|
||
|
let attr = parse_macro_input!(attr as FunctionComponentName);
|
||
|
|
||
|
function_component_impl(attr, item)
|
||
|
.unwrap_or_else(|err| err.to_compile_error())
|
||
|
.into()
|
||
|
}
|
||
|
|
||
|
fn function_component_impl(
|
||
|
name: FunctionComponentName,
|
||
|
component: FunctionComponent,
|
||
|
) -> syn::Result<TokenStream> {
|
||
|
let FunctionComponentName { component_name } = name;
|
||
|
|
||
|
let FunctionComponent {
|
||
|
block,
|
||
|
props_type,
|
||
|
arg,
|
||
|
vis,
|
||
|
attrs,
|
||
|
name: function_name,
|
||
|
return_type,
|
||
|
} = component;
|
||
|
|
||
|
if function_name == component_name {
|
||
|
return Err(syn::Error::new_spanned(
|
||
|
component_name,
|
||
|
"the component must not have the same name as the function",
|
||
|
));
|
||
|
}
|
||
|
|
||
|
let ret_type = quote_spanned!(return_type.span()=> ::yew::html::Html);
|
||
|
|
||
|
let quoted = quote! {
|
||
|
#[doc(hidden)]
|
||
|
#[allow(non_camel_case_types)]
|
||
|
#vis struct #function_name;
|
||
|
|
||
|
impl ::yew_functional::FunctionProvider for #function_name {
|
||
|
type TProps = #props_type;
|
||
|
|
||
|
fn run(#arg) -> #ret_type {
|
||
|
#block
|
||
|
}
|
||
|
}
|
||
|
|
||
|
#(#attrs)*
|
||
|
#vis type #component_name = ::yew_functional::FunctionComponent<#function_name>;
|
||
|
};
|
||
|
Ok(quoted)
|
||
|
}
|