3个版本
0.1.5 | 2020年7月10日 |
---|---|
0.1.4 | 2020年7月7日 |
0.1.3 | 2020年6月16日 |
#1504 在 过程宏
被 2 crate 使用
13KB
205 行
Yui
Yui是Rust的属性读取器。
功能
属性结构
Yui提供了一个通过结构体,StructStruct
,TupleStruct
和NoFieldStruct
创建属性结构的derive宏YuiAttribute
,全部支持。
use yui::YuiAttribute;
#[derive(YuiAttribute)]
struct NoField;
#[derive(YuiAttribute)]
struct Tuple(i32, String);
#[derive(YuiAttribute)]
struct Struct {
int: i32,
float: f64,
bool: bool,
}
类型
- 字符串:
String
在Rust中。 - 布尔:
bool
在Rust中。 - 整数:Rust中的任何整数类型。
- 浮点:Rust中的任何浮点类型。
- 对象:其他属性结构。
- 枚举:定义枚举,记得使用
enum_value=true
选项。 - Vec:T的Vec(T不能是对象、Vec或HashMap)。
- HashMap
:通过 String
Skey映射T的HashMap。如果你想要使字段可选,请在字段类型上使用Option<T>
。
use yui::{YuiEnumValue, YuiAttribute};
#[derive(YuiAttribute)]
struct Bar;
#[derive(YuiEnumValue)]
enum SomeEnum {
A,
B
}
#[derive(YuiAttribute)]
struct Foo {
pub string: String,
pub bool: bool,
pub int: i32, // or other integer types like u32 ...
pub float: f32, // or other float types like f64
pub object: Bar, // any defined object
#[attribute_field(enum_value=true)]
pub enum_field: SomeEnum, // have to add enum_value option
pub list: Vec<i32>, // nested type of vec can`t be Object, Vec or HashMap
pub map: std::collections::HashMap<String, SomeEnum>,
pub optional: Option<i32> // optional field
}
选项
别名
生成的读取器将解析具有给定名称的字段,而不是Rust中字段的名称。#[derive(YuiAttribute)] struct Foo { #[attribute_field(alias = "i32")] pub int32: i32, }
默认值
为该字段设置默认值。如果解析时该值不存在,则将默认值设置为字段,即使字段是可选的。Object
、Vec
或HashMap
字段不能有默认值。#[derive(YuiAttribute)] struct Foo { #[attribute_field(default = 1024)] pub int32: i32 }
枚举值
在枚举类型字段上使用enum_value=true
。
枚举
在枚举上使用YuiEnumValue
derive创建枚举值类型。
use yui::YuiEnumValue;
#[derive(YuiEnumValue)]
enum SomeEnum {
A,
B
}
然后,枚举可以用作字段类型。
variant_value
属性
自定义与变体对应的字符串值(默认为Rust中变体的蛇形命名)。
use yui::YuiEnumValue;
#[derive(YuiEnumValue)]
enum SomeEnum {
#[variant_value("aaa")] // default is 'a'
A,
B
}
使用syn
和quote
解析属性
yui::AttributeStructs<T>
可以用于parse_macro_input!
let attributes = syn::parse_macro_inpit!(input as yui::AttributeStructs<Foo>);
如果您想从 syn::Meta
解析属性,请使用 yui::AttributeStruct::from_meta()
。
属性结构体可以自动转换为令牌。但每个字段的可见性必须是公共的。
use proc_macro::TokenStream;
#[derive(YuiAttribute)]
struct Foo {
#[attribute_field(default = 1024)]
pub int32: i32
}
fn derive_fn(input: TokenStream) -> TokenStream {
let attributes = syn::parse_macro_input!(input as yui::AttributeStructs<Foo>);
let attrs = attributes.attrs;
TokenStream::from(quote::quote! {
fn get_attrs() -> Vec<Foo> {
vec![#(#attrs),*]
}
})
}
生成派生宏
如果您想使用内置的读取生成器,请启用 generate-reader
功能。宏 generate_reader
用于生成派生宏。
use yui::generate_reader;
generated_reader!(
MyDerive,
[StructAttribute1, StructAttribute2],
[FieldAttribute1, FieldAttribute2]
);
该宏将生成一个公共的派生宏,可以用来读取 struct
、enum
或 union
的属性,并通过生成 impl
块来记录元数据。
读取属性
在结构体上使用生成的派生宏,并可以使用宏 has_attribute
和 get_attribute
来处理结构体的属性。此功能需要 nightly rustc,因为需要 proc_macro_hygiene
。
#![feature(proc_macro_hygiene)]
use yui::{get_attribute, has_attribute};
#[derive(MyDerive)]
#[StructAttribute1("some parameters")]
struct Foo {
#[FieldAttribute1("some parameters")]
field: i32
}
fn some_fn() {
assert!(has_attribute!(Foo, StructAttribute1));
assert!(has_attribute!(Foo::field, FieldAttribute1));
let struct_attr1: Option<StructAttribute1> = get_attribute!(Foo, StructAttribute1);
let field_attr1: Option<StructAttribute1> = get_attribute!(Foo::field, StructAttribute1);
}
依赖项
~1.5MB
~35K SLoC