瀏覽代碼

LibJS: Parse "with" statements :^)

Andreas Kling 4 年之前
父節點
當前提交
d617120499
共有 4 個文件被更改,包括 55 次插入0 次删除
  1. 17 0
      Libraries/LibJS/AST.cpp
  2. 21 0
      Libraries/LibJS/AST.h
  3. 16 0
      Libraries/LibJS/Parser.cpp
  4. 1 0
      Libraries/LibJS/Parser.h

+ 17 - 0
Libraries/LibJS/AST.cpp

@@ -255,6 +255,11 @@ Value IfStatement::execute(Interpreter& interpreter, GlobalObject& global_object
     return js_undefined();
 }
 
+Value WithStatement::execute(Interpreter&, GlobalObject&) const
+{
+    ASSERT_NOT_REACHED();
+}
+
 Value WhileStatement::execute(Interpreter& interpreter, GlobalObject& global_object) const
 {
     Value last_value = js_undefined();
@@ -1142,6 +1147,18 @@ void WhileStatement::dump(int indent) const
     body().dump(indent + 1);
 }
 
+void WithStatement::dump(int indent) const
+{
+    ASTNode::dump(indent);
+
+    print_indent(indent + 1);
+    printf("Object\n");
+    object().dump(indent + 2);
+    print_indent(indent + 1);
+    printf("Body\n");
+    body().dump(indent + 2);
+}
+
 void DoWhileStatement::dump(int indent) const
 {
     ASTNode::dump(indent);

+ 21 - 0
Libraries/LibJS/AST.h

@@ -351,6 +351,27 @@ private:
     NonnullRefPtr<Statement> m_body;
 };
 
+class WithStatement final : public Statement {
+public:
+    WithStatement(NonnullRefPtr<Expression> object, NonnullRefPtr<Statement> body)
+        : m_object(move(object))
+        , m_body(move(body))
+    {
+    }
+
+    const Expression& object() const { return *m_object; }
+    const Statement& body() const { return *m_body; }
+
+    virtual Value execute(Interpreter&, GlobalObject&) const override;
+    virtual void dump(int indent) const override;
+
+private:
+    virtual const char* class_name() const override { return "WithStatement"; }
+
+    NonnullRefPtr<Expression> m_object;
+    NonnullRefPtr<Statement> m_body;
+};
+
 class ForStatement final : public Statement {
 public:
     ForStatement(RefPtr<ASTNode> init, RefPtr<Expression> test, RefPtr<Expression> update, NonnullRefPtr<Statement> body)

+ 16 - 0
Libraries/LibJS/Parser.cpp

@@ -327,6 +327,8 @@ NonnullRefPtr<Statement> Parser::parse_statement()
         return parse_do_while_statement();
     case TokenType::While:
         return parse_while_statement();
+    case TokenType::With:
+        return parse_with_statement();
     case TokenType::Debugger:
         return parse_debugger_statement();
     case TokenType::Semicolon:
@@ -1580,6 +1582,19 @@ NonnullRefPtr<SwitchStatement> Parser::parse_switch_statement()
     return create_ast_node<SwitchStatement>(move(determinant), move(cases));
 }
 
+NonnullRefPtr<WithStatement> Parser::parse_with_statement()
+{
+    consume(TokenType::With);
+    consume(TokenType::ParenOpen);
+
+    auto object = parse_expression(0);
+
+    consume(TokenType::ParenClose);
+
+    auto body = parse_statement();
+    return create_ast_node<WithStatement>(move(object), move(body));
+}
+
 NonnullRefPtr<SwitchCase> Parser::parse_switch_case()
 {
     RefPtr<Expression> test;
@@ -1847,6 +1862,7 @@ bool Parser::match_statement() const
         || type == TokenType::Throw
         || type == TokenType::Try
         || type == TokenType::While
+        || type == TokenType::With
         || type == TokenType::For
         || type == TokenType::CurlyOpen
         || type == TokenType::Switch

+ 1 - 0
Libraries/LibJS/Parser.h

@@ -79,6 +79,7 @@ public:
     NonnullRefPtr<ContinueStatement> parse_continue_statement();
     NonnullRefPtr<DoWhileStatement> parse_do_while_statement();
     NonnullRefPtr<WhileStatement> parse_while_statement();
+    NonnullRefPtr<WithStatement> parse_with_statement();
     NonnullRefPtr<DebuggerStatement> parse_debugger_statement();
     NonnullRefPtr<ConditionalExpression> parse_conditional_expression(NonnullRefPtr<Expression> test);
     NonnullRefPtr<Expression> parse_expression(int min_precedence, Associativity associate = Associativity::Right, Vector<TokenType> forbidden = {});