2021-01-11 16:09:33 +01:00
|
|
|
//! This crate contains procedural macros to make creating Rhai plugin modules much easier.
|
2020-08-01 18:52:26 +02:00
|
|
|
//!
|
2020-09-30 17:27:41 +02:00
|
|
|
//! # Export an Entire Rust Module to a Rhai `Module`
|
2020-08-01 18:52:26 +02:00
|
|
|
//!
|
|
|
|
//! ```
|
2020-08-03 02:27:19 +02:00
|
|
|
//! use rhai::{EvalAltResult, FLOAT};
|
2020-08-01 18:52:26 +02:00
|
|
|
//! use rhai::plugin::*;
|
|
|
|
//! use rhai::module_resolvers::*;
|
|
|
|
//!
|
2020-09-30 16:55:40 +02:00
|
|
|
//! #[export_module]
|
|
|
|
//! mod advanced_math {
|
2020-10-22 06:26:44 +02:00
|
|
|
//! pub const MYSTIC_NUMBER: FLOAT = 42.0;
|
2020-08-01 18:52:26 +02:00
|
|
|
//!
|
|
|
|
//! pub fn euclidean_distance(x1: FLOAT, y1: FLOAT, x2: FLOAT, y2: FLOAT) -> FLOAT {
|
|
|
|
//! ((y2 - y1).abs().powf(2.0) + (x2 -x1).abs().powf(2.0)).sqrt()
|
|
|
|
//! }
|
|
|
|
//! }
|
|
|
|
//!
|
2020-10-22 06:26:44 +02:00
|
|
|
//! # fn main() -> Result<(), Box<EvalAltResult>> {
|
|
|
|
//! let mut engine = Engine::new();
|
|
|
|
//! let m = exported_module!(advanced_math);
|
|
|
|
//! let mut r = StaticModuleResolver::new();
|
|
|
|
//! r.insert("Math::Advanced", m);
|
2020-12-26 06:05:57 +01:00
|
|
|
//! engine.set_module_resolver(r);
|
2020-10-22 06:26:44 +02:00
|
|
|
//!
|
|
|
|
//! assert_eq!(engine.eval::<FLOAT>(
|
|
|
|
//! r#"
|
|
|
|
//! import "Math::Advanced" as math;
|
|
|
|
//! math::euclidean_distance(0.0, 1.0, 0.0, math::MYSTIC_NUMBER)
|
|
|
|
//! "#)?, 41.0);
|
|
|
|
//! # Ok(())
|
|
|
|
//! # }
|
2020-08-01 18:52:26 +02:00
|
|
|
//! ```
|
|
|
|
//!
|
2020-09-30 17:27:41 +02:00
|
|
|
//! # Register a Rust Function with a Rhai `Module`
|
2020-08-01 18:52:26 +02:00
|
|
|
//!
|
|
|
|
//! ```
|
2021-03-15 04:36:30 +01:00
|
|
|
//! use rhai::{EvalAltResult, FLOAT, Module};
|
2020-08-01 18:52:26 +02:00
|
|
|
//! use rhai::plugin::*;
|
|
|
|
//! use rhai::module_resolvers::*;
|
|
|
|
//!
|
2020-09-30 16:55:40 +02:00
|
|
|
//! #[export_fn]
|
|
|
|
//! fn distance_function(x1: FLOAT, y1: FLOAT, x2: FLOAT, y2: FLOAT) -> FLOAT {
|
2020-08-01 18:52:26 +02:00
|
|
|
//! ((y2 - y1).abs().powf(2.0) + (x2 -x1).abs().powf(2.0)).sqrt()
|
|
|
|
//! }
|
|
|
|
//!
|
2020-10-22 06:26:44 +02:00
|
|
|
//! # fn main() -> Result<(), Box<EvalAltResult>> {
|
|
|
|
//! let mut engine = Engine::new();
|
|
|
|
//! engine.register_fn("get_mystic_number", || 42.0 as FLOAT);
|
|
|
|
//! let mut m = Module::new();
|
|
|
|
//! set_exported_fn!(m, "euclidean_distance", distance_function);
|
|
|
|
//! let mut r = StaticModuleResolver::new();
|
|
|
|
//! r.insert("Math::Advanced", m);
|
2020-12-26 06:05:57 +01:00
|
|
|
//! engine.set_module_resolver(r);
|
2020-10-22 06:26:44 +02:00
|
|
|
//!
|
|
|
|
//! assert_eq!(engine.eval::<FLOAT>(
|
|
|
|
//! r#"
|
|
|
|
//! import "Math::Advanced" as math;
|
|
|
|
//! math::euclidean_distance(0.0, 1.0, 0.0, get_mystic_number())
|
|
|
|
//! "#)?, 41.0);
|
|
|
|
//! # Ok(())
|
|
|
|
//! # }
|
2020-08-01 18:52:26 +02:00
|
|
|
//! ```
|
|
|
|
//!
|
2020-09-30 17:27:41 +02:00
|
|
|
//! # Register a Plugin Function with an `Engine`
|
2020-08-13 06:34:53 +02:00
|
|
|
//!
|
|
|
|
//! ```
|
2021-03-15 04:36:30 +01:00
|
|
|
//! use rhai::{EvalAltResult, FLOAT, Module};
|
2020-08-13 06:34:53 +02:00
|
|
|
//! use rhai::plugin::*;
|
|
|
|
//! use rhai::module_resolvers::*;
|
|
|
|
//!
|
2020-10-22 06:26:44 +02:00
|
|
|
//! #[export_fn]
|
2020-08-13 06:34:53 +02:00
|
|
|
//! pub fn distance_function(x1: FLOAT, y1: FLOAT, x2: FLOAT, y2: FLOAT) -> FLOAT {
|
|
|
|
//! ((y2 - y1).abs().powf(2.0) + (x2 -x1).abs().powf(2.0)).sqrt()
|
|
|
|
//! }
|
|
|
|
//!
|
2020-10-22 06:26:44 +02:00
|
|
|
//! # fn main() -> Result<(), Box<EvalAltResult>> {
|
|
|
|
//! let mut engine = Engine::new();
|
|
|
|
//! engine.register_fn("get_mystic_number", || { 42 as FLOAT });
|
|
|
|
//! register_exported_fn!(engine, "euclidean_distance", distance_function);
|
2020-08-13 06:34:53 +02:00
|
|
|
//!
|
2020-10-22 06:26:44 +02:00
|
|
|
//! assert_eq!(engine.eval::<FLOAT>(
|
|
|
|
//! "euclidean_distance(0.0, 1.0, 0.0, get_mystic_number())"
|
|
|
|
//! )?, 41.0);
|
|
|
|
//! # Ok(())
|
|
|
|
//! # }
|
2020-08-13 06:34:53 +02:00
|
|
|
//! ```
|
|
|
|
//!
|
2020-08-01 18:52:26 +02:00
|
|
|
|
2020-08-16 17:41:59 +02:00
|
|
|
use quote::quote;
|
2021-10-20 09:30:11 +02:00
|
|
|
use syn::{parse_macro_input, spanned::Spanned};
|
2020-08-01 18:52:26 +02:00
|
|
|
|
2020-08-24 00:53:30 +02:00
|
|
|
mod attrs;
|
2020-08-01 18:52:26 +02:00
|
|
|
mod function;
|
|
|
|
mod module;
|
2020-08-13 06:34:53 +02:00
|
|
|
mod register;
|
2020-08-01 18:52:26 +02:00
|
|
|
mod rhai_module;
|
|
|
|
|
2020-09-05 21:10:46 +02:00
|
|
|
#[cfg(test)]
|
|
|
|
mod test;
|
|
|
|
|
2020-09-30 16:55:40 +02:00
|
|
|
/// Attribute, when put on a Rust function, turns it into a _plugin function_.
|
|
|
|
///
|
|
|
|
/// # Usage
|
|
|
|
///
|
2020-10-07 16:51:43 +02:00
|
|
|
/// ```
|
|
|
|
/// # use rhai::{Engine, EvalAltResult};
|
2020-09-30 16:55:40 +02:00
|
|
|
/// use rhai::plugin::*;
|
|
|
|
///
|
|
|
|
/// #[export_fn]
|
2020-10-07 16:51:43 +02:00
|
|
|
/// fn my_plugin_function(x: i64) -> i64 {
|
|
|
|
/// x * 2
|
2020-09-30 16:55:40 +02:00
|
|
|
/// }
|
2020-10-07 16:51:43 +02:00
|
|
|
///
|
|
|
|
/// # fn main() -> Result<(), Box<EvalAltResult>> {
|
|
|
|
/// let mut engine = Engine::new();
|
|
|
|
///
|
|
|
|
/// register_exported_fn!(engine, "func", my_plugin_function);
|
|
|
|
///
|
|
|
|
/// assert_eq!(engine.eval::<i64>("func(21)")?, 42);
|
|
|
|
/// # Ok(())
|
|
|
|
/// # }
|
2020-09-30 16:55:40 +02:00
|
|
|
/// ```
|
2020-08-01 18:52:26 +02:00
|
|
|
#[proc_macro_attribute]
|
2020-08-02 09:39:08 +02:00
|
|
|
pub fn export_fn(
|
2020-08-08 04:19:17 +02:00
|
|
|
args: proc_macro::TokenStream,
|
2020-08-02 09:39:08 +02:00
|
|
|
input: proc_macro::TokenStream,
|
|
|
|
) -> proc_macro::TokenStream {
|
2020-08-16 12:24:42 +02:00
|
|
|
let mut output = proc_macro2::TokenStream::from(input.clone());
|
2020-08-08 16:31:15 +02:00
|
|
|
|
2020-08-24 00:53:30 +02:00
|
|
|
let parsed_params = match crate::attrs::outer_item_attributes(args.into(), "export_fn") {
|
|
|
|
Ok(args) => args,
|
2021-10-20 09:30:11 +02:00
|
|
|
Err(err) => return err.to_compile_error().into(),
|
2020-08-24 00:53:30 +02:00
|
|
|
};
|
2020-08-24 00:22:29 +02:00
|
|
|
let mut function_def = parse_macro_input!(input as function::ExportedFn);
|
2021-10-20 09:30:11 +02:00
|
|
|
|
|
|
|
if !function_def.cfg_attrs().is_empty() {
|
|
|
|
return syn::Error::new(
|
|
|
|
function_def.cfg_attrs()[0].span(),
|
|
|
|
"`cfg` attributes are not allowed for `export_fn`",
|
|
|
|
)
|
|
|
|
.to_compile_error()
|
|
|
|
.into();
|
|
|
|
}
|
|
|
|
|
2020-08-24 00:22:29 +02:00
|
|
|
if let Err(e) = function_def.set_params(parsed_params) {
|
|
|
|
return e.to_compile_error().into();
|
|
|
|
}
|
2020-08-08 16:31:15 +02:00
|
|
|
|
2020-08-24 00:22:29 +02:00
|
|
|
output.extend(function_def.generate());
|
2020-08-01 18:52:26 +02:00
|
|
|
proc_macro::TokenStream::from(output)
|
|
|
|
}
|
|
|
|
|
2020-09-30 16:55:40 +02:00
|
|
|
/// Attribute, when put on a Rust module, turns it into a _plugin module_.
|
|
|
|
///
|
|
|
|
/// # Usage
|
|
|
|
///
|
2020-10-07 16:51:43 +02:00
|
|
|
/// ```
|
|
|
|
/// # use rhai::{Engine, Module, EvalAltResult};
|
2020-09-30 16:55:40 +02:00
|
|
|
/// use rhai::plugin::*;
|
|
|
|
///
|
|
|
|
/// #[export_module]
|
|
|
|
/// mod my_plugin_module {
|
2020-10-07 16:51:43 +02:00
|
|
|
/// pub fn foo(x: i64) -> i64 { x * 2 }
|
|
|
|
/// pub fn bar() -> i64 { 21 }
|
2020-09-30 16:55:40 +02:00
|
|
|
/// }
|
2020-10-07 16:51:43 +02:00
|
|
|
///
|
|
|
|
/// # fn main() -> Result<(), Box<EvalAltResult>> {
|
|
|
|
/// let mut engine = Engine::new();
|
|
|
|
///
|
|
|
|
/// let module = exported_module!(my_plugin_module);
|
|
|
|
///
|
2020-12-23 08:15:47 +01:00
|
|
|
/// engine.register_global_module(module.into());
|
2020-10-07 16:51:43 +02:00
|
|
|
///
|
|
|
|
/// assert_eq!(engine.eval::<i64>("foo(bar())")?, 42);
|
|
|
|
/// # Ok(())
|
|
|
|
/// # }
|
2020-09-30 16:55:40 +02:00
|
|
|
/// ```
|
2020-08-01 18:52:26 +02:00
|
|
|
#[proc_macro_attribute]
|
2020-08-02 09:39:08 +02:00
|
|
|
pub fn export_module(
|
2020-09-02 06:15:22 +02:00
|
|
|
args: proc_macro::TokenStream,
|
2020-08-02 09:39:08 +02:00
|
|
|
input: proc_macro::TokenStream,
|
|
|
|
) -> proc_macro::TokenStream {
|
2020-09-02 06:15:22 +02:00
|
|
|
let parsed_params = match crate::attrs::outer_item_attributes(args.into(), "export_module") {
|
|
|
|
Ok(args) => args,
|
2021-10-20 09:30:11 +02:00
|
|
|
Err(err) => return err.to_compile_error().into(),
|
2020-09-02 06:15:22 +02:00
|
|
|
};
|
|
|
|
let mut module_def = parse_macro_input!(input as module::Module);
|
|
|
|
if let Err(e) = module_def.set_params(parsed_params) {
|
|
|
|
return e.to_compile_error().into();
|
|
|
|
}
|
|
|
|
|
2020-08-01 18:52:26 +02:00
|
|
|
let tokens = module_def.generate();
|
|
|
|
proc_macro::TokenStream::from(tokens)
|
|
|
|
}
|
|
|
|
|
2021-10-22 04:43:30 +02:00
|
|
|
/// Macro to generate a Rhai `Module` from a _plugin module_ defined via [`#[export_module]`][macro@export_module].
|
2020-09-30 16:55:40 +02:00
|
|
|
///
|
|
|
|
/// # Usage
|
|
|
|
///
|
2020-10-07 16:51:43 +02:00
|
|
|
/// ```
|
|
|
|
/// # use rhai::{Engine, Module, EvalAltResult};
|
2020-09-30 16:55:40 +02:00
|
|
|
/// use rhai::plugin::*;
|
|
|
|
///
|
|
|
|
/// #[export_module]
|
|
|
|
/// mod my_plugin_module {
|
2020-10-07 16:51:43 +02:00
|
|
|
/// pub fn foo(x: i64) -> i64 { x * 2 }
|
|
|
|
/// pub fn bar() -> i64 { 21 }
|
2020-09-30 16:55:40 +02:00
|
|
|
/// }
|
|
|
|
///
|
2020-10-07 16:51:43 +02:00
|
|
|
/// # fn main() -> Result<(), Box<EvalAltResult>> {
|
|
|
|
/// let mut engine = Engine::new();
|
|
|
|
///
|
2020-09-30 16:55:40 +02:00
|
|
|
/// let module = exported_module!(my_plugin_module);
|
2020-10-07 16:51:43 +02:00
|
|
|
///
|
2020-12-23 03:08:43 +01:00
|
|
|
/// engine.register_global_module(module.into());
|
2020-10-07 16:51:43 +02:00
|
|
|
///
|
|
|
|
/// assert_eq!(engine.eval::<i64>("foo(bar())")?, 42);
|
|
|
|
/// # Ok(())
|
|
|
|
/// # }
|
2020-09-30 16:55:40 +02:00
|
|
|
/// ```
|
2020-08-01 18:52:26 +02:00
|
|
|
#[proc_macro]
|
|
|
|
pub fn exported_module(module_path: proc_macro::TokenStream) -> proc_macro::TokenStream {
|
|
|
|
let module_path = parse_macro_input!(module_path as syn::Path);
|
2021-03-24 02:56:25 +01:00
|
|
|
proc_macro::TokenStream::from(quote::quote! {
|
2020-08-06 08:10:27 +02:00
|
|
|
#module_path::rhai_module_generate()
|
2021-03-24 02:56:25 +01:00
|
|
|
})
|
2020-08-01 18:52:26 +02:00
|
|
|
}
|
|
|
|
|
2020-09-30 16:55:40 +02:00
|
|
|
/// Macro to combine a _plugin module_ into an existing module.
|
|
|
|
///
|
|
|
|
/// Functions and variables in the plugin module overrides any existing similarly-named
|
|
|
|
/// functions and variables in the target module.
|
|
|
|
///
|
2021-01-11 16:09:33 +01:00
|
|
|
/// This call is intended to be used within the [`def_package!`][crate::def_package] macro to define
|
|
|
|
/// a custom package based on a plugin module.
|
2020-09-30 16:55:40 +02:00
|
|
|
///
|
|
|
|
/// All sub-modules, if any, in the plugin module are _flattened_ and their functions/variables
|
|
|
|
/// registered at the top level because packages require so.
|
|
|
|
///
|
|
|
|
/// The text string name in the second parameter can be anything and is reserved for future use;
|
|
|
|
/// it is recommended to be an ID string that uniquely identifies the plugin module.
|
|
|
|
///
|
|
|
|
/// # Usage
|
|
|
|
///
|
2020-10-07 16:51:43 +02:00
|
|
|
/// ```
|
|
|
|
/// # use rhai::{Engine, Module, EvalAltResult};
|
2020-09-30 16:55:40 +02:00
|
|
|
/// use rhai::plugin::*;
|
|
|
|
///
|
|
|
|
/// #[export_module]
|
|
|
|
/// mod my_plugin_module {
|
2020-10-07 16:51:43 +02:00
|
|
|
/// pub fn foo(x: i64) -> i64 { x * 2 }
|
|
|
|
/// pub fn bar() -> i64 { 21 }
|
2020-09-30 16:55:40 +02:00
|
|
|
/// }
|
|
|
|
///
|
2020-10-07 16:51:43 +02:00
|
|
|
/// # fn main() -> Result<(), Box<EvalAltResult>> {
|
|
|
|
/// let mut engine = Engine::new();
|
2020-09-30 16:55:40 +02:00
|
|
|
///
|
2020-10-07 16:51:43 +02:00
|
|
|
/// let mut module = Module::new();
|
2020-09-30 16:55:40 +02:00
|
|
|
/// combine_with_exported_module!(&mut module, "my_plugin_module_ID", my_plugin_module);
|
2020-10-07 16:51:43 +02:00
|
|
|
///
|
2020-12-23 03:08:43 +01:00
|
|
|
/// engine.register_global_module(module.into());
|
2020-10-07 16:51:43 +02:00
|
|
|
///
|
|
|
|
/// assert_eq!(engine.eval::<i64>("foo(bar())")?, 42);
|
|
|
|
/// # Ok(())
|
|
|
|
/// # }
|
2020-09-30 16:55:40 +02:00
|
|
|
/// ```
|
2020-09-13 16:12:11 +02:00
|
|
|
#[proc_macro]
|
|
|
|
pub fn combine_with_exported_module(args: proc_macro::TokenStream) -> proc_macro::TokenStream {
|
2021-03-24 02:56:25 +01:00
|
|
|
match crate::register::parse_register_macro(args) {
|
|
|
|
Ok((module_expr, _export_name, module_path)) => proc_macro::TokenStream::from(quote! {
|
|
|
|
#module_path::rhai_generate_into_module(#module_expr, true);
|
|
|
|
}),
|
|
|
|
Err(e) => e.to_compile_error().into(),
|
|
|
|
}
|
2020-09-13 16:12:11 +02:00
|
|
|
}
|
|
|
|
|
2021-10-22 04:43:30 +02:00
|
|
|
/// Macro to register a _plugin function_ (defined via [`#[export_fn]`][macro@export_fn]) into an `Engine`.
|
2020-09-30 16:55:40 +02:00
|
|
|
///
|
|
|
|
/// # Usage
|
|
|
|
///
|
2020-10-07 16:51:43 +02:00
|
|
|
/// ```
|
|
|
|
/// # use rhai::{Engine, EvalAltResult};
|
2020-09-30 16:55:40 +02:00
|
|
|
/// use rhai::plugin::*;
|
|
|
|
///
|
|
|
|
/// #[export_fn]
|
2020-10-07 16:51:43 +02:00
|
|
|
/// fn my_plugin_function(x: i64) -> i64 {
|
|
|
|
/// x * 2
|
2020-09-30 16:55:40 +02:00
|
|
|
/// }
|
|
|
|
///
|
2020-10-07 16:51:43 +02:00
|
|
|
/// # fn main() -> Result<(), Box<EvalAltResult>> {
|
2020-09-30 16:55:40 +02:00
|
|
|
/// let mut engine = Engine::new();
|
|
|
|
///
|
2020-10-07 16:51:43 +02:00
|
|
|
/// register_exported_fn!(engine, "func", my_plugin_function);
|
|
|
|
///
|
|
|
|
/// assert_eq!(engine.eval::<i64>("func(21)")?, 42);
|
|
|
|
/// # Ok(())
|
|
|
|
/// # }
|
2020-09-30 16:55:40 +02:00
|
|
|
/// ```
|
2020-08-01 18:52:26 +02:00
|
|
|
#[proc_macro]
|
|
|
|
pub fn register_exported_fn(args: proc_macro::TokenStream) -> proc_macro::TokenStream {
|
2021-03-24 02:56:25 +01:00
|
|
|
match crate::register::parse_register_macro(args) {
|
|
|
|
Ok((engine_expr, export_name, rust_mod_path)) => {
|
|
|
|
let gen_mod_path = crate::register::generated_module_path(&rust_mod_path);
|
|
|
|
proc_macro::TokenStream::from(quote! {
|
2021-03-24 06:17:52 +01:00
|
|
|
#engine_expr.register_result_fn(#export_name, #gen_mod_path::dynamic_result_fn);
|
2021-03-24 02:56:25 +01:00
|
|
|
})
|
|
|
|
}
|
|
|
|
Err(e) => e.to_compile_error().into(),
|
|
|
|
}
|
2020-08-13 06:34:53 +02:00
|
|
|
}
|
|
|
|
|
2020-09-30 16:55:40 +02:00
|
|
|
/// Macro to register a _plugin function_ into a Rhai `Module`.
|
|
|
|
///
|
|
|
|
/// # Usage
|
|
|
|
///
|
2020-10-07 16:51:43 +02:00
|
|
|
/// ```
|
|
|
|
/// # use rhai::{Engine, EvalAltResult};
|
2020-09-30 16:55:40 +02:00
|
|
|
/// use rhai::plugin::*;
|
|
|
|
///
|
|
|
|
/// #[export_fn]
|
2020-10-07 16:51:43 +02:00
|
|
|
/// fn my_plugin_function(x: i64) -> i64 {
|
|
|
|
/// x * 2
|
2020-09-30 16:55:40 +02:00
|
|
|
/// }
|
|
|
|
///
|
2020-10-07 16:51:43 +02:00
|
|
|
/// # fn main() -> Result<(), Box<EvalAltResult>> {
|
|
|
|
/// let mut engine = Engine::new();
|
|
|
|
///
|
2020-09-30 16:55:40 +02:00
|
|
|
/// let mut module = Module::new();
|
2020-10-07 16:51:43 +02:00
|
|
|
/// set_exported_fn!(module, "func", my_plugin_function);
|
|
|
|
///
|
2020-12-23 03:08:43 +01:00
|
|
|
/// engine.register_global_module(module.into());
|
2020-09-30 16:55:40 +02:00
|
|
|
///
|
2020-10-07 16:51:43 +02:00
|
|
|
/// assert_eq!(engine.eval::<i64>("func(21)")?, 42);
|
|
|
|
/// # Ok(())
|
|
|
|
/// # }
|
2020-09-30 16:55:40 +02:00
|
|
|
/// ```
|
2020-08-13 06:34:53 +02:00
|
|
|
#[proc_macro]
|
|
|
|
pub fn set_exported_fn(args: proc_macro::TokenStream) -> proc_macro::TokenStream {
|
2021-03-24 02:56:25 +01:00
|
|
|
match crate::register::parse_register_macro(args) {
|
|
|
|
Ok((module_expr, export_name, rust_mod_path)) => {
|
|
|
|
let gen_mod_path = crate::register::generated_module_path(&rust_mod_path);
|
2021-03-26 03:59:34 +01:00
|
|
|
|
|
|
|
#[cfg(feature = "metadata")]
|
|
|
|
let param_names = quote! {
|
|
|
|
Some(#gen_mod_path::Token::PARAM_NAMES)
|
|
|
|
};
|
|
|
|
#[cfg(not(feature = "metadata"))]
|
|
|
|
let param_names = quote! { None };
|
|
|
|
|
2021-03-24 02:56:25 +01:00
|
|
|
proc_macro::TokenStream::from(quote! {
|
2021-03-26 11:41:28 +01:00
|
|
|
#module_expr.set_fn(#export_name, FnNamespace::Internal, FnAccess::Public,
|
|
|
|
#param_names,
|
|
|
|
&#gen_mod_path::Token::param_types(),
|
|
|
|
#gen_mod_path::Token().into());
|
2021-03-24 02:56:25 +01:00
|
|
|
})
|
|
|
|
}
|
|
|
|
Err(e) => e.to_compile_error().into(),
|
|
|
|
}
|
2020-11-17 05:09:56 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Macro to register a _plugin function_ into a Rhai `Module` and expose it globally.
|
|
|
|
///
|
|
|
|
/// # Usage
|
|
|
|
///
|
|
|
|
/// ```
|
|
|
|
/// # use rhai::{Engine, EvalAltResult};
|
|
|
|
/// use rhai::plugin::*;
|
|
|
|
///
|
|
|
|
/// #[export_fn]
|
|
|
|
/// fn my_plugin_function(x: i64) -> i64 {
|
|
|
|
/// x * 2
|
|
|
|
/// }
|
|
|
|
///
|
|
|
|
/// # fn main() -> Result<(), Box<EvalAltResult>> {
|
|
|
|
/// let mut engine = Engine::new();
|
|
|
|
///
|
|
|
|
/// let mut module = Module::new();
|
|
|
|
/// set_exported_global_fn!(module, "func", my_plugin_function);
|
|
|
|
///
|
2020-12-23 03:08:43 +01:00
|
|
|
/// engine.register_static_module("test", module.into());
|
2020-11-17 05:09:56 +01:00
|
|
|
///
|
|
|
|
/// assert_eq!(engine.eval::<i64>("func(21)")?, 42);
|
|
|
|
/// # Ok(())
|
|
|
|
/// # }
|
|
|
|
/// ```
|
|
|
|
#[proc_macro]
|
|
|
|
pub fn set_exported_global_fn(args: proc_macro::TokenStream) -> proc_macro::TokenStream {
|
2021-03-24 02:56:25 +01:00
|
|
|
match crate::register::parse_register_macro(args) {
|
|
|
|
Ok((module_expr, export_name, rust_mod_path)) => {
|
|
|
|
let gen_mod_path = crate::register::generated_module_path(&rust_mod_path);
|
2021-03-26 03:59:34 +01:00
|
|
|
|
|
|
|
#[cfg(feature = "metadata")]
|
|
|
|
let param_names = quote! {
|
|
|
|
Some(#gen_mod_path::Token::PARAM_NAMES)
|
|
|
|
};
|
|
|
|
#[cfg(not(feature = "metadata"))]
|
|
|
|
let param_names = quote! { None };
|
|
|
|
|
2021-03-24 02:56:25 +01:00
|
|
|
proc_macro::TokenStream::from(quote! {
|
2021-03-26 11:41:28 +01:00
|
|
|
#module_expr.set_fn(#export_name, FnNamespace::Global, FnAccess::Public,
|
|
|
|
#param_names,
|
|
|
|
&#gen_mod_path::Token::param_types(),
|
|
|
|
#gen_mod_path::Token().into());
|
2021-03-24 02:56:25 +01:00
|
|
|
})
|
|
|
|
}
|
|
|
|
Err(e) => e.to_compile_error().into(),
|
|
|
|
}
|
2020-08-01 18:52:26 +02:00
|
|
|
}
|