mirror of
				https://github.com/rust-lang/rust.git
				synced 2025-10-30 20:44:34 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			208 lines
		
	
	
		
			7.0 KiB
		
	
	
	
		
			Rust
		
	
	
	
	
	
			
		
		
	
	
			208 lines
		
	
	
		
			7.0 KiB
		
	
	
	
		
			Rust
		
	
	
	
	
	
| // Copyright (c) 2015 Anders Kaseorg <andersk@mit.edu>
 | |
| 
 | |
| // Permission is hereby granted, free of charge, to any person obtaining
 | |
| // a copy of this software and associated documentation files (the
 | |
| // “Software”), to deal in the Software without restriction, including
 | |
| // without limitation the rights to use, copy, modify, merge, publish,
 | |
| // distribute, sublicense, and/or sell copies of the Software, and to
 | |
| // permit persons to whom the Software is furnished to do so, subject to
 | |
| // the following conditions:
 | |
| 
 | |
| // The above copyright notice and this permission notice shall be
 | |
| // included in all copies or substantial portions of the Software.
 | |
| 
 | |
| // THE SOFTWARE IS PROVIDED “AS IS”, WITHOUT WARRANTY OF ANY KIND,
 | |
| // EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
 | |
| // MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
 | |
| // IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY
 | |
| // CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,
 | |
| // TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE
 | |
| // SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
 | |
| 
 | |
| //! This crate exports a macro `enum_from_primitive!` that wraps an
 | |
| //! `enum` declaration and automatically adds an implementation of
 | |
| //! `num::FromPrimitive` (reexported here), to allow conversion from
 | |
| //! primitive integers to the enum. It therefore provides an
 | |
| //! alternative to the built-in `#[derive(FromPrimitive)]`, which
 | |
| //! requires the unstable `std::num::FromPrimitive` and is disabled in
 | |
| //! Rust 1.0.
 | |
| //!
 | |
| //! # Example
 | |
| //!
 | |
| //! ```
 | |
| //! #[macro_use] extern crate enum_primitive;
 | |
| //! extern crate num_traits;
 | |
| //! use num_traits::FromPrimitive;
 | |
| //!
 | |
| //! enum_from_primitive! {
 | |
| //! #[derive(Debug, PartialEq)]
 | |
| //! enum FooBar {
 | |
| //!     Foo = 17,
 | |
| //!     Bar = 42,
 | |
| //!     Baz,
 | |
| //! }
 | |
| //! }
 | |
| //!
 | |
| //! fn main() {
 | |
| //!     assert_eq!(FooBar::from_i32(17), Some(FooBar::Foo));
 | |
| //!     assert_eq!(FooBar::from_i32(42), Some(FooBar::Bar));
 | |
| //!     assert_eq!(FooBar::from_i32(43), Some(FooBar::Baz));
 | |
| //!     assert_eq!(FooBar::from_i32(91), None);
 | |
| //! }
 | |
| //! ```
 | |
| 
 | |
| pub mod num_traits {
 | |
|     pub trait FromPrimitive: Sized {
 | |
|         fn from_i64(n: i64) -> Option<Self>;
 | |
|         fn from_u64(n: u64) -> Option<Self>;
 | |
|     }
 | |
| }
 | |
| 
 | |
| pub use std::option::Option;
 | |
| pub use num_traits::FromPrimitive;
 | |
| 
 | |
| /// Helper macro for internal use by `enum_from_primitive!`.
 | |
| #[macro_export]
 | |
| macro_rules! enum_from_primitive_impl_ty {
 | |
|     ($meth:ident, $ty:ty, $name:ident, $( $variant:ident )*) => {
 | |
|         #[allow(non_upper_case_globals, unused)]
 | |
|         fn $meth(n: $ty) -> $crate::Option<Self> {
 | |
|             $( if n == $name::$variant as $ty {
 | |
|                 $crate::Option::Some($name::$variant)
 | |
|             } else )* {
 | |
|                 $crate::Option::None
 | |
|             }
 | |
|         }
 | |
|     };
 | |
| }
 | |
| 
 | |
| /// Helper macro for internal use by `enum_from_primitive!`.
 | |
| #[macro_export]
 | |
| #[macro_use(enum_from_primitive_impl_ty)]
 | |
| macro_rules! enum_from_primitive_impl {
 | |
|     ($name:ident, $( $variant:ident )*) => {
 | |
|         impl $crate::FromPrimitive for $name {
 | |
|             enum_from_primitive_impl_ty! { from_i64, i64, $name, $( $variant )* }
 | |
|             enum_from_primitive_impl_ty! { from_u64, u64, $name, $( $variant )* }
 | |
|         }
 | |
|     };
 | |
| }
 | |
| 
 | |
| /// Wrap this macro around an `enum` declaration to get an
 | |
| /// automatically generated implementation of `num::FromPrimitive`.
 | |
| #[macro_export]
 | |
| #[macro_use(enum_from_primitive_impl)]
 | |
