Struct synstructure::Structure[][src]

pub struct Structure<'a> { /* fields omitted */ }

A wrapper around a syn DeriveInput which provides utilities for creating custom derive trait implementations.

Methods

impl<'a> Structure<'a>
[src]

Create a new Structure with the variants and fields from the passed-in DeriveInput.

Returns a slice of the variants in this Structure.

Returns a mut slice of the variants in this Structure.

Returns a reference to the underlying syn AST node which this Structure was created from.

True if any variants were omitted due to a filter_variants call.

Runs the passed-in function once for each bound field, passing in a BindingInfo. and generating match arms which evaluate the returned tokens.

This method will ignore variants or fields which are ignored through the filter and filter_variant methods.

Example

let di = syn::parse_derive_input(r#"
    enum A {
        B(i32, i32),
        C(u32),
    }
"#).unwrap();
let s = Structure::new(&di);

assert_eq!(
    s.each(|bi| quote!(println!("{:?}", #bi))),

    quote!{
        A::B(ref __binding_0, ref __binding_1,) => {
            { println!("{:?}", __binding_0) }
            { println!("{:?}", __binding_1) }
        }
        A::C(ref __binding_0,) => {
            { println!("{:?}", __binding_0) }
        }
    }
);

Runs the passed-in function once for each bound field, passing in the result of the previous call, and a BindingInfo. generating match arms which evaluate to the resulting tokens.

This method will ignore variants or fields which are ignored through the filter and filter_variant methods.

If a variant has been ignored, it will return the init value.

Example

let di = syn::parse_derive_input(r#"
    enum A {
        B(i32, i32),
        C(u32),
    }
"#).unwrap();
let s = Structure::new(&di);

assert_eq!(
    s.fold(quote!(0), |acc, bi| quote!(#acc + #bi)),

    quote!{
        A::B(ref __binding_0, ref __binding_1,) => {
            0 + __binding_0 + __binding_1
        }
        A::C(ref __binding_0,) => {
            0 + __binding_0
        }
    }
);

Runs the passed-in function once for each variant, passing in a VariantInfo. and generating match arms which evaluate the returned tokens.

This method will ignore variants and not bind fields which are ignored through the filter and filter_variant methods.

Example

let di = syn::parse_derive_input(r#"
    enum A {
        B(i32, i32),
        C(u32),
    }
"#).unwrap();
let s = Structure::new(&di);

assert_eq!(
    s.each_variant(|v| {
        let name = &v.ast().ident;
        quote!(println!(stringify!(#name)))
    }),

    quote!{
        A::B(ref __binding_0, ref __binding_1,) => {
            println!(stringify!(B))
        }
        A::C(ref __binding_0,) => {
            println!(stringify!(C))
        }
    }
);

Filter the bindings created by this Structure object. This has 2 effects:

  • The bindings will no longer appear in match arms generated by methods on this Structure or its subobjects.

  • Impl blocks created with the bound_impl or unsafe_bound_impl method only consider type parameters referenced in the types of non-filtered fields.

Example

let di = syn::parse_derive_input(r#"
    enum A {
        B{ a: i32, b: i32 },
        C{ a: u32 },
    }
"#).unwrap();
let mut s = Structure::new(&di);

s.filter(|bi| { bi.ast().ident == Some("a".into()) });

assert_eq!(
    s.each(|bi| quote!(println!("{:?}", #bi))),

    quote!{
        A::B{ a: ref __binding_0, .. } => {
            { println!("{:?}", __binding_0) }
        }
        A::C{ a: ref __binding_0, } => {
            { println!("{:?}", __binding_0) }
        }
    }
);

Filter the variants matched by this Structure object. This has 2 effects:

  • Match arms destructuring these variants will no longer be generated by methods on this Structure

  • Impl blocks created with the bound_impl or unsafe_bound_impl method only consider type parameters referenced in the types of fields in non-fitered variants.

Example

let di = syn::parse_derive_input(r#"
    enum A {
        B(i32, i32),
        C(u32),
    }
"#).unwrap();
let mut s = Structure::new(&di);

s.filter_variants(|v| v.ast().ident != "B");

assert_eq!(
    s.each(|bi| quote!(println!("{:?}", #bi))),

    quote!{
        A::C(ref __binding_0,) => {
            { println!("{:?}", __binding_0) }
        }
        _ => {}
    }
);

Remove the variant at the given index.

Panics

Panics if the index is out of range.

Updates the BindStyle for each of the passed-in fields by calling the passed-in function for each BindingInfo.

Example

let di = syn::parse_derive_input(r#"
    enum A {
        B(i32, i32),
        C(u32),
    }
"#).unwrap();
let mut s = Structure::new(&di);

s.bind_with(|bi| BindStyle::RefMut);

assert_eq!(
    s.each(|bi| quote!(println!("{:?}", #bi))),

    quote!{
        A::B(ref mut __binding_0, ref mut __binding_1,) => {
            { println!("{:?}", __binding_0) }
            { println!("{:?}", __binding_1) }
        }
        A::C(ref mut __binding_0,) => {
            { println!("{:?}", __binding_0) }
        }
    }
);

Updates the binding name for each fo the passed-in fields by calling the passed-in function for each BindingInfo.

The function will be called with the BindingInfo and its index in the enclosing variant.

The default name is __binding_{} where {} is replaced with an increasing number.

Example

let di = syn::parse_derive_input(r#"
    enum A {
        B{ a: i32, b: i32 },
        C{ a: u32 },
    }
"#).unwrap();
let mut s = Structure::new(&di);

s.binding_name(|bi, i| bi.ident.clone().unwrap());

assert_eq!(
    s.each(|bi| quote!(println!("{:?}", #bi))),

    quote!{
        A::B{ a: ref a, b: ref b, } => {
            { println!("{:?}", a) }
            { println!("{:?}", b) }
        }
        A::C{ a: ref a, } => {
            { println!("{:?}", a) }
        }
    }
);

Returns a list of the type parameters which are refrenced in the types of non-filtered fields / variants.

Caveat

If the struct contains any macros in type position, all parameters will be considered bound. This is because we cannot determine which type parameters are bound by type macros.

Example

let di = syn::parse_derive_input(r#"
    enum A<T, U> {
        B(T, i32),
        C(Option<U>),
    }
"#).unwrap();
let mut s = Structure::new(&di);

s.filter_variants(|v| v.ast().ident != "C");

assert_eq!(
    s.referenced_ty_params(),
    &[&(syn::Ident::from("T"))]
);

Add trait bounds for a trait with the given path for each type parmaeter referenced in the types of non-filtered fields.

Caveat

If the method contains any macros in type position, all parameters will be considered bound. This is because we cannot determine which type parameters are bound by type macros.

Creates an impl block with the required generic type fields filled in to implement the trait path.

This method also adds where clauses to the impl requiring that all referenced type parmaeters implement the trait path.

Caveat

If the method contains any macros in type position, all parameters will be considered bound. This is because we cannot determine which type parameters are bound by type macros.

Panics

Panics if the path string parameter is not a valid TyParamBound.

Example

let di = syn::parse_derive_input(r#"
    enum A<T, U> {
        B(T),
        C(Option<U>),
    }
"#).unwrap();
let mut s = Structure::new(&di);

s.filter_variants(|v| v.ast().ident != "B");

assert_eq!(
    s.bound_impl("::krate::Trait", quote!{
        fn a() {}
    }),
    quote!{
        impl<T, U> ::krate::Trait for A<T, U>
            where Option<U>: ::krate::Trait,
                  U: ::krate::Trait
        {
            fn a() {}
        }
    }
);

This method is the same as bound_impl, except also includes the unsafe keyword for implementing unsafe traits.

This method is like bound_impl but doesn't add the additional bounds to the where clause.

This method is the same as unbound_impl, except also includes the unsafe keyword for implementing unsafe traits.

Trait Implementations

impl<'a> Debug for Structure<'a>
[src]

Formats the value using the given formatter. Read more

impl<'a> Clone for Structure<'a>
[src]

Returns a copy of the value. Read more

Performs copy-assignment from source. Read more

impl<'a> PartialEq for Structure<'a>
[src]

This method tests for self and other values to be equal, and is used by ==. Read more

This method tests for !=.

impl<'a> Eq for Structure<'a>
[src]

impl<'a> Hash for Structure<'a>
[src]

Feeds this value into the given [Hasher]. Read more

Feeds a slice of this type into the given [Hasher]. Read more

Auto Trait Implementations

impl<'a> Send for Structure<'a>

impl<'a> Sync for Structure<'a>