derive_component.rs

 1use convert_case::{Case, Casing};
 2use proc_macro::TokenStream;
 3use quote::quote;
 4use syn::{DeriveInput, Lit, Meta, MetaList, MetaNameValue, NestedMeta, parse_macro_input};
 5
 6pub fn derive_into_component(input: TokenStream) -> TokenStream {
 7    let input = parse_macro_input!(input as DeriveInput);
 8    let mut scope_val = None;
 9    let mut description_val = None;
10
11    for attr in &input.attrs {
12        if attr.path.is_ident("component") {
13            if let Ok(Meta::List(MetaList { nested, .. })) = attr.parse_meta() {
14                for item in nested {
15                    if let NestedMeta::Meta(Meta::NameValue(MetaNameValue {
16                        path,
17                        lit: Lit::Str(s),
18                        ..
19                    })) = item
20                    {
21                        let ident = path.get_ident().map(|i| i.to_string()).unwrap_or_default();
22                        if ident == "scope" {
23                            scope_val = Some(s.value());
24                        } else if ident == "description" {
25                            description_val = Some(s.value());
26                        }
27                    }
28                }
29            }
30        }
31    }
32
33    let name = &input.ident;
34
35    let scope_impl = if let Some(s) = scope_val {
36        let scope_str = s.clone();
37        quote! {
38            fn scope() -> Option<component::ComponentScope> {
39                Some(component::ComponentScope::from(#scope_str))
40            }
41        }
42    } else {
43        quote! {
44            fn scope() -> Option<component::ComponentScope> {
45                None
46            }
47        }
48    };
49
50    let description_impl = if let Some(desc) = description_val {
51        quote! {
52            fn description() -> Option<&'static str> {
53                Some(#desc)
54            }
55        }
56    } else {
57        quote! {}
58    };
59
60    let register_component_name = syn::Ident::new(
61        &format!(
62            "__register_component_{}",
63            Casing::to_case(&name.to_string(), Case::Snake)
64        ),
65        name.span(),
66    );
67    let register_preview_name = syn::Ident::new(
68        &format!(
69            "__register_preview_{}",
70            Casing::to_case(&name.to_string(), Case::Snake)
71        ),
72        name.span(),
73    );
74
75    let expanded = quote! {
76        impl component::Component for #name {
77            #scope_impl
78
79            fn name() -> &'static str {
80                stringify!(#name)
81            }
82
83            #description_impl
84        }
85
86        #[linkme::distributed_slice(component::__ALL_COMPONENTS)]
87        fn #register_component_name() {
88            component::register_component::<#name>();
89        }
90
91        #[linkme::distributed_slice(component::__ALL_PREVIEWS)]
92        fn #register_preview_name() {
93            component::register_preview::<#name>();
94        }
95    };
96
97    expanded.into()
98}