| macro_rules! enum_from_primitive {
 | |
|     (
 | |
|         $( #[$enum_attr:meta] )*
 | |
|         enum $name:ident {
 | |
|             $( $( #[$variant_attr:meta] )* $variant:ident ),+
 | |
|             $( = $discriminator:expr, $( $( #[$variant_two_attr:meta] )* $variant_two:ident ),+ )*
 | |
|         }
 | |
|     ) => {
 | |
|         $( #[$enum_attr] )*
 | |
|         enum $name {
 | |
|             $( $( #[$variant_attr] )* $variant ),+
 | |
|             $( = $discriminator, $( $( #[$variant_two_attr] )* $variant_two ),+ )*
 | |
|         }
 | |
|         enum_from_primitive_impl! { $name, $( $variant )+ $( $( $variant_two )+ )* }
 | |
|     };
 | |
| 
 | |
|     (
 | |
|         $( #[$enum_attr:meta] )*
 | |
|         enum $name:ident {
 | |
|             $( $( $( #[$variant_attr:meta] )* $variant:ident ),+ = $discriminator:expr ),*
 | |
|         }
 | |
|     ) => {
 | |
|         $( #[$enum_attr] )*
 | |
|         enum $name {
 | |
|             $( $( $( #[$variant_attr] )* $variant ),+ = $discriminator ),*
 | |
|         }
 | |
|         enum_from_primitive_impl! { $name, $( $( $variant )+ )* }
 | |
|     };
 | |
| 
 | |
|     (
 | |
|         $( #[$enum_attr:meta] )*
 | |
|         enum $name:ident {
 | |
|             $( $( #[$variant_attr:meta] )* $variant:ident ),+
 | |
|             $( = $discriminator:expr, $( $( #[$variant_two_attr:meta] )* $variant_two:ident ),+ )*,
 | |
|         }
 | |
|     ) => {
 | |
|         $( #[$enum_attr] )*
 | |
|         enum $name {
 | |
|             $( $( #[$variant_attr] )* $variant ),+
 | |
|             $( = $discriminator, $( $( #[$variant_two_attr] )* $variant_two ),+ )*,
 | |
|         }
 | |
|         enum_from_primitive_impl! { $name, $( $variant )+ $( $( $variant_two )+ )* }
 | |
|     };
 | |
| 
 | |
|     (
 | |
|         $( #[$enum_attr:meta] )*
 | |
|         enum $name:ident {
 | |
|             $( $( $( #[$variant_attr:meta] )* $variant:ident ),+ = $discriminator:expr ),+,
 | |
|         }
 | |
|     ) => {
 | |
|         $( #[$enum_attr] )*
 | |
|         enum $name {
 | |
|             $( $( $( #[$variant_attr] )* $variant ),+ = $discriminator ),+,
 | |
|         }
 | |
|         enum_from_primitive_impl! { $name, $( $( $variant )+ )+ }
 | |
|     };
 | |
| 
 | |
|     (
 | |
|         $( #[$enum_attr:meta] )*
 | |
|         pub enum $name:ident {
 | |
|             $( $( #[$variant_attr:meta] )* $variant:ident ),+
 | |
|             $( = $discriminator:expr, $( $( #[$variant_two_attr:meta] )* $variant_two:ident ),+ )*
 | |
|         }
 | |
|     ) => {
 | |
|         $( #[$enum_attr] )*
 | |
|         pub enum $name {
 | |
|             $( $( #[$variant_attr] )* $variant ),+
 | |
|             $( = $discriminator, $( $( #[$variant_two_attr] )* $variant_two ),+ )*
 | |
|         }
 | |
|         enum_from_primitive_impl! { $name, $( $variant )+ $( $( $variant_two )+ )* }
 | |
|     };
 | |
| 
 | |
|     (
 | |
|         $( #[$enum_attr:meta] )*
 | |
|         pub enum $name:ident {
 | |
|             $( $( $( #[$variant_attr:meta] )* $variant:ident ),+ = $discriminator:expr ),*
 | |
|         }
 | |
|     ) => {
 | |
|         $( #[$enum_attr] )*
 | |
|         pub enum $name {
 | |
|             $( $( $( #[$variant_attr] )* $variant ),+ = $discriminator ),*
 | |
|         }
 | |
|         enum_from_primitive_impl! { $name, $( $( $variant )+ )* }
 | |
|     };
 | |
| 
 | |
|     (
 | |
|         $( #[$enum_attr:meta] )*
 | |
|         pub enum $name:ident {
 | |
|             $( $( #[$variant_attr:meta] )* $variant:ident ),+
 | |
|             $( = $discriminator:expr, $( $( #[$variant_two_attr:meta] )* $variant_two:ident ),+ )*,
 | |
|         }
 | |
|     ) => {
 | |
|         $( #[$enum_attr] )*
 | |
|         pub enum $name {
 | |
|             $( $( #[$variant_attr] )* $variant ),+
 | |
|             $( = $discriminator, $( $( #[$variant_two_attr] )* $variant_two ),+ )*,
 | |
|         }
 | |
|         enum_from_primitive_impl! { $name, $( $variant )+ $( $( $variant_two )+ )* }
 | |
|     };
 | |
| 
 | |
|     (
 | |
|         $( #[$enum_attr:meta] )*
 | |
|         pub enum $name:ident {
 | |
|             $( $( $( #[$variant_attr:meta] )* $variant:ident ),+ = $discriminator:expr ),+,
 | |
|         }
 | |
|     ) => {
 | |
|         $( #[$enum_attr] )*
 | |
|         pub enum $name {
 | |
|             $( $( $( #[$variant_attr] )* $variant ),+ = $discriminator ),+,
 | |
|         }
 | |
|         enum_from_primitive_impl! { $name, $( $( $variant )+ )+ }
 | |
|     };
 | |
| }
 | 
