1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
extern crate proc_macro;

use self::proc_macro::TokenStream;
use quote::quote;
use rainbow_shared as util;
use syn::parse::{Parse, ParseStream, Result};
use syn::{parse_macro_input, parse_str, Ident, LitInt, Token};

struct Rgb {
    name: Ident,
    rgb: LitInt,
}

impl Parse for Rgb {
    fn parse(input: ParseStream) -> Result<Self> {
        let name: Ident = input.parse()?;
        input.parse::<Token![,]>()?;
        let rgb: LitInt = input.parse()?;
        Ok(Self { name, rgb })
    }
}

impl Rgb {
    fn name_srgb(&self) -> Ident {
        parse_str::<Ident>(&format!("{}_SRGB", self.name)).expect("suffixed identifier invalid")
    }

    fn packed(&self) -> u32 {
        self.rgb.base10_parse::<u32>().expect("color invalid")
    }

    fn srgb(&self) -> [f32; 4] {
        let bytes = self.packed().to_be_bytes();
        let divisor = u8::max_value() as f32;
        [
            bytes[1] as f32 / divisor,
            bytes[2] as f32 / divisor,
            bytes[3] as f32 / divisor,
            1.0,
        ]
    }
}

struct Rgba {
    rgb: Rgb,
    a: LitInt,
}

impl Parse for Rgba {
    fn parse(input: ParseStream) -> Result<Self> {
        let rgb: Rgb = input.parse()?;
        input.parse::<Token![,]>()?;
        let a: LitInt = input.parse()?;
        Ok(Self { rgb, a })
    }
}

impl Rgba {
    fn srgb(&self) -> [f32; 4] {
        let bytes = self.rgb.packed().to_be_bytes();
        let a = self.a.base10_parse::<u8>().expect("alpha invalid");
        let divisor = u8::max_value() as f32;
        [
            bytes[0] as f32 / divisor,
            bytes[1] as f32 / divisor,
            bytes[2] as f32 / divisor,
            a as f32 / divisor,
        ]
    }
}

fn expand(name: &Ident, name_srgb: Ident, srgb: [f32; 4]) -> TokenStream {
    let lin = util::map_color(srgb, util::srgb_to_linear);
    let (lr, lg, lb, la) = (lin[0], lin[1], lin[2], lin[3]);
    let (sr, sg, sb, sa) = (srgb[0], srgb[1], srgb[2], srgb[3]);
    let expanded = quote! {
        pub const #name: LinRgba = LinRgba::from_f32(#lr, #lg, #lb, #la);
        pub const #name_srgb: SrgbRgba = SrgbRgba::from_f32(#sr, #sg, #sb, #sa);
    };
    TokenStream::from(expanded)
}

#[proc_macro]
pub fn rgb(input: TokenStream) -> TokenStream {
    let input = parse_macro_input!(input as Rgb);
    expand(&input.name, input.name_srgb(), input.srgb())
}

#[proc_macro]
pub fn rgba(input: TokenStream) -> TokenStream {
    let input = parse_macro_input!(input as Rgba);
    expand(&input.rgb.name, input.rgb.name_srgb(), input.srgb())
}