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
pub mod orbital;

use proc_macro2::{Delimiter, Group, TokenStream, TokenTree};
use quote::{quote, ToTokens, TokenStreamExt};

use orbital::Orbital;

#[derive(Copy, Clone, Debug, PartialEq, PartialOrd)]
pub struct ElectronData {
    pub shells: [u8; 8],
    pub ionisation_energies: [f64; 30],
    pub electron_configuration: ElectronConfiguration,
    pub electron_affinity: Option<f64>,
    pub electronegativity: Option<f64>,
}

/// Representation of electron configuration using StaticVec
#[derive(Copy, Clone, Debug, PartialEq, PartialOrd)]
pub struct ElectronConfiguration(pub [Orbital; 8]);

impl ToTokens for ElectronData {
    fn to_tokens(&self, tokens: &mut TokenStream) {
        let electron_configuration = &self.electron_configuration;
        let ionisation_energies = self.ionisation_energies;

        let ionisation_energies = slice_to_tokens(ionisation_energies);
        let shells = slice_to_tokens(self.shells);
        let electron_affinity = match self.electron_affinity {
            Some(v) => quote!(Some(#v)),
            None => quote!(None),
        };

        let electronegativity = match self.electronegativity {
            Some(v) => quote!(Some(#v)),
            None => quote!(None),
        };

        let add_tokens = quote! {
            chemistru_elements::data::electron::ElectronData {
                electron_configuration: #electron_configuration,
                ionisation_energies: #ionisation_energies,
                shells: #shells,
                electron_affinity: #electron_affinity,
                electronegativity: #electronegativity,
            }
        };

        let group = TokenTree::Group(Group::new(Delimiter::None, add_tokens));

        tokens.append(group);
    }
}

impl ToTokens for ElectronConfiguration {
    fn to_tokens(&self, tokens: &mut TokenStream) {
        let orbitals = self.0;

        let add_tokens = quote! {
            chemistru_elements::data::electron::ElectronConfiguration([#(#orbitals),*])
        };

        tokens.append(TokenTree::Group(Group::new(Delimiter::None, add_tokens)));
    }
}

fn slice_to_tokens<T: ToTokens, const N: usize>(s: [T; N]) -> TokenStream {
    let item = s.iter();

    quote!([#(#item),*])
}