Amplifying Rust language capabilities: multiple generic trait implementations, type wrappers, derive macros.
This is a part of Rust language amplification library providing required derive macros.
Minimum supported rust compiler version (MSRV): 1.59.0. Rust edition 2021.
Display
] descriptions using other formatting trait:
rust
#[derive(Display, Debug)]
#[display(Debug)]
struct Some { /* ... */ }
Use existing function for displaying descriptions: ```rust
union Int { uint: u32, int: i32 };
impl Int {
pub fn print(&self) -> String {
s!("Integer representation")
}
}
``
Formatting function must return [
String] and take a single
self`
argument (if you need formatting with streamed output, use one of
existing formatting traits as shown in pt. 1).
rust
#[derive(Display)]
#[display("({x}, {y})")]
struct Point { x: u32, y: u32 }
Use of doc comments for descrition representation. In this case doc comments may also contain formatting like in the case 3: ```rust
enum Variants { /// Letter A A,
/// Letter B
B,
/// This comment is ignored
#[display("Letter C")]
C,
/// Letter {_0}
Letter(String)
};
asserteq!(format!("{}", Variants::C), "Letter C"); asserteq!(format!("{}", Variants::Letter(s!("K"))), " Letter K"); ``` You can also mix in this mode with other fors of display tags on a specific options; in this case doc comments are ignored
Advanced use with enums: ```rust
enum Test { Some,
#[display = "OtherName"]
Other,
Named {
x: u8,
},
#[display = "Custom{x}"]
NamedCustom {
x: u8,
},
Unnamed(u16),
// NB: Use `_`-prefixed indexes for tuple values
#[display = "Custom{_0}"]
UnnamedCustom(String),
} ```
Error derive macro works to the full extend only when other derive macros
are used. With #[derive(Display)]
and [display(doc_comments)]
it uses
doc comments for generating error descriptions; with #[derive(From)]
it
may automatically implement transofrations from other error types.
```rust
enum Error { /// I/O operation error Io, /// Math overflow Overflow, /// Zero division with {_0} ZeroDivision(u16), } ```
Implements [From
] trait for the whole entity and/or its separate fields.
Works well with #[derive(Error)]
and, in many cases may require
[Default
] implementation (for details, pls see Examples below)
```rust
// Structure may contain no parameters pub struct IoErrorUnit;
Default
pub struct IoError { details: String,
#[from]
kind: IoErrorUnit,
}
pub enum Error { // You can specify multiple conversions with separate attributes #[from(::std::io::Error)] #[from(IoError)] Io,
#[from]
Format(::std::fmt::Error),
#[from]
WithFields { details: ::std::str::Utf8Error },
MultipleFields {
// ...and you can also covert error type
#[from(IoErrorUnit)]
// rest of parameters must implement `Default`
io: IoError,
details: String,
},
}
pub struct Wrapper(u32, i16); ```
Creates rust new type wrapping existing type. Can be used in sturctures
containing multiple named or unnamed fields; in this case the field you'd
like to wrap should be marked with #[wrap]
attribute; otherwise the first
field is assumed to be the wrapped one.
Use with multiple fileds requires that you do From
and Default
derive
on the main structure.
Supports automatic implementation of the following traits:
* amplify::Wrapper
* AsRef
* AsMut
* Borrow
* BorrowMut
* Deref
* DerefMut
Complete usage of this derive macro is possible only with nightly rust
compiler with trivial_bounds
feature gate set for the crate and nightly
feature set. This will give you an automatic implementation for additional
traits, it they are implemented for the wrapped type:
* Display
* LowerHex
* UpperHex
* LowerExp
* UpperExp
* Octal
* Index
* IndexMut
* Add
* AddAssign
* Sub
* SubAssign
* Mul
* MulAssign
* Div
* DivAssign
Other traits, such as PartialEq
, Eq
, PartialOrd
, Ord
,
Hash
can be implemented using standard #[derive]
attribute in the
same manner as Default
, Debug
and From
```rust use std::marker::PhantomData; use amplify::Wrapper;
struct Wrapped
let w = Wrapped::<(), u8>::default();
asserteq!(w.intoinner(), HashMap::
Creates getter methods matching field names for all fields within a structure (including public and private fields). Getters return reference types.
```
struct One {
a: Vec
let one = One::default();
asserteq!(one.a(), &Vec::
Trait [amplify::AsAny
] allows simple conversion of any type into a generic
"thick" pointer &dyn Any
(see [::core::any::Any
]), that can be later
converted back to the original type with a graceful failing for all other
conversions. AsAny
derive macro allows to implement this trait for
arbitrary time without much hussle:
```
extern crate amplify; use amplify::AsAny;
struct Point { pub x: u64, pub y: u64, }
struct Circle { pub radius: f64, pub center: Point, }
let mut point = Point { x: 1, y: 2 }; let pointptr = point.asany();
let mut circle = Circle { radius: 18., center: point, }; let circleptr = circle.asany();
asserteq!(pointptr.downcastref(), Some(&point));
asserteq!(circleptr.downcastref(), Some(&circle));
asserteq!(circleptr.downcast_ref::
let p = pointptr.downcastref::