⚠ This crate currently requires the beta Rust toolchain. The corresponding stable version 1.40 is planned to be released on 2019-12-19.
Delegating the implementation of traits to enum variants or fields of a struct normally requires a lot of boilerplate code. Ambassador is an attempt to eliminate that boilerplate by deriving the delegating trait implementation via procedural macros.
cargo add ambassador
#[delegatable_trait]
First we need to make our trait available for delegation by adding a #[delegatable_trait]
attribute to it (this also makes your trait delegatable in other crates):
```rust use ambassador::delegatable_trait;
pub trait Shout { fn shout(&self, input: &str) -> String; } ```
#[derive(Delegate)]
& #[delegate(Trait)]
Now we can delegate the implementation of our trait to a struct field/enum variants by adding #[derive(Delegate)]
and its associated attribute #[delegate(Trait)]
to it:
```rust use ambassador::Delegate;
pub struct Cat;
impl Shout for Cat { fn shout(&self, input: &str) -> String { format!("{} - meow!", input) } }
pub struct WrappedCat(Cat); ```
#[delegate(..., target = "foo")]
- target
keyFor structs with multiple fields, the field that should act as delegation target can be specified via the target
key:
```rust
pub struct WrappedCats { foo: Cat, bar: Cat, } ```
This also works for tuple structs with multiple fields, by using their index as a target key:
```rust
pub struct WrappedAnimals(Cat, Dog); ```
#[delegatable_trait_remote]
If you want to make an existing trait that lives outside you crate available for delegation, you can do so by copy-pasting it's signature into your code and using the #[delegatable_trait_remote]
attribute (see full code sample):
```rust use ambassador::delegatabletraitremote; use std::fmt::Display;
trait Display { fn fmt(&self, f: &mut ::std::fmt::Formatter) -> Result<(), ::std::fmt::Error>; }
pub struct WrappedCat(Cat); ```
In this example we have a trait Shout
that is implemented for both Cat
and Dog
.
We now add an Animal
enum and add a delegated trait implementation for Shout
,
that calls the respective implementations of the enum variants:
```rust use ambassador::{delegatable_trait, Delegate};
pub trait Shout { fn shout(&self, input: &str) -> String; }
pub struct Cat;
impl Shout for Cat { fn shout(&self, input: &str) -> String { format!("{} - meow!", input) } }
pub struct Dog;
impl Shout for Dog { fn shout(&self, input: &str) -> String { format!("{} - wuff!", input) } }
pub enum Animal { Cat(Cat), Dog(Dog), }
pub fn main() { let fooanimal = Animal::Cat(Cat); println!("{}", fooanimal.shout("BAR")); } ```
Delegating a trait implementation for a tuple struct (only single-field tuples supported for now), for e.g. a newtype pattern.
```rust use ambassador::{delegatable_trait, Delegate};
pub trait Shout { fn shout(&self, input: &str) -> String; }
pub struct Cat;
impl Shout for Cat { fn shout(&self, input: &str) -> String { format!("{} - meow!", input) } }
pub struct WrappedCat(Cat);
pub fn main() { let fooanimal = WrappedCat(Cat); println!("{}", fooanimal.shout("BAR")); } ```
Delegating a trait implementation for a normal struct
```rust use ambassador::{delegatable_trait, Delegate};
pub trait Shout { fn shout(&self, input: &str) -> String; }
pub struct Cat;
impl Shout for Cat { fn shout(&self, input: &str) -> String { format!("{} - meow!", input) } }
pub struct WrappedCat { inner_cat: Cat, }
pub fn main() { let fooanimal = WrappedCat { innercat: Cat }; println!("{}", foo_animal.shout("BAR")); } ```
Licensed under either of Apache License, Version 2.0 or MIT license at your option.
Unless you explicitly state otherwise, any contribution intentionally submitted for inclusion in this crate by you, as defined in the Apache-2.0 license, shall be dual licensed as above, without any additional terms or conditions.