blob: 1e792d0410d289bdf483dfab28e04626e6489b8f [file] [log] [blame]
// Copyright (c) 2011, the Dart project authors. Please see the AUTHORS file
// for details. All rights reserved. Use of this source code is governed by a
// BSD-style license that can be found in the LICENSE file.
class PartialParser extends Parser {
PartialParser(Listener listener) : super(listener);
Token parseClassBody(Token token) => skipClassBody(token);
Token fullParseClassBody(Token token) => super.parseClassBody(token);
Token parseExpression(Token token) => skipExpression(token);
Token skipExpression(Token token) {
while (true) {
final kind = token.kind;
final value = token.stringValue;
if ((kind === EOF_TOKEN) ||
(value === ';') ||
(value === ',') ||
(value === ']'))
return token;
if (value === '=') {
var nextValue = token.next.stringValue;
if (nextValue === 'const') {
token = token.next;
nextValue = token.next.stringValue;
}
if (nextValue === '{') {
// Handle cases like this:
// class Foo {
// var map;
// Foo() : map = {};
// }
BeginGroupToken begin = token.next;
token = (begin.endGroup !== null) ? begin.endGroup : token;
token = token.next;
continue;
}
if (nextValue === '<') {
// Handle cases like this:
// class Foo {
// var map;
// Foo() : map = <Foo>{};
// }
BeginGroupToken begin = token.next;
token = (begin.endGroup !== null) ? begin.endGroup : token;
token = token.next;
if (token.stringValue === '{') {
begin = token;
token = (begin.endGroup !== null) ? begin.endGroup : token;
token = token.next;
}
continue;
}
}
if (!mayParseFunctionExpressions && value === '{')
return token;
if ((value !== '<') && (token is BeginGroupToken)) {
BeginGroupToken begin = token;
token = (begin.endGroup !== null) ? begin.endGroup : token;
}
token = token.next;
}
}
Token skipClassBody(Token token) {
if (!optional('{', token)) {
return listener.expectedClassBodyToSkip(token);
}
BeginGroupToken beginGroupToken = token;
assert(beginGroupToken.endGroup === null ||
beginGroupToken.endGroup.kind === $CLOSE_CURLY_BRACKET);
return beginGroupToken.endGroup;
}
Token parseFunctionBody(Token token, bool isExpression) {
assert(!isExpression);
String value = token.stringValue;
if (value === ';') {
// No body.
} else if (value === '=>') {
token = parseExpression(token.next);
expectSemicolon(token);
} else {
token = skipBlock(token);
}
// There is no "skipped function body event", so we use
// handleNoFunctionBody instead.
listener.handleNoFunctionBody(token);
return token;
}
Token parseFormalParameters(Token token) => skipFormals(token);
Token skipFormals(Token token) {
listener.beginOptionalFormalParameters(token);
if (!optional('(', token)) {
if (optional(';', token)) {
listener.recoverableError("expected '('", token: token);
return token;
}
return listener.unexpected(token);
}
BeginGroupToken beginGroupToken = token;
Token endToken = beginGroupToken.endGroup;
listener.endFormalParameters(0, token, endToken);
return endToken.next;
}
}