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
use nom::{
    branch::alt,
    bytes::complete::tag,
    character::complete::multispace0,
    combinator::map,
    multi::many1,
    number::complete::float,
    sequence::{
        preceded,
        tuple,
    },
    IResult,
};
use torin::{
    geometry::Length,
    size::{
        DynamicCalculation,
        Size,
    },
};

use crate::{
    Parse,
    ParseError,
};

impl Parse for Size {
    fn parse(value: &str) -> Result<Self, ParseError> {
        if value == "auto" {
            Ok(Size::Inner)
        } else if value == "flex" {
            Ok(Size::Flex(Length::new(1.0)))
        } else if value.contains("flex") {
            Ok(Size::Flex(Length::new(
                value
                    .replace("flex(", "")
                    .replace(')', "")
                    .parse::<f32>()
                    .map_err(|_| ParseError)?,
            )))
        } else if value == "fill" {
            Ok(Size::Fill)
        } else if value == "fill-min" {
            Ok(Size::FillMinimum)
        } else if value.contains("calc") {
            Ok(Size::DynamicCalculations(Box::new(parse_calc(value)?)))
        } else if value.contains('%') {
            Ok(Size::Percentage(Length::new(
                value
                    .replace('%', "")
                    .parse::<f32>()
                    .map_err(|_| ParseError)?,
            )))
        } else if value.contains('v') {
            Ok(Size::RootPercentage(Length::new(
                value
                    .replace('v', "")
                    .parse::<f32>()
                    .map_err(|_| ParseError)?,
            )))
        } else if value.contains('a') {
            Ok(Size::InnerPercentage(Length::new(
                value
                    .replace('a', "")
                    .parse::<f32>()
                    .map_err(|_| ParseError)?,
            )))
        } else {
            Ok(Size::Pixels(Length::new(
                value.parse::<f32>().map_err(|_| ParseError)?,
            )))
        }
    }
}

pub fn parse_calc(mut value: &str) -> Result<Vec<DynamicCalculation>, ParseError> {
    // No need to parse this using nom

    value = value
        .strip_prefix("calc(")
        .ok_or(ParseError)?
        .strip_suffix(')')
        .ok_or(ParseError)?;
    fn inner_parse(value: &str) -> IResult<&str, Vec<DynamicCalculation>> {
        many1(preceded(
            multispace0,
            alt((
                map(tag("+"), |_| DynamicCalculation::Add),
                map(tag("-"), |_| DynamicCalculation::Sub),
                map(tag("*"), |_| DynamicCalculation::Mul),
                map(tag("/"), |_| DynamicCalculation::Div),
                map(tag("("), |_| DynamicCalculation::OpenParenthesis),
                map(tag(")"), |_| DynamicCalculation::ClosedParenthesis),
                map(tuple((float, tag("%"))), |(v, _)| {
                    DynamicCalculation::Percentage(v)
                }),
                map(tuple((float, tag("v"))), |(v, _)| {
                    DynamicCalculation::RootPercentage(v)
                }),
                map(float, DynamicCalculation::Pixels),
            )),
        ))(value)
    }
    let tokens = inner_parse(value).map_err(|_| ParseError)?.1;

    Ok(tokens)
}