rhai/src/fn_args.rs

94 lines
2.7 KiB
Rust
Raw Normal View History

2020-11-20 09:52:28 +01:00
//! Helper module which defines [`FuncArgs`] to make function calling easier.
2021-01-28 09:48:56 +01:00
#![cfg(not(feature = "no_function"))]
#![allow(non_snake_case)]
2020-11-16 16:10:14 +01:00
use crate::dynamic::Variant;
2021-01-28 08:29:55 +01:00
use crate::stdlib::vec::Vec;
2020-11-16 16:10:14 +01:00
use crate::{Dynamic, StaticVec};
2021-01-28 08:29:55 +01:00
/// Trait that parses arguments to a function call.
///
/// Any data type can implement this trait in order to pass arguments to a function call.
pub trait FuncArgs {
2021-01-28 08:29:55 +01:00
/// Parse function call arguments into a container.
///
/// # Example
///
/// ```
/// use rhai::{Engine, Dynamic, FuncArgs, Scope};
///
/// // A struct containing function arguments
/// struct Options {
/// pub foo: bool,
/// pub bar: String,
/// pub baz: i64,
/// }
///
/// impl FuncArgs for Options {
2021-03-24 03:02:50 +01:00
/// fn parse<CONTAINER: Extend<Dynamic>>(self, container: &mut CONTAINER) {
2021-01-28 08:29:55 +01:00
/// container.extend(std::iter::once(self.foo.into()));
/// container.extend(std::iter::once(self.bar.into()));
/// container.extend(std::iter::once(self.baz.into()));
/// }
/// }
///
/// # fn main() -> Result<(), Box<rhai::EvalAltResult>> {
/// let options = Options { foo: false, bar: "world".to_string(), baz: 42 };
///
/// let engine = Engine::new();
/// let mut scope = Scope::new();
///
/// let ast = engine.compile(r#"
/// fn hello(x, y, z) {
/// if x { "hello " + y } else { y + z }
/// }
/// "#)?;
///
/// let result: String = engine.call_fn(&mut scope, &ast, "hello", options)?;
///
/// assert_eq!(result, "world42");
/// # Ok(())
/// # }
/// ```
2021-03-24 03:02:50 +01:00
fn parse<CONTAINER: Extend<Dynamic>>(self, container: &mut CONTAINER);
2021-01-28 08:29:55 +01:00
}
impl<T: Variant + Clone> FuncArgs for Vec<T> {
2021-03-24 03:02:50 +01:00
#[inline(always)]
fn parse<CONTAINER: Extend<Dynamic>>(self, container: &mut CONTAINER) {
2021-01-28 08:29:55 +01:00
container.extend(self.into_iter().map(Variant::into_dynamic));
}
}
2020-11-20 09:52:28 +01:00
/// Macro to implement [`FuncArgs`] for tuples of standard types (each can be
/// converted into a [`Dynamic`]).
macro_rules! impl_args {
($($p:ident),*) => {
impl<$($p: Variant + Clone),*> FuncArgs for ($($p,)*)
{
2021-01-25 04:31:54 +01:00
#[inline(always)]
2021-01-28 08:29:55 +01:00
fn parse<CONTAINER: Extend<Dynamic>>(self, container: &mut CONTAINER) {
let ($($p,)*) = self;
2020-07-26 09:53:22 +02:00
let mut _v = StaticVec::new();
$(_v.push($p.into_dynamic());)*
2021-01-28 08:29:55 +01:00
container.extend(_v.into_iter());
}
}
impl_args!(@pop $($p),*);
};
(@pop) => {
};
(@pop $head:ident) => {
impl_args!();
};
(@pop $head:ident $(, $tail:ident)+) => {
impl_args!($($tail),*);
};
}
impl_args!(A, B, C, D, E, F, G, H, J, K, L, M, N, P, Q, R, S, T, U, V);