Source code

Revision control

Copy as Markdown

Other Tools

# 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.
```rust
# use derive_more::AsMut;
#
#[derive(AsMut)]
struct MyWrapper(String);
```
Generates:
```rust
# struct MyWrapper(String);
impl 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.
```rust
# 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:
```rust
# struct SingleFieldForward(Vec<i32>);
impl<T: ?Sized> AsMut<T> for SingleFieldForward
where
Vec<i32>: 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`.
```rust
# 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).
>
> ```rust
> # 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:
```rust
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.
```rust
# use derive_more::AsMut;
#
#[derive(AsMut)]
struct MyWrapper {
#[as_mut(str)]
name: String,
#[as_mut]
num: i32,
valid: bool,
}
```
Generates:
```rust
# 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:
```rust,compile_fail
# 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`](crate::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.
```rust
# use derive_more::AsMut;
#
#[derive(AsMut)]
struct MyWrapper {
#[as_mut(skip)]
name: String,
#[as_mut(ignore)]
num: i32,
valid: bool,
}
```
Generates:
```rust
# 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.
```rust,compile_fail
# 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.
```rust,compile_fail
# 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.
```rust
# 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.