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
use crate::error::Result;
use crate::expression::ast::expression::ExpressionTrait;
use crate::proto::gen::expression::{Expression, MemberExpression};
use std::fmt::{Display, Formatter};

impl MemberExpression {
    pub fn new_computed(object: Expression, property: Expression) -> Self {
        Self {
            object: Some(Box::new(object)),
            property: Some(Box::new(property)),
            computed: true,
        }
    }

    pub fn new_static(object: Expression, property: Expression) -> Result<Self> {
        // Make sure property is an identifier
        property.as_identifier()?;
        Ok(Self {
            object: Some(Box::new(object)),
            property: Some(Box::new(property)),
            computed: false,
        })
    }

    pub fn member_binding_power() -> (f64, f64) {
        // See https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Operators/Operator_Precedence
        //  - left-to-right operators have larger number to the right
        //  - right-to-left have larger number to the left
        (20.0, 20.5)
    }

    pub fn property(&self) -> &Expression {
        self.property.as_ref().unwrap()
    }

    pub fn object(&self) -> &Expression {
        self.object.as_ref().unwrap()
    }
}

impl ExpressionTrait for MemberExpression {
    fn binding_power(&self) -> (f64, f64) {
        Self::member_binding_power()
    }
}

impl Display for MemberExpression {
    fn fmt(&self, f: &mut Formatter<'_>) -> std::fmt::Result {
        let object_rhs_bp = self.object.as_ref().unwrap().binding_power().1;

        // Write object, using binding power to determine whether to wrap it in parenthesis
        if object_rhs_bp < Self::member_binding_power().0 {
            write!(f, "({})", self.object.as_ref().unwrap())?;
        } else {
            write!(f, "{}", self.object.as_ref().unwrap())?;
        }

        // Write property
        if self.computed {
            write!(f, "[{}]", self.property.as_ref().unwrap())
        } else {
            write!(f, ".{}", self.property.as_ref().unwrap())
        }
    }
}