blob: e579dfb96964dd0aeebd3ac55ac6f31b267bb9d1 [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.psi.impl.statements.expressions;
import com.intellij.lang.ASTNode;
import com.intellij.openapi.util.TextRange;
import com.intellij.pom.java.LanguageLevel;
import com.intellij.psi.*;
import com.intellij.psi.impl.source.resolve.ResolveCache;
import com.intellij.psi.util.InheritanceUtil;
import com.intellij.util.ArrayUtil;
import com.intellij.util.Function;
import com.intellij.util.IncorrectOperationException;
import com.intellij.util.NullableFunction;
import com.intellij.util.containers.ContainerUtil;
import org.jetbrains.annotations.NotNull;
import org.jetbrains.annotations.Nullable;
import org.jetbrains.plugins.groovy.lang.psi.GroovyElementVisitor;
import org.jetbrains.plugins.groovy.lang.psi.GroovyPsiElementFactory;
import org.jetbrains.plugins.groovy.lang.psi.api.GroovyResolveResult;
import org.jetbrains.plugins.groovy.lang.psi.api.statements.arguments.GrArgumentList;
import org.jetbrains.plugins.groovy.lang.psi.api.statements.arguments.GrNamedArgument;
import org.jetbrains.plugins.groovy.lang.psi.api.statements.expressions.GrArrayDeclaration;
import org.jetbrains.plugins.groovy.lang.psi.api.statements.expressions.GrExpression;
import org.jetbrains.plugins.groovy.lang.psi.api.statements.expressions.GrNewExpression;
import org.jetbrains.plugins.groovy.lang.psi.api.statements.typedef.GrAnonymousClassDefinition;
import org.jetbrains.plugins.groovy.lang.psi.api.types.GrBuiltInTypeElement;
import org.jetbrains.plugins.groovy.lang.psi.api.types.GrCodeReferenceElement;
import org.jetbrains.plugins.groovy.lang.psi.api.types.GrTypeArgumentList;
import org.jetbrains.plugins.groovy.lang.psi.api.types.GrTypeElement;
import org.jetbrains.plugins.groovy.lang.psi.dataFlow.types.TypeInferenceHelper;
import org.jetbrains.plugins.groovy.lang.psi.impl.GrAnonymousClassType;
import org.jetbrains.plugins.groovy.lang.psi.impl.GrClassReferenceType;
import org.jetbrains.plugins.groovy.lang.psi.impl.GrMapType;
import org.jetbrains.plugins.groovy.lang.psi.impl.PsiImplUtil;
import org.jetbrains.plugins.groovy.lang.psi.impl.statements.expressions.path.GrCallExpressionImpl;
import org.jetbrains.plugins.groovy.lang.psi.util.GrInnerClassConstructorUtil;
import org.jetbrains.plugins.groovy.lang.psi.util.PsiUtil;
import org.jetbrains.plugins.groovy.lang.resolve.ResolveUtil;
import java.util.ArrayList;
import java.util.List;
/**
* @author ilyas
*/
public class GrNewExpressionImpl extends GrCallExpressionImpl implements GrNewExpression {
private static final Function<GrNewExpressionImpl,PsiType> MY_TYPE_CALCULATOR = new NullableFunction<GrNewExpressionImpl, PsiType>() {
@Override
public PsiType fun(GrNewExpressionImpl newExpression) {
final GrAnonymousClassDefinition anonymous = newExpression.getAnonymousClassDefinition();
if (anonymous != null) {
return new GrAnonymousClassType(LanguageLevel.JDK_1_5, anonymous.getResolveScope(),
JavaPsiFacade.getInstance(newExpression.getProject()), anonymous);
}
PsiType type = null;
GrCodeReferenceElement refElement = newExpression.getReferenceElement();
if (refElement != null) {
type = new GrClassReferenceType(refElement);
}
else {
GrBuiltInTypeElement builtin = newExpression.findChildByClass(GrBuiltInTypeElement.class);
if (builtin != null) type = builtin.getType();
}
if (type != null) {
for (int i = 0; i < newExpression.getArrayCount(); i++) {
type = type.createArrayType();
}
return type;
}
return null;
}
};
private static final ResolveCache.PolyVariantResolver<MyFakeReference> RESOLVER = new ResolveCache.PolyVariantResolver<MyFakeReference>() {
@NotNull
@Override
public GroovyResolveResult[] resolve(@NotNull MyFakeReference reference, boolean incompleteCode) {
return reference.getElement().resolveImpl(incompleteCode);
}
};
private final MyFakeReference myFakeReference = new MyFakeReference();
public GrNewExpressionImpl(@NotNull ASTNode node) {
super(node);
}
public String toString() {
return "NEW expression";
}
@Override
public void accept(GroovyElementVisitor visitor) {
visitor.visitNewExpression(this);
}
@Override
public PsiType getType() {
return TypeInferenceHelper.getCurrentContext().getExpressionType(this, MY_TYPE_CALCULATOR);
}
@Override
public GrNamedArgument addNamedArgument(final GrNamedArgument namedArgument) throws IncorrectOperationException {
final GrArgumentList list = getArgumentList();
if (list == null) { //so it is not anonymous class declaration
final GroovyPsiElementFactory factory = GroovyPsiElementFactory.getInstance(getProject());
final GrArgumentList newList = factory.createExpressionArgumentList();
PsiElement last = getLastChild();
assert last != null;
while (last.getPrevSibling() instanceof PsiWhiteSpace || last.getPrevSibling() instanceof PsiErrorElement) {
last = last.getPrevSibling();
assert last != null;
}
ASTNode astNode = last.getNode();
assert astNode != null;
getNode().addChild(newList.getNode(), astNode);
}
return super.addNamedArgument(namedArgument);
}
@Override
public GrArgumentList getArgumentList() {
final GrAnonymousClassDefinition anonymous = getAnonymousClassDefinition();
if (anonymous != null) return anonymous.getArgumentListGroovy();
return super.getArgumentList();
}
@Override
@Nullable
public GrExpression getQualifier() {
final PsiElement[] children = getChildren();
for (PsiElement child : children) {
if (child instanceof GrExpression) return (GrExpression)child;
if (PsiKeyword.NEW.equals(child.getText())) return null;
}
return null;
}
@Override
public GrCodeReferenceElement getReferenceElement() {
final GrAnonymousClassDefinition anonymous = getAnonymousClassDefinition();
if (anonymous != null) return anonymous.getBaseClassReferenceGroovy();
return findChildByClass(GrCodeReferenceElement.class);
}
@Override
public GroovyResolveResult[] multiResolveClass() {
final GrCodeReferenceElement referenceElement = getReferenceElement();
if (referenceElement != null) {
return referenceElement.multiResolve(false);
}
return GroovyResolveResult.EMPTY_ARRAY;
}
@Override
public int getArrayCount() {
final GrArrayDeclaration arrayDeclaration = getArrayDeclaration();
if (arrayDeclaration == null) return 0;
return arrayDeclaration.getArrayCount();
}
@Override
public GrAnonymousClassDefinition getAnonymousClassDefinition() {
return findChildByClass(GrAnonymousClassDefinition.class);
}
@Nullable
@Override
public GrArrayDeclaration getArrayDeclaration() {
return findChildByClass(GrArrayDeclaration.class);
}
@Nullable
@Override
public GrTypeArgumentList getConstructorTypeArguments() {
return findChildByClass(GrTypeArgumentList.class);
}
@Override
@Nullable
public PsiMethod resolveMethod() {
return PsiImplUtil.extractUniqueElement(multiResolve(false));
}
@NotNull
@Override
public GroovyResolveResult advancedResolve() {
return PsiImplUtil.extractUniqueResult(multiResolve(false));
}
@Override
@NotNull
public GroovyResolveResult[] getCallVariants(@Nullable GrExpression upToArgument) {
final GrCodeReferenceElement referenceElement = getReferenceElement();
if (referenceElement == null) return GroovyResolveResult.EMPTY_ARRAY;
List<GroovyResolveResult> result = new ArrayList<GroovyResolveResult>();
for (GroovyResolveResult classResult : referenceElement.multiResolve(false)) {
final PsiElement element = classResult.getElement();
if (element instanceof PsiClass) {
ContainerUtil.addAll(result, ResolveUtil.getAllClassConstructors((PsiClass)element, classResult.getSubstitutor(), null, this));
}
}
return result.toArray(new GroovyResolveResult[result.size()]);
}
@Override
public GrTypeElement getTypeElement() {
return findChildByClass(GrTypeElement.class);
}
@NotNull
@Override
public GroovyResolveResult[] multiResolve(boolean incompleteCode) {
if (getArrayCount() > 0 || getReferenceElement() == null) {
return GroovyResolveResult.EMPTY_ARRAY;
}
return TypeInferenceHelper.getCurrentContext().multiResolve(myFakeReference, incompleteCode, RESOLVER);
}
private GroovyResolveResult[] resolveImpl(boolean incompleteCode) {
GrCodeReferenceElement ref = getReferenceElement();
if (ref == null) return GroovyResolveResult.EMPTY_ARRAY;
GroovyResolveResult classCandidate = inferClassCandidate(ref);
if (classCandidate == null) return GroovyResolveResult.EMPTY_ARRAY;
assert classCandidate.getElement() instanceof PsiClass;
if (incompleteCode) {
return PsiUtil.getConstructorCandidates(ref, classCandidate, null);
}
final GrArgumentList argumentList = getArgumentList();
if (argumentList == null) return GroovyResolveResult.EMPTY_ARRAY;
if (argumentList.getNamedArguments().length > 0 && argumentList.getExpressionArguments().length == 0) {
PsiType mapType = GrMapType.createFromNamedArgs(argumentList, getNamedArguments());
GroovyResolveResult[] constructorResults = PsiUtil.getConstructorCandidates(ref, classCandidate, new PsiType[]{mapType}); //one Map parameter, actually
for (GroovyResolveResult result : constructorResults) {
final PsiElement resolved = result.getElement();
if (resolved instanceof PsiMethod) {
PsiMethod constructor = (PsiMethod)resolved;
final PsiParameter[] parameters = constructor.getParameterList().getParameters();
if (parameters.length == 1 && InheritanceUtil.isInheritor(parameters[0].getType(), CommonClassNames.JAVA_UTIL_MAP)) {
return constructorResults;
}
}
}
final GroovyResolveResult[] emptyConstructors = PsiUtil.getConstructorCandidates(ref, classCandidate, PsiType.EMPTY_ARRAY);
if (emptyConstructors.length > 0) {
return emptyConstructors;
}
}
PsiType[] types = PsiUtil.getArgumentTypes(ref, true);
if (types != null) {
types = GrInnerClassConstructorUtil.addEnclosingArgIfNeeded(types, this, (PsiClass)classCandidate.getElement());
}
return PsiUtil.getConstructorCandidates(ref, classCandidate, types);
}
@Nullable
private static GroovyResolveResult inferClassCandidate(@NotNull GrCodeReferenceElement ref) {
final GroovyResolveResult[] classResults = ref.multiResolve(false);
for (GroovyResolveResult result : classResults) {
if (result.getElement() instanceof PsiClass) {
return result;
}
}
return null;
}
private class MyFakeReference implements PsiPolyVariantReference {
@NotNull
@Override
public ResolveResult[] multiResolve(boolean incompleteCode) {
return GrNewExpressionImpl.this.multiResolve(incompleteCode);
}
@Override
public GrNewExpressionImpl getElement() {
return GrNewExpressionImpl.this;
}
@Override
public TextRange getRangeInElement() {
return TextRange.EMPTY_RANGE;
}
@Nullable
@Override
public PsiElement resolve() {
return resolveMethod();
}
@NotNull
@Override
public String getCanonicalText() {
return "new expression";
}
@Override
public PsiElement handleElementRename(String newElementName) throws IncorrectOperationException {
throw new UnsupportedOperationException("unsupported!");
}
@Override
public PsiElement bindToElement(@NotNull PsiElement element) throws IncorrectOperationException {
throw new UnsupportedOperationException("unsupported!");
}
@Override
public boolean isReferenceTo(PsiElement element) {
return getManager().areElementsEquivalent(element, resolve());
}
@NotNull
@Override
public Object[] getVariants() {
return ArrayUtil.EMPTY_OBJECT_ARRAY;
}
@Override
public boolean isSoft() {
return false;
}
}
}