blob: 33175565ab375622e02320632595b5237a56efe4 [file] [log] [blame]
/*
* Copyright 2000-2013 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.ide.hierarchy.call;
import com.intellij.ide.hierarchy.HierarchyNodeDescriptor;
import com.intellij.ide.hierarchy.HierarchyTreeStructure;
import com.intellij.openapi.project.Project;
import com.intellij.psi.*;
import com.intellij.psi.search.searches.OverridingMethodsSearch;
import com.intellij.util.ArrayUtil;
import com.intellij.util.containers.HashMap;
import org.jetbrains.annotations.NotNull;
import java.util.ArrayList;
public final class CalleeMethodsTreeStructure extends HierarchyTreeStructure {
private final String myScopeType;
/**
* Should be called in read action
*/
public CalleeMethodsTreeStructure(final Project project, final PsiMethod method, final String scopeType) {
super(project, new CallHierarchyNodeDescriptor(project, null, method, true, false));
myScopeType = scopeType;
}
@NotNull
protected final Object[] buildChildren(@NotNull final HierarchyNodeDescriptor descriptor) {
final PsiMember enclosingElement = ((CallHierarchyNodeDescriptor)descriptor).getEnclosingElement();
if (!(enclosingElement instanceof PsiMethod)) {
return ArrayUtil.EMPTY_OBJECT_ARRAY;
}
final PsiMethod method = (PsiMethod)enclosingElement;
final ArrayList<PsiMethod> methods = new ArrayList<PsiMethod>();
final PsiCodeBlock body = method.getBody();
if (body != null) {
visitor(body, methods);
}
final PsiMethod baseMethod = (PsiMethod)((CallHierarchyNodeDescriptor)getBaseDescriptor()).getTargetElement();
final PsiClass baseClass = baseMethod.getContainingClass();
final HashMap<PsiMethod,CallHierarchyNodeDescriptor> methodToDescriptorMap = new HashMap<PsiMethod, CallHierarchyNodeDescriptor>();
final ArrayList<CallHierarchyNodeDescriptor> result = new ArrayList<CallHierarchyNodeDescriptor>();
for (final PsiMethod calledMethod : methods) {
if (!isInScope(baseClass, calledMethod, myScopeType)) continue;
CallHierarchyNodeDescriptor d = methodToDescriptorMap.get(calledMethod);
if (d == null) {
d = new CallHierarchyNodeDescriptor(myProject, descriptor, calledMethod, false, false);
methodToDescriptorMap.put(calledMethod, d);
result.add(d);
}
else {
d.incrementUsageCount();
}
}
// also add overriding methods as children
final PsiMethod[] overridingMethods = OverridingMethodsSearch.search(method, true).toArray(PsiMethod.EMPTY_ARRAY);
for (final PsiMethod overridingMethod : overridingMethods) {
if (!isInScope(baseClass, overridingMethod, myScopeType)) continue;
final CallHierarchyNodeDescriptor node = new CallHierarchyNodeDescriptor(myProject, descriptor, overridingMethod, false, false);
if (!result.contains(node)) result.add(node);
}
/*
// show method implementations in EJB Class
final PsiMethod[] ejbImplementations = EjbUtil.findEjbImplementations(method, null);
for (int i = 0; i < ejbImplementations.length; i++) {
PsiMethod ejbImplementation = ejbImplementations[i];
result.add(new CallHierarchyNodeDescriptor(myProject, descriptor, ejbImplementation, false));
}
*/
return ArrayUtil.toObjectArray(result);
}
private static void visitor(final PsiElement element, final ArrayList<PsiMethod> methods) {
final PsiElement[] children = element.getChildren();
for (final PsiElement child : children) {
visitor(child, methods);
if (child instanceof PsiMethodCallExpression) {
final PsiMethodCallExpression callExpression = (PsiMethodCallExpression)child;
final PsiReferenceExpression methodExpression = callExpression.getMethodExpression();
final PsiMethod method = (PsiMethod)methodExpression.resolve();
if (method != null) {
methods.add(method);
}
}
else if (child instanceof PsiNewExpression) {
final PsiNewExpression newExpression = (PsiNewExpression)child;
final PsiMethod method = newExpression.resolveConstructor();
if (method != null) {
methods.add(method);
}
}
}
}
}