blob: 8ccfed10f4b6b5598c183c2c58d2359afeca31c2 [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.template.expressions;
import com.intellij.codeInsight.completion.InsertHandler;
import com.intellij.codeInsight.completion.InsertionContext;
import com.intellij.codeInsight.lookup.LookupElement;
import com.intellij.codeInsight.lookup.LookupElementBuilder;
import com.intellij.codeInsight.lookup.PsiTypeLookupItem;
import com.intellij.codeInsight.template.*;
import com.intellij.codeInsight.template.impl.JavaTemplateUtil;
import com.intellij.openapi.editor.Document;
import com.intellij.psi.*;
import com.intellij.psi.search.GlobalSearchScope;
import com.intellij.util.containers.ContainerUtil;
import org.jetbrains.annotations.NotNull;
import org.jetbrains.plugins.groovy.lang.completion.GroovyCompletionUtil;
import org.jetbrains.plugins.groovy.lang.psi.api.auxiliary.modifiers.GrModifier;
import org.jetbrains.plugins.groovy.lang.psi.expectedTypes.SubtypeConstraint;
import org.jetbrains.plugins.groovy.lang.psi.expectedTypes.SupertypeConstraint;
import org.jetbrains.plugins.groovy.lang.psi.expectedTypes.TypeConstraint;
import org.jetbrains.plugins.groovy.lang.psi.impl.statements.expressions.TypesUtil;
import java.util.List;
/**
* @author ven
*/
public class ChooseTypeExpression extends Expression {
public static final InsertHandler<PsiTypeLookupItem> IMPORT_FIXER = new InsertHandler<PsiTypeLookupItem>() {
@Override
public void handleInsert(InsertionContext context, PsiTypeLookupItem item) {
GroovyCompletionUtil.addImportForItem(context.getFile(), context.getStartOffset(), item);
}
};
protected final SmartTypePointer myTypePointer;
private final List<SmartTypePointer> myItems;
private final boolean myForGroovy;
private final boolean mySelectDef;
public ChooseTypeExpression(@NotNull TypeConstraint[] constraints, PsiManager manager, GlobalSearchScope resolveScope) {
this(constraints, manager, resolveScope, true);
}
public ChooseTypeExpression(TypeConstraint[] constraints,
PsiManager manager,
GlobalSearchScope resolveScope,
boolean forGroovy) {
this(constraints, manager, resolveScope, forGroovy, false);
}
public ChooseTypeExpression(TypeConstraint[] constraints,
PsiManager manager,
GlobalSearchScope resolveScope,
boolean forGroovy,
boolean selectDef) {
myForGroovy = forGroovy;
SmartTypePointerManager typePointerManager = SmartTypePointerManager.getInstance(manager.getProject());
myTypePointer = typePointerManager.createSmartTypePointer(chooseType(constraints, resolveScope, manager));
myItems = createItems(constraints, typePointerManager);
mySelectDef = selectDef;
}
@NotNull
private static List<SmartTypePointer> createItems(@NotNull TypeConstraint[] constraints, @NotNull SmartTypePointerManager typePointerManager) {
List<SmartTypePointer> result = ContainerUtil.newArrayList();
for (TypeConstraint constraint : constraints) {
if (constraint instanceof SubtypeConstraint) {
PsiType type = constraint.getDefaultType();
result.add(typePointerManager.createSmartTypePointer(type));
}
else if (constraint instanceof SupertypeConstraint) {
processSuperTypes(constraint.getType(), result, typePointerManager);
}
}
return result;
}
private static void processSuperTypes(@NotNull PsiType type, @NotNull List<SmartTypePointer> result, @NotNull SmartTypePointerManager typePointerManager) {
result.add(typePointerManager.createSmartTypePointer(type));
PsiType[] superTypes = type.getSuperTypes();
for (PsiType superType : superTypes) {
processSuperTypes(superType, result, typePointerManager);
}
}
@NotNull
private static PsiType chooseType(@NotNull TypeConstraint[] constraints, @NotNull GlobalSearchScope scope, @NotNull PsiManager manager) {
if (constraints.length > 0) return constraints[0].getDefaultType();
return PsiType.getJavaLangObject(manager, scope);
}
@Override
public Result calculateResult(ExpressionContext context) {
PsiDocumentManager.getInstance(context.getProject()).commitAllDocuments();
PsiType type = myTypePointer.getType();
if (type != null) {
if (myForGroovy && (type.equalsToText(CommonClassNames.JAVA_LANG_OBJECT) || mySelectDef)) {
return new TextResult(GrModifier.DEF);
}
type = TypesUtil.unboxPrimitiveTypeWrapper(type);
if (type == null) return null;
final PsiType finalType = type;
return new PsiTypeResult(finalType, context.getProject()) {
@Override
public void handleRecalc(PsiFile psiFile, Document document, int segmentStart, int segmentEnd) {
if (myItems.size() <= 1) {
super.handleRecalc(psiFile, document, segmentStart, segmentEnd);
}
else {
JavaTemplateUtil.updateTypeBindings(getType(), psiFile, document, segmentStart, segmentEnd, true);
}
}
@Override
public String toString() {
return myItems.size() == 1 ? super.toString() : finalType.getPresentableText();
}
};
}
return null;
}
@Override
public Result calculateQuickResult(ExpressionContext context) {
return calculateResult(context);
}
@Override
public LookupElement[] calculateLookupItems(ExpressionContext context) {
List<LookupElement> result = ContainerUtil.newArrayList();
for (SmartTypePointer item : myItems) {
PsiType type = TypesUtil.unboxPrimitiveTypeWrapper(item.getType());
if (type == null) continue;
PsiTypeLookupItem lookupItem = PsiTypeLookupItem.createLookupItem(type, null, PsiTypeLookupItem.isDiamond(type), IMPORT_FIXER);
result.add(lookupItem);
}
if (myForGroovy) {
LookupElementBuilder def = LookupElementBuilder.create(GrModifier.DEF).bold();
if (mySelectDef) {
result.add(0, def);
}
else {
result.add(def);
}
}
return result.toArray(new LookupElement[result.size()]);
}
}