palette_derive 0.7.6

Automatically implement traits from the palette crate.
Documentation
use std::collections::{HashMap, HashSet};

use syn::{spanned::Spanned, Expr, ExprLit};
use syn::{Lit, Meta, MetaNameValue, Type};

use super::{assert_path_meta, FieldAttributeArgumentParser, IdentOrIndex};

#[derive(Default)]
pub struct FieldAttributes {
    pub alpha_property: Option<(IdentOrIndex, Type)>,
    pub zero_size_fields: HashSet<IdentOrIndex>,
    pub type_substitutes: HashMap<IdentOrIndex, Type>,
}

impl FieldAttributeArgumentParser for FieldAttributes {
    fn argument(
        &mut self,
        field_name: &IdentOrIndex,
        ty: &Type,
        argument: Meta,
    ) -> Result<(), Vec<syn::Error>> {
        let argument_name = argument.path().get_ident().map(ToString::to_string);

        match argument_name.as_deref() {
            Some("alpha") => {
                assert_path_meta(&argument).map_err(|error| vec![error])?;
                self.alpha_property = Some((field_name.clone(), ty.clone()));
            }
            Some("unsafe_same_layout_as") => {
                let substitute = if let Meta::NameValue(MetaNameValue {
                    value:
                        Expr::Lit(ExprLit {
                            lit: Lit::Str(string),
                            ..
                        }),
                    ..
                }) = argument
                {
                    string.parse().map_err(|error| vec![error])?
                } else {
                    return Err(vec![::syn::parse::Error::new(
                        argument.span(),
                        "expected `unsafe_same_layout_as = \"SomeType\"`",
                    )]);
                };

                self.type_substitutes.insert(field_name.clone(), substitute);
            }
            Some("unsafe_zero_sized") => {
                assert_path_meta(&argument).map_err(|error| vec![error])?;
                self.zero_size_fields.insert(field_name.clone());
            }
            _ => {
                return Err(vec![::syn::parse::Error::new(
                    argument.span(),
                    "unknown field attribute",
                )]);
            }
        }

        Ok(())
    }
}