/* * Copyright 2000-2016 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 com.intellij.psi.search.searches; import com.intellij.openapi.application.ReadAction; import com.intellij.openapi.extensions.ExtensionPointName; import com.intellij.openapi.progress.ProcessCanceledException; import com.intellij.openapi.util.Condition; import com.intellij.openapi.util.Conditions; import com.intellij.psi.PsiClass; import com.intellij.psi.PsiFile; import com.intellij.psi.SmartPointerManager; import com.intellij.psi.search.SearchScope; import com.intellij.util.AbstractQuery; import com.intellij.util.FilteredQuery; import com.intellij.util.Query; import com.intellij.util.QueryExecutor; import com.intellij.util.containers.ContainerUtil; import org.jetbrains.annotations.NotNull; /** * @author max */ public class ClassInheritorsSearch extends ExtensibleQueryFactory<PsiClass, ClassInheritorsSearch.SearchParameters> { public static final ExtensionPointName<QueryExecutor> EP_NAME = ExtensionPointName.create("com.intellij.classInheritorsSearch"); public static final ClassInheritorsSearch INSTANCE = new ClassInheritorsSearch(); public static class SearchParameters { @NotNull private final PsiClass myClass; @NotNull private final SearchScope myScope; private final boolean myCheckDeep; private final boolean myCheckInheritance; private final boolean myIncludeAnonymous; @NotNull private final Condition<String> myNameCondition; public SearchParameters(@NotNull final PsiClass aClass, @NotNull SearchScope scope, final boolean checkDeep, final boolean checkInheritance, boolean includeAnonymous) { this(aClass, scope, checkDeep, checkInheritance, includeAnonymous, Conditions.alwaysTrue()); } public SearchParameters(@NotNull final PsiClass aClass, @NotNull SearchScope scope, final boolean checkDeep, final boolean checkInheritance, boolean includeAnonymous, @NotNull final Condition<String> nameCondition) { myClass = aClass; myScope = scope; myCheckDeep = checkDeep; myCheckInheritance = checkInheritance; assert checkInheritance; myIncludeAnonymous = includeAnonymous; myNameCondition = nameCondition; } @NotNull public PsiClass getClassToProcess() { return myClass; } @NotNull public Condition<String> getNameCondition() { return myNameCondition; } public boolean isCheckDeep() { return myCheckDeep; } @NotNull public SearchScope getScope() { return myScope; } public boolean isCheckInheritance() { return myCheckInheritance; } public boolean isIncludeAnonymous() { return myIncludeAnonymous; } @Override public String toString() { return "'"+myClass.getQualifiedName()+ "' scope="+myScope+ (myCheckDeep ? " (deep)" : "") + (myCheckInheritance ? " (check inheritance)":"") + (myIncludeAnonymous ? " (anonymous)":"")+ (myNameCondition == Conditions.<String>alwaysTrue() ? "" : " condition: "+myNameCondition); } @Override public boolean equals(Object o) { if (this == o) return true; if (o == null || getClass() != o.getClass()) return false; SearchParameters that = (SearchParameters)o; if (myCheckDeep != that.myCheckDeep) return false; if (myCheckInheritance != that.myCheckInheritance) return false; if (myIncludeAnonymous != that.myIncludeAnonymous) return false; if (!myClass.equals(that.myClass)) return false; if (!myScope.equals(that.myScope)) return false; return myNameCondition.equals(that.myNameCondition); } @Override public int hashCode() { int result = myClass.hashCode(); result = 31 * result + myScope.hashCode(); result = 31 * result + (myCheckDeep ? 1 : 0); result = 31 * result + (myCheckInheritance ? 1 : 0); result = 31 * result + (myIncludeAnonymous ? 1 : 0); result = 31 * result + myNameCondition.hashCode(); return result; } } private ClassInheritorsSearch() {} @NotNull public static Query<PsiClass> search(@NotNull final PsiClass aClass, @NotNull SearchScope scope, final boolean checkDeep, final boolean checkInheritance, boolean includeAnonymous) { return search(new SearchParameters(aClass, scope, checkDeep, checkInheritance, includeAnonymous)); } @NotNull public static Query<PsiClass> search(@NotNull SearchParameters parameters) { if (!parameters.isCheckDeep()) { Query<PsiClass> directQuery = DirectClassInheritorsSearch.search(parameters.getClassToProcess(), parameters.getScope(), parameters.isIncludeAnonymous()); if (parameters.getNameCondition() != Conditions.<String>alwaysTrue()) { directQuery = new FilteredQuery<>(directQuery, psiClass -> parameters.getNameCondition() .value(ReadAction.compute(psiClass::getName))); } return AbstractQuery.wrapInReadAction(directQuery); } return INSTANCE.createUniqueResultsQuery(parameters, ContainerUtil.canonicalStrategy(), psiClass -> ReadAction.compute(() -> SmartPointerManager.getInstance(psiClass.getProject()).createSmartPsiElementPointer(psiClass))); } /** * @deprecated use {@link #search(PsiClass, SearchScope, boolean)} instead */ @NotNull @Deprecated //todo to be removed in IDEA 17 public static Query<PsiClass> search(@NotNull final PsiClass aClass, @NotNull SearchScope scope, final boolean checkDeep, final boolean checkInheritance) { return search(aClass, scope, checkDeep, checkInheritance, true); } @NotNull public static Query<PsiClass> search(@NotNull final PsiClass aClass, @NotNull SearchScope scope, final boolean checkDeep) { return search(aClass, scope, checkDeep, true, true); } @NotNull public static Query<PsiClass> search(@NotNull final PsiClass aClass, final boolean checkDeep) { return search(aClass, ReadAction.compute(() -> { if (!aClass.isValid()) { throw new ProcessCanceledException(); } PsiFile file = aClass.getContainingFile(); return (file != null ? file : aClass).getUseScope(); }), checkDeep); } @NotNull public static Query<PsiClass> search(@NotNull PsiClass aClass) { return search(aClass, true); } }