blob: fb86c2c2551551a642b845b0c23f767e1f2f3b4f [file] [log] [blame]
/*
* Copyright 2000-2014 JetBrains s.r.o.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.jetbrains.plugins.groovy.lang.parser.parsing.statements.expressions.primary;
import com.intellij.lang.PsiBuilder;
import com.intellij.psi.tree.IElementType;
import org.jetbrains.plugins.groovy.GroovyBundle;
import org.jetbrains.plugins.groovy.lang.lexer.GroovyElementType;
import org.jetbrains.plugins.groovy.lang.lexer.GroovyTokenTypes;
import org.jetbrains.plugins.groovy.lang.lexer.TokenSets;
import org.jetbrains.plugins.groovy.lang.parser.GroovyElementTypes;
import org.jetbrains.plugins.groovy.lang.parser.GroovyParser;
import org.jetbrains.plugins.groovy.lang.parser.parsing.statements.blocks.OpenOrClosableBlock;
import org.jetbrains.plugins.groovy.lang.parser.parsing.statements.expressions.AssignmentExpression;
import org.jetbrains.plugins.groovy.lang.parser.parsing.statements.expressions.arguments.ArgumentList;
import org.jetbrains.plugins.groovy.lang.parser.parsing.statements.typeDefinitions.ReferenceElement;
import org.jetbrains.plugins.groovy.lang.parser.parsing.statements.typeDefinitions.TypeDefinition;
import org.jetbrains.plugins.groovy.lang.parser.parsing.types.TypeArguments;
import org.jetbrains.plugins.groovy.lang.parser.parsing.util.ParserUtils;
/**
* @author ilyas
*/
public class PrimaryExpression {
public static IElementType parsePrimaryExpression(PsiBuilder builder, GroovyParser parser) {
return parsePrimaryExpression(builder, parser, false);
}
public static IElementType parsePrimaryExpression(PsiBuilder builder, GroovyParser parser, boolean literalsAsRefExprs) {
final IElementType tokenType = builder.getTokenType();
if (TokenSets.BUILT_IN_TYPES.contains(tokenType)) {
ParserUtils.eatElement(builder, GroovyElementTypes.BUILT_IN_TYPE_EXPRESSION);
return GroovyElementTypes.BUILT_IN_TYPE_EXPRESSION;
}
if (GroovyTokenTypes.kNEW == tokenType) {
PsiBuilder.Marker marker = builder.mark();
final GroovyElementType type = newExprParse(builder, parser);
marker.done(type);
return type;
}
if (GroovyTokenTypes.mIDENT == tokenType || GroovyTokenTypes.kSUPER == tokenType || GroovyTokenTypes.kTHIS == tokenType) {
ParserUtils.eatElement(builder, GroovyElementTypes.REFERENCE_EXPRESSION);
return GroovyElementTypes.REFERENCE_EXPRESSION;
}
if (GroovyTokenTypes.mGSTRING_BEGIN == tokenType) {
final boolean result = CompoundStringExpression.parse(builder, parser, false, GroovyTokenTypes.mGSTRING_BEGIN,
GroovyTokenTypes.mGSTRING_CONTENT, GroovyTokenTypes.mGSTRING_END, null,
GroovyElementTypes.GSTRING, GroovyBundle.message("string.end.expected"));
return result ? GroovyElementTypes.GSTRING : GroovyElementTypes.LITERAL;
}
if (GroovyTokenTypes.mREGEX_BEGIN == tokenType) {
CompoundStringExpression.parse(builder, parser, false, GroovyTokenTypes.mREGEX_BEGIN, GroovyTokenTypes.mREGEX_CONTENT,
GroovyTokenTypes.mREGEX_END, GroovyTokenTypes.mREGEX_LITERAL,
GroovyElementTypes.REGEX, GroovyBundle.message("regex.end.expected"));
return GroovyElementTypes.REGEX;
}
if (GroovyTokenTypes.mDOLLAR_SLASH_REGEX_BEGIN == tokenType) {
CompoundStringExpression
.parse(builder, parser, false, GroovyTokenTypes.mDOLLAR_SLASH_REGEX_BEGIN, GroovyTokenTypes.mDOLLAR_SLASH_REGEX_CONTENT,
GroovyTokenTypes.mDOLLAR_SLASH_REGEX_END,
GroovyTokenTypes.mDOLLAR_SLASH_REGEX_LITERAL,
GroovyElementTypes.REGEX, GroovyBundle.message("dollar.slash.end.expected"));
return GroovyElementTypes.REGEX;
}
if (GroovyTokenTypes.mLBRACK == tokenType) {
return ListOrMapConstructorExpression.parse(builder, parser);
}
if (GroovyTokenTypes.mLPAREN == tokenType) {
return parenthesizedExprParse(builder, parser);
}
if (GroovyTokenTypes.mLCURLY == tokenType) {
return OpenOrClosableBlock.parseClosableBlock(builder, parser);
}
if (tokenType == GroovyTokenTypes.mSTRING_LITERAL || tokenType == GroovyTokenTypes.mGSTRING_LITERAL) {
return ParserUtils.eatElement(builder, literalsAsRefExprs ? GroovyElementTypes.REFERENCE_EXPRESSION : GroovyElementTypes.LITERAL);
}
if (TokenSets.CONSTANTS.contains(tokenType)) {
return ParserUtils.eatElement(builder, GroovyElementTypes.LITERAL);
}
return GroovyElementTypes.WRONGWAY;
}
public static GroovyElementType parenthesizedExprParse(PsiBuilder builder, GroovyParser parser) {
PsiBuilder.Marker marker = builder.mark();
ParserUtils.getToken(builder, GroovyTokenTypes.mLPAREN);
if (!AssignmentExpression.parse(builder, parser)) {
builder.error(GroovyBundle.message("expression.expected"));
}
ParserUtils.getToken(builder, GroovyTokenTypes.mNLS);
ParserUtils.getToken(builder, GroovyTokenTypes.mRPAREN, GroovyBundle.message("rparen.expected"));
marker.done(GroovyElementTypes.PARENTHESIZED_EXPRESSION);
return GroovyElementTypes.PARENTHESIZED_EXPRESSION;
}
public static GroovyElementType newExprParse(PsiBuilder builder, GroovyParser parser) {
ParserUtils.getToken(builder, GroovyTokenTypes.kNEW);
ParserUtils.getToken(builder, GroovyTokenTypes.mNLS);
PsiBuilder.Marker rb = builder.mark();
TypeArguments.parseTypeArguments(builder, false);
if (!TokenSets.BUILT_IN_TYPES.contains(builder.getTokenType()) && GroovyTokenTypes.mIDENT != builder.getTokenType()) {
rb.rollbackTo();
}
else {
rb.drop();
}
PsiBuilder.Marker anonymousMarker = builder.mark();
String name;
if (TokenSets.BUILT_IN_TYPES.contains(builder.getTokenType())) {
ParserUtils.eatElement(builder, GroovyElementTypes.BUILT_IN_TYPE);
name = null;
}
else if (TokenSets.CODE_REFERENCE_ELEMENT_NAME_TOKENS.contains(builder.getTokenType())) {
name = builder.getTokenText();
ReferenceElement.parse(builder, false, true, false, true, true);
}
else {
builder.error(GroovyBundle.message("type.specification.expected"));
anonymousMarker.drop();
return GroovyElementTypes.NEW_EXPRESSION;
}
if (builder.getTokenType() == GroovyTokenTypes.mLPAREN || ParserUtils.lookAhead(builder, GroovyTokenTypes.mNLS,
GroovyTokenTypes.mLPAREN)) {
ParserUtils.getToken(builder, GroovyTokenTypes.mNLS);
methodCallArgsParse(builder, parser);
if (builder.getTokenType() == GroovyTokenTypes.mLCURLY || ParserUtils.lookAhead(builder, GroovyTokenTypes.mNLS,
GroovyTokenTypes.mLCURLY)) {
ParserUtils.getToken(builder, GroovyTokenTypes.mNLS);
TypeDefinition.parseBody(builder, name, parser, false);
anonymousMarker.done(GroovyElementTypes.ANONYMOUS_CLASS_DEFINITION);
return GroovyElementTypes.NEW_EXPRESSION;
}
}
else if (builder.getTokenType() == GroovyTokenTypes.mLBRACK) {
PsiBuilder.Marker forArray = builder.mark();
ParserUtils.getToken(builder, GroovyTokenTypes.mNLS);
ParserUtils.getToken(builder, GroovyTokenTypes.mLBRACK);
if (!AssignmentExpression.parse(builder, parser)) {
builder.error(GroovyBundle.message("expression.expected"));
}
ParserUtils.getToken(builder, GroovyTokenTypes.mNLS);
ParserUtils.getToken(builder, GroovyTokenTypes.mRBRACK, GroovyBundle.message("rbrack.expected"));
while (ParserUtils.getToken(builder, GroovyTokenTypes.mLBRACK)) {
ParserUtils.getToken(builder, GroovyTokenTypes.mNLS);
AssignmentExpression.parse(builder, parser);
ParserUtils.getToken(builder, GroovyTokenTypes.mNLS);
ParserUtils.getToken(builder, GroovyTokenTypes.mRBRACK, GroovyBundle.message("rbrack.expected"));
}
forArray.done(GroovyElementTypes.ARRAY_DECLARATOR);
}
else {
builder.error(GroovyBundle.message("lparen.expected"));
}
anonymousMarker.drop();
return GroovyElementTypes.NEW_EXPRESSION;
}
/**
* Parses method arguments
*
* @param builder
* @return
*/
public static boolean methodCallArgsParse(PsiBuilder builder, GroovyParser parser) {
PsiBuilder.Marker marker = builder.mark();
if (ParserUtils.getToken(builder, GroovyTokenTypes.mLPAREN, GroovyBundle.message("lparen.expected"))) {
ParserUtils.getToken(builder, GroovyTokenTypes.mNLS);
ArgumentList.parseArgumentList(builder, GroovyTokenTypes.mRPAREN, parser);
ParserUtils.getToken(builder, GroovyTokenTypes.mNLS);
ParserUtils.getToken(builder, GroovyTokenTypes.mRPAREN, GroovyBundle.message("rparen.expected"));
}
marker.done(GroovyElementTypes.ARGUMENTS);
return true;
}
}