96 lines
3.3 KiB
Markdown
96 lines
3.3 KiB
Markdown
Calling Rhai Functions from Rust
|
|
===============================
|
|
|
|
{{#include ../links.md}}
|
|
|
|
Rhai also allows working _backwards_ from the other direction - i.e. calling a Rhai-scripted function
|
|
from Rust via `Engine::call_fn`.
|
|
|
|
Functions declared with `private` are hidden and cannot be called from Rust (see also [modules]).
|
|
|
|
```rust
|
|
// Define functions in a script.
|
|
let ast = engine.compile(true,
|
|
r#"
|
|
// a function with two parameters: string and i64
|
|
fn hello(x, y) {
|
|
x.len + y
|
|
}
|
|
|
|
// functions can be overloaded: this one takes only one parameter
|
|
fn hello(x) {
|
|
x * 2
|
|
}
|
|
|
|
// this one takes no parameters
|
|
fn hello() {
|
|
42
|
|
}
|
|
|
|
// this one is private and cannot be called by 'call_fn'
|
|
private hidden() {
|
|
throw "you shouldn't see me!";
|
|
}
|
|
"#)?;
|
|
|
|
// A custom scope can also contain any variables/constants available to the functions
|
|
let mut scope = Scope::new();
|
|
|
|
// Evaluate a function defined in the script, passing arguments into the script as a tuple.
|
|
// Beware, arguments must be of the correct types because Rhai does not have built-in type conversions.
|
|
// If arguments of the wrong types are passed, the Engine will not find the function.
|
|
|
|
let result: i64 = engine.call_fn(&mut scope, &ast, "hello", ( String::from("abc"), 123_i64 ) )?;
|
|
// ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
|
|
// put arguments in a tuple
|
|
|
|
let result: i64 = engine.call_fn(&mut scope, &ast, "hello", (123_i64,) )?;
|
|
// ^^^^^^^^^^ tuple of one
|
|
|
|
let result: i64 = engine.call_fn(&mut scope, &ast, "hello", () )?;
|
|
// ^^ unit = tuple of zero
|
|
|
|
// The following call will return a function-not-found error because
|
|
// 'hidden' is declared with 'private'.
|
|
let result: () = engine.call_fn(&mut scope, &ast, "hidden", ())?;
|
|
```
|
|
|
|
|
|
Low-Level API - `Engine::call_fn_dynamic`
|
|
----------------------------------------
|
|
|
|
For more control, construct all arguments as `Dynamic` values and use `Engine::call_fn_dynamic`, passing it
|
|
anything that implements `AsMut<Dynamic>` (such as a simple array or a `Vec<Dynamic>`):
|
|
|
|
```rust
|
|
let result = engine.call_fn_dynamic(
|
|
&mut scope, // scope to use
|
|
ast.into(), // get 'Module' from 'AST'
|
|
"hello", // function entry-point
|
|
None, // 'this' pointer, if any
|
|
[ String::from("abc").into(), 123_i64.into() ] // arguments
|
|
)?;
|
|
```
|
|
|
|
|
|
Binding the `this` Pointer
|
|
-------------------------
|
|
|
|
`Engine::call_fn_dynamic` can also bind a value to the `this` pointer of a script-defined function.
|
|
|
|
```rust
|
|
let ast = engine.compile("fn action(x) { this += x; }")?;
|
|
|
|
let mut value: Dynamic = 1_i64.into();
|
|
|
|
let result = engine.call_fn_dynamic(
|
|
&mut scope,
|
|
ast.into(),
|
|
"action",
|
|
Some(&mut value), // binding the 'this' pointer
|
|
[ 41_i64.into() ]
|
|
)?;
|
|
|
|
assert_eq!(value.as_int()?, 42);
|
|
```
|