/* * 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 com.intellij.psi.infos; import com.intellij.openapi.projectRoots.JavaSdkVersion; import com.intellij.openapi.projectRoots.JavaVersionService; import com.intellij.psi.*; import com.intellij.psi.util.PsiUtilCore; import org.jetbrains.annotations.NotNull; /** * @author ik,dsl */ public class CandidateInfo implements JavaResolveResult { public static final CandidateInfo[] EMPTY_ARRAY = new CandidateInfo[0]; private final PsiElement myPlace; private final PsiClass myAccessClass; @NotNull private final PsiElement myCandidate; private final boolean myStaticsProblem; protected final PsiSubstitutor mySubstitutor; private final PsiElement myCurrentFileResolveContext; private final boolean myPackagePrefixPackageReference; private Boolean myAccessible; // benign datarace private CandidateInfo(@NotNull PsiElement candidate, @NotNull PsiSubstitutor substitutor, Boolean accessible, boolean staticsProblem, PsiElement currFileContext, PsiElement place, PsiClass accessClass, boolean packagePrefixPackageReference) { myCandidate = candidate; myAccessible = accessible; myStaticsProblem = staticsProblem; mySubstitutor = substitutor; myCurrentFileResolveContext = currFileContext; myAccessClass = accessClass; myPlace = place; myPackagePrefixPackageReference = packagePrefixPackageReference; } public CandidateInfo(@NotNull PsiElement candidate, @NotNull PsiSubstitutor substitutor, boolean accessProblem, boolean staticsProblem, PsiElement currFileContext) { this(candidate, substitutor, !accessProblem, staticsProblem, currFileContext, null, null, false); } public CandidateInfo(@NotNull PsiElement candidate, @NotNull PsiSubstitutor substitutor, boolean accessProblem, boolean staticsProblem){ this(candidate, substitutor, accessProblem, staticsProblem, null); } public CandidateInfo(@NotNull PsiElement candidate, @NotNull PsiSubstitutor substitutor, PsiElement place, PsiClass accessClass, boolean staticsProblem, PsiElement currFileContext){ this(candidate, substitutor, null, staticsProblem, currFileContext, place, accessClass, false); } public CandidateInfo(@NotNull PsiElement candidate, @NotNull PsiSubstitutor substitutor, PsiElement place, boolean staticsProblem){ this(candidate, substitutor, place, null, staticsProblem, null); } public CandidateInfo(@NotNull PsiElement candidate, @NotNull PsiSubstitutor substitutor){ this(candidate, substitutor, null, null, false, null); } public CandidateInfo(@NotNull CandidateInfo candidate, @NotNull PsiSubstitutor newSubstitutor){ this(candidate.myCandidate, newSubstitutor, candidate.myAccessible, candidate.myStaticsProblem, candidate.myCurrentFileResolveContext, candidate.myPlace, null, false); } @Override public boolean isValidResult(){ return isAccessible() && isStaticsScopeCorrect(); } @Override public boolean isPackagePrefixPackageReference() { return myPackagePrefixPackageReference; } @Override @NotNull public PsiElement getElement(){ return myCandidate; } @NotNull @Override public PsiSubstitutor getSubstitutor(){ return mySubstitutor; } @Override public boolean isAccessible() { Boolean Accessible = myAccessible; boolean accessible = true; if(Accessible == null) { if (myPlace != null && myCandidate instanceof PsiMember) { final PsiMember member = (PsiMember)myCandidate; accessible = JavaPsiFacade.getInstance(myPlace.getProject()).getResolveHelper() .isAccessible(member, member.getModifierList(), myPlace, myAccessClass, myCurrentFileResolveContext); if (accessible && member.hasModifierProperty(PsiModifier.PRIVATE) && myPlace instanceof PsiReferenceExpression && JavaVersionService.getInstance().isAtLeast(myPlace, JavaSdkVersion.JDK_1_7)) { accessible = !isAccessedThroughTypeParameterBound(); } } myAccessible = accessible; } else { accessible = Accessible; } return accessible; } private boolean isAccessedThroughTypeParameterBound() { final PsiExpression qualifierExpression = ((PsiReferenceExpression)myPlace).getQualifierExpression(); if (qualifierExpression instanceof PsiMethodCallExpression) { final JavaResolveResult resolveResult = ((PsiMethodCallExpression)qualifierExpression).resolveMethodGenerics(); final PsiElement element = resolveResult.getElement(); if (element instanceof PsiMethod) { final PsiType returnType = ((PsiMethod)element).getReturnType(); final PsiType substitutedReturnType = resolveResult.getSubstitutor().substitute(returnType); if (substitutedReturnType instanceof PsiCapturedWildcardType || substitutedReturnType instanceof PsiWildcardType) { return true; } } } return false; } @Override public boolean isStaticsScopeCorrect(){ return !myStaticsProblem; } @Override public PsiElement getCurrentFileResolveScope() { return myCurrentFileResolveContext; } public boolean equals(final Object o) { if (this == o) return true; if (o == null || getClass() != o.getClass()) return false; final CandidateInfo that = (CandidateInfo)o; if (myPackagePrefixPackageReference != that.myPackagePrefixPackageReference) return false; if (myStaticsProblem != that.myStaticsProblem) return false; if (myAccessClass != null ? !myAccessClass.equals(that.myAccessClass) : that.myAccessClass != null) return false; if (isAccessible() != that.isAccessible()) return false; if (!myCandidate.equals(that.myCandidate)) return false; if (myCurrentFileResolveContext != null ? !myCurrentFileResolveContext.equals(that.myCurrentFileResolveContext) : that.myCurrentFileResolveContext != null) { return false; } if (myPlace != null ? !myPlace.equals(that.myPlace) : that.myPlace != null) return false; return mySubstitutor.equals(that.mySubstitutor); } public int hashCode() { int result = myPlace != null ? myPlace.hashCode() : 0; result = 31 * result + (myAccessClass != null ? myAccessClass.hashCode() : 0); result = 31 * result + myCandidate.hashCode(); result = 31 * result + (isAccessible() ? 1 : 0); result = 31 * result + (myStaticsProblem ? 1 : 0); result = 31 * result + mySubstitutor.hashCode(); result = 31 * result + (myCurrentFileResolveContext != null ? myCurrentFileResolveContext.hashCode() : 0); result = 31 * result + (myPackagePrefixPackageReference ? 1 : 0); return result; } @NotNull public static final JavaResolveResult[] RESOLVE_RESULT_FOR_PACKAGE_PREFIX_PACKAGE = {new CandidateInfo(PsiUtilCore.NULL_PSI_ELEMENT, PsiSubstitutor.EMPTY, Boolean.TRUE, false, null, null, null, true)}; }