Default Struct Builder

Crates.io Docs MIT/Apache 2.0 Build Status

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.

Installation

In your project folder run

sh cargo add default-struct-builder

Usage

It is very easy to use:

```rust use defaultstructbuilder::DefaultBuilder;

[derive(DefaultBuilder, Default)]

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); ```

Generics

The macro is ready to be used on generic structs.

```rust use defaultstructbuilder::DefaultBuilder;

[derive(DefaultBuilder, Default)]

pub struct SomeOptions where T: Default, { some_field: T, } ```

Doc comments

All doc comments on fields are directly passed on to their generated setter methods.

How it works

The derive macro generates the following code:

``rust impl SomeOptions { // setter methods are given that consumeselfand return a newSelf` 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

} ```

Generics

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;

[derive(DefaultBuilder, Default)]

pub struct SomeOptions where T: Default, { somefield: T, otherfield: i16, }

impl SomeOptions { pub fn new() -> Self { Self { somefield: 42.0, otherfield: 0, }
} } ```

This generates the setter method below.

```rust impl SomeOptions where T: Default, { pub fn somefield(self, value: NewT) -> SomeOptions where NewT: Default, { SomeOptions:: { somefield: value, otherfield: self.otherfield, } } }

fn main() { let options = SomeOptions::new() // at first SomeOptions .some_field("string"); // changed to SomeOptions<&str> } ```

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

[derive(DefaultBuilder)]

struct SomeOptions { #[builder(keeptype)] thefield: T, } ```

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 {}

[derive(DefaultBuilder)]

struct SomeOptions { thefield: Box, otherfield: 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
    }   
}

} ```

Related Work

For more general purposes please check out the much more powerful derive_builder crate.