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
extern crate proc_macro;
use heck::{CamelCase, ShoutySnakeCase};
use proc_macro::{Ident, Span, TokenStream, TokenTree};

/// Expands an input `ident` as UPPERCASE
#[proc_macro]
pub fn upper(input: TokenStream) -> TokenStream {
    let ident = Ident::new(&input.to_string().to_uppercase(), Span::call_site());
    TokenStream::from(TokenTree::Ident(ident))
}

/// Expands an input `ident` as lowercase
#[proc_macro]
pub fn lower(input: TokenStream) -> TokenStream {
    let ident = Ident::new(&input.to_string().to_lowercase(), Span::call_site());
    TokenStream::from(TokenTree::Ident(ident))
}

/// Expands an input `ident` as snake_case
/// e.g. `HelloWorld` -> `hello_world`
#[proc_macro]
pub fn snake(input: TokenStream) -> TokenStream {
    let raw_ident = &input.to_string();
    let mut s = String::new();
    for (i, c) in raw_ident.chars().enumerate() {
        if c.is_uppercase() || c.is_numeric() {
            if i > 0 {
                s.push('_');
            }
            s.push(c.to_lowercase().to_string().chars().next().unwrap());
        } else {
            s.push(c)
        }
    }
    let ident = Ident::new(&s, Span::call_site());
    TokenStream::from(TokenTree::Ident(ident))
}

/// Expands an input `ident` as CamelCase
/// e.g. `helloWorld` -> `HelloWorld`
#[proc_macro]
pub fn camel(input: TokenStream) -> TokenStream {
    let ident = Ident::new(&input.to_string().to_camel_case(), Span::call_site());
    TokenStream::from(TokenTree::Ident(ident))
}

/// Expands an input `ident` as SHOUTY_CASE
/// e.g. `HelloWorld` -> `HELLO_WORLD`
#[proc_macro]
pub fn shouty(input: TokenStream) -> TokenStream {
    let ident = Ident::new(&input.to_string().to_shouty_snake_case(), Span::call_site());
    TokenStream::from(TokenTree::Ident(ident))
}