Generates builder methods of every field of a struct. It is meant to be used on structs that
implement Default
. There is no separate builder struct generated and no need to call a
build()
method at the end or .unwrap()
.
This crate is used by the crate leptos-use
for the option structs that
can be passed to the various functions.
In your project folder run
sh
cargo add default-struct-builder
It is very easy to use:
```rust use defaultstructbuilder::DefaultBuilder;
pub struct SomeOptions { throttle: f64,
#[builder(into)]
offset: Option<f64>,
#[builder(skip)]
not_included: u32,
} ```
you can then use the struct like this:
```rust let options = SomeOptions::default().offset(4.0);
asserteq!(options.offset, Some(4.0)); asserteq!(options.throttle, 0.0); asserteq!(options.notincluded, 0); ```
The macro is ready to be used on generic structs.
```rust use defaultstructbuilder::DefaultBuilder;
pub struct SomeOptions
All doc comments on fields are directly passed on to their generated setter methods.
The derive macro generates the following code:
``rust
impl SomeOptions {
// setter methods are given that consume
selfand return a new
Self` with the field value changed
pub fn throttle(self, value: f64) -> Self {
Self {
throttle: value,
..self
}
}
// because `into` was specified this method is generic and calls `.into()` when setting the value
pub fn offset<T>(self, value: T) -> Self
where
T: Into<Option<f64>>,
{
Self {
offset: value.into(),
..self
}
}
// no method for field `not_included` because `skip` was specified
} ```
In the case of a generic field the generated method is a bit more complex because by calling the method the type of the type parameter can be different than before.
Let's look at the following example.
```rust use defaultstructbuilder::DefaultBuilder;
pub struct SomeOptions
impl SomeOptions
}
}
```
This generates the setter method below.
```rust
impl
fn main() {
let options = SomeOptions::new() // at first SomeOptions
In cases where you don't want a generic field to be able to change the generic type you
can annotate it with keep_type
.
```rust
struct SomeOptions
this will generate a standard builder method as if T
wasn't generic.
Box
The macro detects if a field is a Box
and generates a builder method that
accepts the inner type (without Box
) and adds the box in the body.
In case it's a Box<dyn Trait>
the builder method will have an argument of type
impl Trait
.
If you want to prevent this auto un-boxing you can use the #[builder(keep_box)]
attribute.
```rust trait Test {}
struct SomeOptions {
thefield: Box
#[builder(keep_box)]
keep: Box<String>,
} ```
This will generate the following code:
```rust
impl SomeOptions {
pub fn thefield(self, value: impl Test + 'static) -> Self {
Self {
thefield: Box::new(value),
..self
}
}
pub fn other_field(self, value: String) -> Self {
Self {
other_field: Box::new(value),
..self
}
}
pub fn keep(self, value: Box<String>) -> Self {
Self {
keep: value,
..self
}
}
} ```
For more general purposes please check out the much more powerful
derive_builder
crate.