4.3 KiB
Manually Create a Custom Package
{{#include ../../links.md}}
Sometimes specific functionalities are needed, so custom packages can be created.
A custom package is a convenient means to gather up a number of functions for later use.
An [Engine
] only needs to Engine::load_package
the custom package once to gain access
to the entire set of functions within.
Loading a package into an [Engine
] is functionally equivalent to calling Engine::register_fn
etc.
on each of the functions inside the package. But because packages are shared, loading an existing
package is much cheaper than registering all the functions one by one.
The macro rhai::def_package!
can be used to create a new custom package.
Macro Parameters
def_package!(root:package_name:description, variable, block)
-
root
- root namespace, usually"rhai"
. -
package_name
- name of the package, usually ending inPackage
. -
description
- doc comment for the package. -
variable
- a variable name holding a reference to the [module] that is to form the package. -
block
- a code block that initializes the package.
// Import necessary types and traits.
use rhai::{
def_package, // 'def_package!' macro
packages::Package, // 'Package' trait
packages::{ // pre-defined packages
ArithmeticPackage, BasicArrayPackage, BasicMapPackage, LogicPackage
}
};
// Define the package 'MyPackage'.
def_package!(rhai:MyPackage:"My own personal super package", module, {
// Aggregate existing packages simply by calling 'init' on each.
ArithmeticPackage::init(module);
LogicPackage::init(module);
BasicArrayPackage::init(module);
BasicMapPackage::init(module);
// Register additional Rust functions using the standard 'set_fn_XXX' module API.
module.set_fn_1("foo", |s: ImmutableString| {
Ok(foo(s.into_owned()))
});
});
Create a Custom Package from a Plugin Module
By far the easiest way to create a custom module is to call rhai::plugins::combine_with_exported_module!
from within rhai::def_package!
which simply merges in all the functions defined within a [plugin module].
In fact, this exactly is how Rhai's built-in packages, such as BasicMathPackage
, are implemented.
Because of the specific requirements of a [package], all sub-modules are flattened (i.e. all functions defined within sub-modules are pulled up and registered at the top level instead) and so there will not be any sub-modules added to the package.
Variables in the [plugin module] are ignored.
// Import necessary types and traits.
use rhai::{
def_package,
packages::Package,
packages::{ArithmeticPackage, BasicArrayPackage, BasicMapPackage, LogicPackage}
};
use rhai::plugin::*;
// Define plugin module.
#[export_module]
mod my_module {
pub const MY_NUMBER: i64 = 42;
pub fn greet(name: &str) -> String {
format!("hello, {}!", name)
}
pub fn get_num() -> i64 {
42
}
// This is a sub-module, but if using combine_with_exported_module!, it will
// be flattened and all functions registered at the top level.
pub mod my_sub_module {
pub fn get_sub_num() -> i64 {
0
}
}
}
// Define the package 'MyPackage'.
def_package!(rhai:MyPackage:"My own personal super package", module, {
// Aggregate existing packages simply by calling 'init' on each.
ArithmeticPackage::init(module);
LogicPackage::init(module);
BasicArrayPackage::init(module);
BasicMapPackage::init(module);
// Merge all registered functions and constants from the plugin module into the custom package.
//
// The sub-module 'my_sub_module' is flattened and its functions registered at the top level.
//
// The text string name in the middle parameter can be anything and is reserved for future use;
// it is recommended to be an ID string that uniquely identifies the module.
//
// The constant variable, 'MY_NUMBER', is ignored.
//
// This call ends up registering three functions at the top level of the package:
// 1) greet
// 2) get_num
// 3) get_sub_num (pulled up from 'my_sub_module')
//
combine_with_exported_module!(module, "my-functions", my_module));
});