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
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
// Copyright (C) 2023 Benjamin Stürz
//
// This program is free software: you can redistribute it and/or modify
// it under the terms of the GNU General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
// (at your option) any later version.
//
// This program is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
// GNU General Public License for more details.
//
// You should have received a copy of the GNU General Public License
// along with this program.  If not, see <http://www.gnu.org/licenses/>.
#![allow(dead_code)]
use proc_macro::TokenStream;
use proc_macro2::{Ident, Group};
use syn::{Token, LitInt, parse_macro_input, ExprBlock, LitFloat};
use quote::ToTokens;

mod parse;
mod gen;

struct CSS(Vec<Rule>);

struct Rule {
    selector: Selector,
    group: Group,
    declarations: Declarations,
}

enum Selector {
    Simple {
        ident: Option<CSSIdent>,
        class: Option<(Token![.], CSSIdent)>,
        id: Option<(Token![#], CSSIdent)>,
    },
    Colon {
        colon: Token![:],
        ident: CSSIdent,
        arg: Option<(Group, Value)>,
    },
    Sub {
        left: Box<Selector>,
        right: Box<Selector>,
    },
    Gt {
        left: Box<Selector>,
        gt: Token![>],
        right: Box<Selector>,
    },
    Plus {
        left: Box<Selector>,
        plus: Token![+],
        right: Box<Selector>,
    },
    Tilde {
        left: Box<Selector>,
        tilde: Token![~],
        right: Box<Selector>,
    },
    Comma {
        left: Box<Selector>,
        comma: Token![,],
        right: Box<Selector>,
    },
}

struct Declarations(Vec<Declaration>);

struct Declaration {
    ident: CSSIdent,
    colon: Token![:],
    value: Vec<Value>,
    semicolon: Token![;],
}

enum Value {
    ColorCode { pound: Token![#], value: LitInt },
    Ident(CSSIdent),
    Int(LitInt),
    Float(LitFloat),
    IntPerc(LitInt, Token![%]),
    FloatPerc(LitFloat, Token![%]),
    Function { ident: CSSIdent, group: Group, args: Args },
    Block(ExprBlock),
}

struct Args(Vec<Value>);

struct CSSIdent(Vec<CSSIdentSegment>);

enum CSSIdentSegment {
    Ident(Ident),
    Minus(Token![-]),
}

#[proc_macro]
pub fn css(input: TokenStream) -> TokenStream {
    parse_macro_input!(input as CSS).into_token_stream().into()
}

#[proc_macro]
pub fn css_rule(input: TokenStream) -> TokenStream {
    parse_macro_input!(input as Rule).into_token_stream().into()
}

#[proc_macro]
pub fn css_value(input: TokenStream) -> TokenStream {
    parse_macro_input!(input as Value).into_token_stream().into()
}