Files
another-boids-in-rust/vendor/derive_more-impl/doc/as_mut.md

5.9 KiB

What #[derive(AsMut)] generates

Deriving AsMut generates one or more implementations of AsMut, each corresponding to one of the fields of the decorated type. This allows types which contain some T to be passed anywhere that an AsMut<T> is accepted.

Newtypes and Structs with One Field

When AsMut is derived for a newtype or struct with one field, a single implementation is generated to expose the underlying field.

# use derive_more::AsMut;
#
#[derive(AsMut)]
struct MyWrapper(String);

Generates:

# struct MyWrapper(String);
impl derive_more::AsMut<String> for MyWrapper {
    fn as_mut(&mut self) -> &mut String {
        &mut self.0
    }
}

It's also possible to use the #[as_mut(forward)] attribute to forward to the as_mut implementation of the field. So here SingleFieldForward implements all AsMut for all types that Vec<i32> implements AsMut for.

# use derive_more::AsMut;
#
#[derive(AsMut)]
#[as_mut(forward)]
struct SingleFieldForward(Vec<i32>);

let mut item = SingleFieldForward(vec![]);
let _: &mut [i32] = (&mut item).as_mut();

This generates code equivalent to:

# struct SingleFieldForward(Vec<i32>);
impl<T: ?Sized> derive_more::AsMut<T> for SingleFieldForward
where
    Vec<i32>: derive_more::AsMut<T>,
{
    #[inline]
    fn as_mut(&mut self) -> &mut T {
        self.0.as_mut()
    }
}

Specifying concrete types, to derive impls for, is also supported via #[as_mut(<types>)] attribute. These types can include both the type of the field itself, and types for which the field type implements AsMut.

# use derive_more::AsMut;
#
#[derive(AsMut)]
#[as_mut(str, [u8], String)]
struct Types(String);

let mut item = Types("test".to_owned());
let _: &mut str = item.as_mut();
let _: &mut [u8] = item.as_mut();
let _: &mut String = item.as_mut();_

Warning

: When either the field type, or the specified conversion type, contains generic parameters, they are considered as the same type only if are named string-equally, otherwise are assumed as different types even when represent the same type in fact (type aliases, for example).

# use derive_more::AsMut;
#
#[derive(AsMut)]
#[as_mut(i32)] // generates `impl<T: AsMut<i32>> AsMut<i32> for Generic<T>`
struct Generic<T>(T);

#[derive(AsMut)]
#[as_mut(T)] // generates `impl<T> AsMut<T> for Transparent<T>`
struct Transparent<T>(T);

#[derive(AsMut)]
// #[as_mut(RenamedVec<T>)] // not supported, as types are not named string-equally
struct Foo<T>(Vec<T>);
type RenamedVec<T> = Vec<T>;

#[derive(AsMut)]
#[as_mut(RenamedString)] // generates `impl AsMut<RenamedString> for Bar`,
struct Bar(String);      // as generics are not involved
type RenamedString = String;

Generating code like this is not supported:

struct Generic<T>(T);

impl AsMut<i32> for Generic<i32> {
    fn as_mut(&mut self) -> &mut i32 {
        &mut self.0
    }
}

Structs with Multiple Fields

When AsMut is derived for a struct with more than one field (including tuple structs), you must also mark one or more fields with the #[as_mut] attribute. An implementation will be generated for each indicated field.

# use derive_more::AsMut;
#
#[derive(AsMut)]
struct MyWrapper {
    #[as_mut(str)]
    name: String,
    #[as_mut]
    num: i32,
    valid: bool,
}

Generates:

# struct MyWrapper {
#     name: String,
#     num: i32,
#     valid: bool,
# }
impl AsMut<str> for MyWrapper {
    fn as_mut(&mut self) -> &mut String {
        self.name.as_mut()
    }
}

impl AsMut<i32> for MyWrapper {
    fn as_mut(&mut self) -> &mut i32 {
        &mut self.num
    }
}

Tuples (not supported)

Only conversions that use a single field are possible with this derive. Something like this wouldn't work, due to the nature of the AsMut trait itself:

# use derive_more::AsMut
#
#[derive(AsMut)]
#[as_mut((str, [u8]))]
struct MyWrapper(String, Vec<u8>)

If you need to convert into a tuple of references, consider using the Into derive with #[into(ref_mut)].

Skipping

Or vice versa: you can exclude a specific field by using #[as_mut(skip)] (or #[as_mut(ignore)]). Then, implementations will be generated for non-indicated fields.

# use derive_more::AsMut;
#
#[derive(AsMut)]
struct MyWrapper {
    #[as_mut(skip)]
    name: String,
    #[as_mut(ignore)]
    num: i32,
    valid: bool,
}

Generates:

# struct MyWrapper {
#     name: String,
#     num: i32,
#     valid: bool,
# }
impl AsMut<bool> for MyWrapper {
    fn as_mut(&mut self) -> &mut bool {
        &mut self.valid
    }
}

Coherence

Note that AsMut<T> may only be implemented once for any given type T. This means any attempt to mark more than one field of the same type with #[as_mut] will result in a compilation error.

# use derive_more::AsMut;
#
// Error! Conflicting implementations of AsMut<String>
#[derive(AsMut)]
struct MyWrapper {
    #[as_mut]
    str1: String,
    #[as_mut]
    str2: String,
}

Similarly, if some field is annotated with #[as_mut(forward)], no other field can be marked.

# use derive_more::AsMut;
#
// Error! Conflicting implementations of `AsMut<i32>`
// note: upstream crates may add a new impl of trait `AsMut<i32>`
// for type `String` in future versions
#[derive(AsMut)]
struct ForwardWithOther {
    #[as_mut(forward)]
    str: String,
    #[as_mut]
    number: i32,
}

Multiple forwarded impls with different concrete types, however, can be used.

# use derive_more::AsMut;
#
#[derive(AsMut)]
struct Types {
    #[as_mut(str)]
    str: String,
    #[as_mut([u8])]
    vec: Vec<u8>,
}

let mut item = Types {
    str: "test".to_owned(),
    vec: vec![0u8],
};

let _: &mut str = item.as_mut();
let _: &mut [u8] = item.as_mut();

Enums

Deriving AsMut for enums is not supported.