dioxus/packages/core-macro/src/lib.rs

108 lines
3.3 KiB
Rust
Raw Normal View History

2021-02-28 22:30:10 +00:00
use proc_macro::TokenStream;
2022-05-29 13:04:08 +00:00
use quote::ToTokens;
2021-03-04 17:03:22 +00:00
use syn::parse_macro_input;
2021-01-20 17:04:27 +00:00
2022-02-19 04:14:17 +00:00
mod inlineprops;
mod props;
2022-04-24 06:35:52 +00:00
// mod rsx;
use dioxus_rsx as rsx;
2022-02-19 04:14:17 +00:00
#[proc_macro]
2021-02-28 22:30:10 +00:00
pub fn format_args_f(input: TokenStream) -> TokenStream {
2022-05-25 13:58:59 +00:00
use rsx::*;
let item = parse_macro_input!(input as IfmtInput);
2021-03-08 02:28:20 +00:00
format_args_f_impl(item)
.unwrap_or_else(|err| err.to_compile_error())
.into()
2021-01-20 17:04:27 +00:00
}
2021-12-21 05:46:10 +00:00
#[proc_macro_derive(Props, attributes(props))]
pub fn derive_typed_builder(input: proc_macro::TokenStream) -> proc_macro::TokenStream {
let input = parse_macro_input!(input as syn::DeriveInput);
match props::impl_my_derive(&input) {
Ok(output) => output.into(),
Err(error) => error.to_compile_error().into(),
}
}
/// The rsx! macro makes it easy for developers to write jsx-style markup in their components.
///
/// ## Complete Reference Guide:
/// ```ignore
#[doc = include_str!("../../../examples/rsx_usage.rs")]
/// ```
#[proc_macro]
pub fn rsx(s: TokenStream) -> TokenStream {
2022-06-26 06:23:19 +00:00
#[cfg(feature = "hot-reload")]
let rsx_text = s.to_string();
2022-02-22 21:34:06 +00:00
match syn::parse::<rsx::CallBody>(s) {
Err(err) => err.to_compile_error().into(),
2022-05-25 13:58:59 +00:00
Ok(body) => {
2022-06-26 06:23:19 +00:00
#[cfg(feature = "hot-reload")]
2022-05-25 13:58:59 +00:00
{
2022-06-02 18:33:08 +00:00
use dioxus_rsx_interpreter::captuered_context::CapturedContextBuilder;
2022-05-25 13:58:59 +00:00
2022-05-30 23:32:57 +00:00
match CapturedContextBuilder::from_call_body(body) {
2022-06-05 12:38:05 +00:00
Ok(captured) => {
let lazy = quote::quote! {
2022-06-04 19:26:37 +00:00
LazyNodes::new(move |__cx|{
let code_location = get_line_num!();
let captured = #captured;
let text = #rsx_text;
2022-06-15 17:58:08 +00:00
resolve_scope(code_location, text, captured, __cx)
})
2022-06-05 12:38:05 +00:00
};
if let Some(cx) = captured.custom_context {
quote::quote! {
#cx.render(#lazy)
}
.into()
} else {
lazy.into()
2022-05-30 23:32:57 +00:00
}
2022-05-25 16:20:07 +00:00
}
2022-05-30 23:32:57 +00:00
Err(err) => err.into_compile_error().into(),
2022-05-25 16:20:07 +00:00
}
2022-05-25 13:58:59 +00:00
}
2022-06-26 06:23:19 +00:00
#[cfg(not(feature = "hot-reload"))]
2022-05-25 13:58:59 +00:00
body.to_token_stream().into()
}
}
}
2021-11-03 04:35:56 +00:00
/// Derive props for a component within the component definition.
///
/// This macro provides a simple transformation from `Scope<{}>` to `Scope<P>`,
/// removing some boilerplate when defining props.
///
/// You don't *need* to use this macro at all, but it can be helpful in cases where
/// you would be repeating a lot of the usual Rust boilerplate.
///
/// # Example
/// ```ignore
/// #[inline_props]
2021-12-29 04:20:01 +00:00
/// fn app(cx: Scope, bob: String) -> Element {
/// cx.render(rsx!("hello, {bob}"))
/// }
///
/// // is equivalent to
///
/// #[derive(PartialEq, Props)]
/// struct AppProps {
/// bob: String,
/// }
///
/// fn app(cx: Scope<AppProps>) -> Element {
/// cx.render(rsx!("hello, {bob}"))
/// }
/// ```
#[proc_macro_attribute]
pub fn inline_props(_args: proc_macro::TokenStream, s: TokenStream) -> TokenStream {
match syn::parse::<inlineprops::InlinePropsBody>(s) {
Err(e) => e.to_compile_error().into(),
Ok(s) => s.to_token_stream().into(),
}
}