/* * 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()]); } }