1
Fork 0
mirror of https://github.com/RGBCube/serenity synced 2025-05-29 14:05:11 +00:00

LibJS: Parse computed MemberExpressions

MemberExpression comes in two flavors:

    computed: a[b]
non-computed: a.b

We can now parse both of the types. :^)
This commit is contained in:
Andreas Kling 2020-03-20 20:51:03 +01:00
parent 86642add2f
commit a3d2e07446
3 changed files with 26 additions and 12 deletions

View file

@ -480,7 +480,7 @@ Value AssignmentExpression::execute(Interpreter& interpreter) const
commit = [&](Value value) { commit = [&](Value value) {
auto object = static_cast<const MemberExpression&>(*m_lhs).object().execute(interpreter).to_object(interpreter.heap()); auto object = static_cast<const MemberExpression&>(*m_lhs).object().execute(interpreter).to_object(interpreter.heap());
ASSERT(object.is_object()); ASSERT(object.is_object());
auto property_name = static_cast<const Identifier&>(static_cast<const MemberExpression&>(*m_lhs).property()).string(); auto property_name = static_cast<const MemberExpression&>(*m_lhs).computed_property_name(interpreter);
object.as_object()->put(property_name, value); object.as_object()->put(property_name, value);
}; };
} else { } else {
@ -638,24 +638,26 @@ Value ObjectExpression::execute(Interpreter& interpreter) const
void MemberExpression::dump(int indent) const void MemberExpression::dump(int indent) const
{ {
ASTNode::dump(indent); print_indent(indent);
printf("%s (computed=%s)\n", class_name(), is_computed() ? "true" : "false");
m_object->dump(indent + 1); m_object->dump(indent + 1);
m_property->dump(indent + 1); m_property->dump(indent + 1);
} }
String MemberExpression::computed_property_name(Interpreter& interpreter) const
{
if (!is_computed()) {
ASSERT(m_property->is_identifier());
return static_cast<const Identifier&>(*m_property).string();
}
return m_property->execute(interpreter).to_string();
}
Value MemberExpression::execute(Interpreter& interpreter) const Value MemberExpression::execute(Interpreter& interpreter) const
{ {
auto object_result = m_object->execute(interpreter).to_object(interpreter.heap()); auto object_result = m_object->execute(interpreter).to_object(interpreter.heap());
ASSERT(object_result.is_object()); ASSERT(object_result.is_object());
return object_result.as_object()->get(computed_property_name(interpreter));
String property_name;
if (m_property->is_identifier()) {
property_name = static_cast<const Identifier&>(*m_property).string();
} else {
ASSERT_NOT_REACHED();
}
return object_result.as_object()->get(property_name);
} }
Value StringLiteral::execute(Interpreter& interpreter) const Value StringLiteral::execute(Interpreter& interpreter) const

View file

@ -601,24 +601,29 @@ private:
class MemberExpression final : public Expression { class MemberExpression final : public Expression {
public: public:
MemberExpression(NonnullRefPtr<Expression> object, NonnullRefPtr<Expression> property) MemberExpression(NonnullRefPtr<Expression> object, NonnullRefPtr<Expression> property, bool computed = false)
: m_object(move(object)) : m_object(move(object))
, m_property(move(property)) , m_property(move(property))
, m_computed(computed)
{ {
} }
virtual Value execute(Interpreter&) const override; virtual Value execute(Interpreter&) const override;
virtual void dump(int indent) const override; virtual void dump(int indent) const override;
bool is_computed() const { return m_computed; }
const Expression& object() const { return *m_object; } const Expression& object() const { return *m_object; }
const Expression& property() const { return *m_property; } const Expression& property() const { return *m_property; }
String computed_property_name(Interpreter&) const;
private: private:
virtual bool is_member_expression() const override { return true; } virtual bool is_member_expression() const override { return true; }
virtual const char* class_name() const override { return "MemberExpression"; } virtual const char* class_name() const override { return "MemberExpression"; }
NonnullRefPtr<Expression> m_object; NonnullRefPtr<Expression> m_object;
NonnullRefPtr<Expression> m_property; NonnullRefPtr<Expression> m_property;
bool m_computed { false };
}; };
} }

View file

@ -369,6 +369,12 @@ NonnullRefPtr<Expression> Parser::parse_secondary_expression(NonnullRefPtr<Expre
case TokenType::Period: case TokenType::Period:
consume(); consume();
return create_ast_node<MemberExpression>(move(lhs), parse_expression(min_precedence, associativity)); return create_ast_node<MemberExpression>(move(lhs), parse_expression(min_precedence, associativity));
case TokenType::BracketOpen: {
consume(TokenType::BracketOpen);
auto expression = create_ast_node<MemberExpression>(move(lhs), parse_expression(min_precedence, associativity), true);
consume(TokenType::BracketClose);
return expression;
}
case TokenType::PlusPlus: case TokenType::PlusPlus:
consume(); consume();
return create_ast_node<UpdateExpression>(UpdateOp::Increment, move(lhs)); return create_ast_node<UpdateExpression>(UpdateOp::Increment, move(lhs));
@ -587,6 +593,7 @@ bool Parser::match_secondary_expression() const
|| type == TokenType::LessThanEquals || type == TokenType::LessThanEquals
|| type == TokenType::ParenOpen || type == TokenType::ParenOpen
|| type == TokenType::Period || type == TokenType::Period
|| type == TokenType::BracketOpen
|| type == TokenType::PlusPlus || type == TokenType::PlusPlus
|| type == TokenType::MinusMinus; || type == TokenType::MinusMinus;
} }